ZenML provides several approaches to configure your pipelines and steps:
Understanding .configure() vs .with_options()
ZenML provides two primary methods to configure pipelines and steps: .configure() and .with_options(). While they accept the same parameters, they behave differently:
.configure(): Modifies the configuration in-place and returns the same object.
.with_options(): Creates a new copy with the applied configuration, leaving the original unchanged.
When to use each:
Use .with_options() in most cases, especially inside pipeline definitions:
@pipelinedefmy_pipeline():# This creates a new configuration just for this instance my_step.with_options(parameters={"param": "value"})()
Use .configure() only when you intentionally want to modify a step globally, and are aware that the change will affect all subsequent invocations of that step.
Approaches to Configuration
Pipeline Configuration with configure
You can configure various aspects of a pipeline using the configure method:
from zenml import pipeline# Assuming MyPipeline is your pipeline function# @pipeline# def MyPipeline():# ...# Create a pipelinemy_pipeline =MyPipeline()# Configure the pipelinemy_pipeline.configure( enable_cache=False, enable_artifact_metadata=True, settings={"docker": {"parent_image": "zenml-io/zenml-cuda:latest" } })# Run the pipelinemy_pipeline()
Runtime Configuration with with_options
You can configure a pipeline at runtime using the with_options method:
Step-Level Configuration
You can configure individual steps with the @step decorator:
Direct Component Assignment
If you have an experiment tracker or step operator in your active stack, you can enable them for specific steps like this:
If you want to make sure a step can only run with a specific experiment tracker/step operator, you can also specify the component names like this:
You can combine both approaches with settings to configure the specific behavior of those components:
This approach allows you to use different components for different steps in your pipeline while also customizing their runtime behavior.
Types of Settings
Settings in ZenML are categorized into three main types:
General settings that can be used on all ZenML pipelines:
DockerSettings for container configuration
ResourceSettings for CPU, memory, and GPU allocation
DeploymentSettings for pipeline deployment configuration - can only be set at the pipeline level
Stack-component-specific settings for configuring behaviors of components in your stack:
These use the pattern <COMPONENT_CATEGORY> or <COMPONENT_CATEGORY>.<COMPONENT_FLAVOR> as keys
Examples include experiment_tracker.mlflow or just step_operator
Configuration Hierarchy
There are a few general rules when it comes to settings and configurations that are applied in multiple places. Generally the following is true:
Configurations in code override configurations made inside of the yaml file
Configurations at the step level override those made at the pipeline level
In case of attributes the dictionaries are merged
Common Setting Types
Resource Settings
Resource settings allow you to specify the CPU, memory, and GPU requirements for your steps.
When both pipeline and step resource settings are specified, they are merged with step settings taking precedence:
Note that ResourceSettings are not always applied by all orchestrators. The ability to enforce resource constraints depends on the specific orchestrator being used. Some orchestrators like Kubernetes fully support these settings, while others may ignore them. In order to learn more, read the individual pages of the orchestrator you are using.
Resource settings also allow you to configure scaling options - including minimum and maximum number of instances, and scaling policy - for your pipeline deployments, when used at the pipeline level:
Note that ResourceSettings are not always applied exactly as specified by all deployers. Some deployers fully support these settings, while others may adjust them automatically to match a set of predefined static values or simply ignore them. In order to learn more, read the individual pages of the deployer you are using.
Docker Settings
Docker settings allow you to customize the containerization process:
For more detailed information on containerization options, see the containerization guide.
Deployment Settings
Deployment settings allow you to customize the web server and ASGI application used to run your pipeline deployments. You can specify a range of options, including custom endpoints, middleware, extensions and even custom files used to serve an entire single-page application alongside your pipeline:
Registration-time vs Runtime Stack Component Settings
Stack components have two types of configuration:
Registration-time configuration: Static settings defined when registering a component
Runtime settings: Dynamic settings that can change between pipeline runs
Even for runtime settings, you can set default values during registration:
Using the Right Key for Stack Component Settings
When specifying stack-component-specific settings, the key follows this pattern:
If you specify just the category (e.g., step_operator), ZenML applies these settings to whatever flavor of component is in your stack. If the settings don't apply to that flavor, they are ignored.
Making Configurations Flexible with Environment Variables
You can make your configurations more flexible by referencing environment variables using the placeholder syntax ${ENV_VARIABLE_NAME}:
In code:
In configuration files:
This allows you to easily adapt your pipelines to different environments without changing code.
Autogenerate a template yaml file
If you want to generate a template yaml file of your specific pipeline, you can do so by using the .write_run_configuration_template() method. This will generate a yaml file with all options commented out. This way you can pick and choose the settings that are relevant to you.
An example of a generated YAML configuration template
When you want to configure your pipeline with a certain stack in mind, you can do so as well: ...write_run_configuration_template(stack=<Insert_stack_here>)
# Configure specific step parameters
my_pipeline.with_options(steps={"trainer": {"parameters": {"learning_rate": 0.01}}})()
# Or using a YAML configuration file
my_pipeline.with_options(config_file="path_to_yaml_file")()
import tensorflow as tf
from zenml import step
@step(
settings={
# Custom materializer for handling output serialization
"output_materializers": {
"output": "zenml.materializers.tensorflow_materializer.TensorflowModelMaterializer"
},
# Step-specific experiment tracker settings
"experiment_tracker.mlflow": {
"experiment_name": "custom_experiment"
}
}
)
def train_model() -> tf.keras.Model:
model = build_and_train_model()
return model
from zenml import step
@step(experiment_tracker=True, step_operator=True)
def train_model():
# This step will use the experiment tracker and step operator of the active stack
...
from zenml import step
@step(experiment_tracker="mlflow_tracker", step_operator="vertex_ai")
def train_model():
# This step will use MLflow for tracking and run on Vertex AI
...
from zenml import step
@step(step_operator=True, settings={"step_operator": {"estimator_args": {"instance_type": "m7g.medium"}}})
def my_step():
# This step will use the step operator of the active stack with custom instance type
...
# Alternatively, using the step operator name and appropriate settings class:
@step(step_operator="nameofstepoperator", settings={"step_operator": SagemakerStepOperatorSettings(instance_type="m7g.medium")})
def my_step():
# Same configuration using the settings class
...
from zenml import pipeline, step
from zenml.config import ResourceSettings
@step
def load_data(parameter: int) -> dict:
...
@step(settings={"resources": ResourceSettings(gpu_count=1, memory="2GB")})
def train_model(data: dict) -> None:
...
@pipeline(settings={"resources": ResourceSettings(cpu_count=2, memory="1GB")})
def simple_ml_pipeline(parameter: int):
...
# ZenMl merges the two configurations and uses the step configuration to override
# values defined on the pipeline level
train_model.configuration.settings["resources"]
# -> cpu_count: 2, gpu_count=1, memory="2GB"
simple_ml_pipeline.configuration.settings["resources"]
# -> cpu_count: 2, memory="1GB"
# Example: Setting a fixed tracking URL for MLflow
zenml experiment-tracker register mlflow_tracker --flavor=mlflow --tracking_url=http://localhost:5000
# Example: Setting experiment name that changes for each run
@step(settings={"experiment_tracker.mlflow": {"experiment_name": "custom_experiment"}})
def my_step():
...
# Setting a default value for "nested" setting
zenml experiment-tracker register <n> --flavor=mlflow --nested=True
# Using just the component category
@step(settings={"step_operator": {"estimator_args": {"instance_type": "m7g.medium"}}})
# Or using the component category and flavor
@step(settings={"experiment_tracker.mlflow": {"experiment_name": "custom_experiment"}})