- check runtime parameter values against the schema (from the UI or the SDK)
- build a user-friendly form in the Prefect UI
- easily reuse parameter types in similar workflows
Motivation
Let’s say you have a workflow that triggers a marketing email blast which looks like:
- there are only a few valid values for
mailing_lists
- the
subject
must not exceed 30 characters - no more than 5
attachments
are allowed
- you have to spin up the infrastructure associated with your flow in order to check the constraints, which is wasteful if it turns out that bad parameters were provided
- this might get duplicative, especially if you have similarly constrained parameters for different workflows
pydantic
to build a convenient, self-documenting, and reusable flow signature that the Prefect UI can build a better form from.
Building a convenient flow signature
Let’s address the constraints onmailing_lists
, subject
, and attachments
.
Using Literal
to restrict valid values
there are only a few valid values for mailing_lists
Say our valid mailing lists are: ["newsletter", "customers", "beta-testers"]
We can define a Literal
to specify the valid values for the mailing_lists
parameter.
You can use an
Enum
to achieve the same effect.Using a BaseModel
subclass to group and constrain parameters
Both the subject
and attachments
parameters have constraints that we want to enforce.
the subject
must not exceed 30 characters
the attachments
must not exceed 5 items
Additionally, the subject
, body
, and attachments
parameters are all related to the same thing: the content of the email.
We can define a BaseModel
subclass to group these parameters together and apply these constraints.
pydantic.Field
accepts a description
kwarg that is displayed in the form above the field input.
- pass
title
toField
to override the field name in the form - define a docstring for
EmailContent
to add a description to this group of parameters in the form
Rewriting the flow signature
Now that we have defined theMailingList
and EmailContent
types, we can use them in our flow signature:

mailing_lists
parameter renders as a multi-select dropdown that only allows the Literal
values from our MailingList
type.

EmailContent
fields will be enforced before the run is submitted.

Full example code so far
Full example code so far
Using json_schema_extra
to order fields in the form
By default, your flow parameters are rendered in the order defined by your @flow
function signature.
Within a given BaseModel
subclass, parameters are rendered in the following order:
- parameters with a
default
value are rendered first, alphabetically - parameters without a
default
value are rendered next, alphabetically
BaseModel
subclass by passing json_schema_extra
to the Field
constructor with a position
key.
Taking our EmailContent
model from the previous example, let’s enforce that subject
should be displayed first, then body
, then attachments
.

Recap
We have now embedded the constraints on our parameters in the types that describe our flow signature, which means:- the UI can enforce these constraints before the run is submitted - less wasted infra cycles
- workflow inputs are self-documenting, both in the UI and in the code defining your workflow
- the types used in this signature can be easily reused for other similar workflows
Debugging and related resources
As you craft a schema for your flow signature, you may want to inspect the raw OpenAPI schema thatpydantic
generates, as it is what the Prefect UI uses to build the form.
Call model_json_schema()
on your BaseModel
subclass to inspect the raw schema.
pydantic
, see their documentation on models and types.