Feature Overview
The dynamic form configuration feature allows you to declaratively configure a friendly interactive experience for resources such as Pipeline and Task. Without writing frontend code, simply by adding style.tekton.dev/descriptors configuration in the resource's annotations, you can achieve:
- Dynamic form generation: Automatically generate interactive forms for resource parameters, providing a better experience when orchestrating and executing Pipelines and Tasks.
- Rich component support: Provides various form components such as text boxes, selectors, switches, YAML editors, etc.
- Field validation: Built-in validation rules to ensure user input meets requirements.
- Dynamic data fetching: Supports dynamically loading option data through APIs.
Use Cases
- Pipeline orchestration/runtime: Allow users to fill in parameters such as namespace and secrets through friendly forms when orchestrating or triggering
Pipeline
- Task reuse: Provide default values, dropdown options, and validation for common parameters at the
Task or Pipeline template level
- Multi-environment selection: Dynamically query available environments or configurations from APIs and use them as selection options
Quick Start
Basic Configuration Structure
All configurations are defined through the style.tekton.dev/descriptors field in annotations:
metadata:
annotations:
style.tekton.dev/descriptors: |
- path: params.parameter-name
x-descriptors:
- configuration-item1
- configuration-item2
Configuration Description
path: Specifies the parameter path in the format params.parameter-name
x-descriptors: Configuration item array, each configuration item starts with urn:alm:descriptor:
Simple Example
Configure a required text input box for a Task parameter:
apiVersion: tekton.dev/v1
kind: Task
metadata:
name: simple-task
annotations:
style.tekton.dev/descriptors: |
- path: params.gitURL
x-descriptors:
- urn:alm:descriptor:label:zh:GitRepo
- urn:alm:descriptor:com.tectonic.ui:text
- urn:alm:descriptor:placeholder:zh:Git URL
- urn:alm:descriptor:com.tectonic.ui:validation:required
spec:
params:
- name: gitURL
type: string
steps:
- name: clone
image: alpine/git
script: |
git clone $(params.gitURL)
# Label for the form
- urn:alm:descriptor:label:en:English Name
- urn:alm:descriptor:label:zh:Chinese name
# Description for the form
- urn:alm:descriptor:description:en:English Description
- urn:alm:descriptor:description:zh:Chinese Description
Placeholder Hint
# Placeholder for the form
- urn:alm:descriptor:placeholder:en:This is a Placeholder
- urn:alm:descriptor:placeholder:zh:This is a Placeholder
# Help tooltip for the form
- urn:alm:descriptor:tooltip:en:This is a Help Tip
- urn:alm:descriptor:tooltip:zh:This is a Help Tip
# Whether the form is disabled
- urn:alm:descriptor:com.tectonic.ui:disabled
Supported Field Types
Text Input
# value: string
- urn:alm:descriptor:com.tectonic.ui:text
Multi-line Text Input
# value: string
- urn:alm:descriptor:com.tectonic.ui:textarea
# value: array
- urn:alm:descriptor:com.tectonic.ui:tagsInput
Radio
# Declare a radio option with value "option-a"
- urn:alm:descriptor:com.tectonic.ui:radio:option-a
# Set the English display label of option-a to "Option A"
- urn:alm:descriptor:com.tectonic.ui:radio:option-a:en:Option A
# Set the Chinese display label of option-a to "Option A(zh)"
- urn:alm:descriptor:com.tectonic.ui:radio:option-a:zh:Option A(zh)
# Declare a radio option with value "option-b"
- urn:alm:descriptor:com.tectonic.ui:radio:option-b
# Set the English display label of option-b to "Option B"
- urn:alm:descriptor:com.tectonic.ui:radio:option-b:en:Option B
# Set the Chinese display label of option-b to "Option B(zh)"
- urn:alm:descriptor:com.tectonic.ui:radio:option-b:zh:Option B(zh)
# Set the default selected option to "option-a"
- urn:alm:descriptor:com.tectonic.default:option-a
Switch
# value: boolean
- urn:alm:descriptor:com.tectonic.ui:booleanSwitch
# Map value to specified string
# When switch is on, value is True
- urn:alm:descriptor:props:booleanSwitch:true:True
# When switch is off, value is False
- urn:alm:descriptor:props:booleanSwitch:false:False
YAML Editor
# value: string
- urn:alm:descriptor:com.tectonic.ui:yaml
Select Box
Basic Content Configuration
# value: string
- urn:alm:descriptor:com.tectonic.ui:select
# Options
- urn:alm:descriptor:props:select:options:valueA
- urn:alm:descriptor:props:select:options:valueB
# Set the select box to multiple selection mode
# value: array
- urn:alm:descriptor:props:select:multiple
# Manually entered content can be used as selection items
- urn:alm:descriptor:props:select:allowCreate
# Content can be cleared
- urn:alm:descriptor:props:select:clearable
API Dynamic Options Configuration
When form controls need to dynamically load available options based on the current environment (such as available namespaces, secrets, etc.), you can use API dynamic configuration to retrieve dropdown options from backend interfaces in real-time.
In this mode, API URLs are typically used in combination with context variables to dynamically construct request paths based on the user's current cluster, namespace, project, or filled parameters. For example:
# Use ${context.cluster} variable in URL to dynamically retrieve namespace list for current cluster
- urn:alm:descriptor:expression:props.options:api:apiPath
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/api/v1/namespaces
Available Context Variables
In dynamic API paths, you can use ${variable} to reference runtime context information.
- General Context Variables
- Pipeline/Task Related Context Variables
These variables are typically used in combination with API URLs to dynamically construct request addresses in different environments, enabling environment-aware option loading logic. For example:
- urn:alm:descriptor:expression:props.options:api:apiPath
- urn:alm:descriptor:expression:props.options:api:/api/v1/${context.namespace}/deployments?repo=${context.params.repo}
Configure Data Path for Returned Data
- urn:alm:descriptor:expression:props.options:path:spec.items
When the API returns a complex structure, for example:
{
"spec": {
"items": [ ... ]
}
}
You can use path to specify how to extract the options array from the structure.
Configure Label and Value Mapping Fields for Options
If the returned data object is complex and requires specifying inner fields as "display name" and "option value", you can use:
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
- urn:alm:descriptor:expression:props.options:value:path:metadata.name
When the API returns a simple string[] array, this configuration is not needed.
Configure API Query Parameters
If you need to pass query parameters to the API request (such as search keywords or filters), you can use:
# Add query parameter named "search" with user input as value
- urn:alm:descriptor:widgets:select
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/apis/tekton.dev/v1/namespaces/${context.namespace}/pipelineruns
- urn:alm:descriptor:expression:props.options:path:items
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
- urn:alm:descriptor:expression:props.options:value:path:metadata.name
- urn:alm:descriptor:expression:props.options:api:params:search:${current.search}
When user types "test" in the dropdown, the actual request will be:
/kubernetes/.../pipelineruns?search=test
Advanced Usage: Supports JavaScript expressions, for example:
${context.params.map(p => p.name).join(',')}
Parameter Default Values
Fixed Default Value
- urn:alm:descriptor:expression:props.default:default-value
Expression-based Dynamic Default Value
Available variables:
INFO
For conditional syntax rules in complex scenarios, please refer to JavaScript syntax.
Supports configuring various validation rules, multiple rules can be applied simultaneously.
Required Validation
- urn:alm:descriptor:com.tectonic.ui:validation:required
Length Validation
# Minimum length
- urn:alm:descriptor:com.tectonic.ui:validation:minLength:6
# Maximum length
- urn:alm:descriptor:com.tectonic.ui:validation:maxLength:64
Numeric Range Validation
# Minimum value (applicable to numeric types)
- urn:alm:descriptor:com.tectonic.ui:validation:minimum:1
# Maximum value
- urn:alm:descriptor:com.tectonic.ui:validation:maximum:100
Regular Expression Validation
# Positive integer
- urn:alm:descriptor:com.tectonic.ui:validation:pattern:[1-9]\d*
# Kubernetes resource name (lowercase letters, numbers, hyphens)
- urn:alm:descriptor:com.tectonic.ui:validation:pattern:[a-z0-9]([-a-z0-9]*[a-z0-9])?
Combined Example: Configure a password field with complete validation
- path: params.password
x-descriptors:
- urn:alm:descriptor:label:zh:password
- urn:alm:descriptor:com.tectonic.ui:text
- urn:alm:descriptor:description:zh:At least 8 characters, including letters and numbers.
- urn:alm:descriptor:com.tectonic.ui:validation:required
- urn:alm:descriptor:com.tectonic.ui:validation:minLength:8
- urn:alm:descriptor:com.tectonic.ui:validation:maxLength:32
Advanced Features
In complex CI/CD scenarios, the following advanced form components can help you handle more sophisticated configuration requirements.
Combine Component
In complex CI/CD scenarios, you may encounter situations where a single array parameter needs to collect data from multiple different sources (different APIs). For example, a workloads parameter might need to include both "Deployments" and "StatefulSets", which require fetching data from entirely different Kubernetes APIs.
To achieve this, you can use the Combine Component (widgets:combine). It allows you to split a single parameter into multiple independent sub-form controls on the UI (e.g., a radio button to select the resource type, and a select box to pick the actual resources), and then combine their values into a final array or JSON object via JavaScript expressions before submitting to the Pipeline.
Configuration
# Declare the use of the combine widget
- urn:alm:descriptor:widgets:combine
# Parent combine logic: defines how to merge all sub-component values into the final parameter
# Available variable: items (contains all sub-component values)
# Example: (items.deployments || []).concat(items.statefulsets || []) - merge multiple arrays into one
- urn:alm:descriptor:combine:value.exp:<js-expression>
# Parent: define sub-components
- urn:alm:descriptor:combine:items:
- path: <sub-component-name-1>
x-descriptors:
# Child split logic: defines how to extract this sub-component's value from the parent for echo display
# Available variable: parent (the current parameter value)
# Example: (parent || []).filter(p => p.startsWith('deploy:')) - extract items with 'deploy:' prefix
- urn:alm:descriptor:combine:value.exp:<js-expression>
# ... other descriptors for this sub-component
- path: <sub-component-name-2>
x-descriptors:
# Child split logic: defines how to extract this sub-component's value from the parent for echo display
- urn:alm:descriptor:combine:value.exp:<js-expression>
# ... other descriptors for this sub-component
For more js-expression syntax, please refer to JavaScript syntax.
Example: Configure a multi-select parameter target_resources that allows users to pick resources from both Deployments and StatefulSets. The final parameter value will be an array combining selections from both APIs, with specific prefixes applied to distinguish their types (e.g., ["deploy:app-v1", "sts:db-v1"]).
- path: params.target_resources
x-descriptors:
# Declare the use of the combine widget
- urn:alm:descriptor:widgets:combine
# [Parent] Combine logic: concat the arrays from both 'deployments' and 'statefulsets' sub-components
- urn:alm:descriptor:combine:value.exp:(items.deployments || []).concat(items.statefulsets || [])
# Define sub-components
- urn:alm:descriptor:combine:items:
# Sub-component 1: Select Deployments
- path: deployments
x-descriptors:
# [Child 1] Split logic: extract items starting with 'deploy:' from the parent array
- urn:alm:descriptor:combine:value.exp:(parent || []).filter(p => p.startsWith('deploy:'))
- urn:alm:descriptor:label:en:Deployments
- urn:alm:descriptor:label:zh:Deployments
- urn:alm:descriptor:widgets:select
- urn:alm:descriptor:props:select:multiple
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/apis/apps/v1/namespaces/${context.namespace}/deployments
- urn:alm:descriptor:expression:props.options:path:items
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
# Prepend 'deploy:' prefix
- urn:alm:descriptor:expression:props.options:value:path.exp:'deploy:'+$.metadata.name
# Sub-component 2: Select StatefulSets
- path: statefulsets
x-descriptors:
# [Child 2] Split logic: extract items starting with 'sts:' from the parent array
- urn:alm:descriptor:combine:value.exp:(parent || []).filter(p => p.startsWith('sts:'))
- urn:alm:descriptor:label:en:StatefulSets
- urn:alm:descriptor:label:zh:StatefulSets
- urn:alm:descriptor:widgets:select
- urn:alm:descriptor:props:select:multiple
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/apis/apps/v1/namespaces/${context.namespace}/statefulsets
- urn:alm:descriptor:expression:props.options:path:items
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
# Prepend 'sts:' prefix
- urn:alm:descriptor:expression:props.options:value:path.exp:'sts:'+$.metadata.name
How it works in the UI:
-
The user sees two separate multi-select boxes simultaneously: "Deployments" and "StatefulSets".
-
The Deployments box fetches data from the Deployment API, and the StatefulSets box fetches from the StatefulSet API.
-
When the user selects dummy and alm in Deployments, and db-1 in StatefulSets, the final target_resources array submitted to the Task is naturally combined:
- name: target_resources
value:
- deploy:dummy
- deploy:alm
- sts:db-1
Preview Component
The Preview Component allows users to preview the YAML content of a selected resource directly from the form. This is particularly useful in scenarios where users need to verify resource configurations before using them in Pipelines, such as checking ConfigMap content, validating Secret data, or reviewing Task/Pipeline YAML definitions.
For example, when a user selects a ConfigMap from a dropdown to use as a volume mount or environment variable source in a Task, they can preview the ConfigMap's data directly in the form to ensure the correct resource is selected.
Configuration
# Enable preview - show button when parameter has value
- urn:alm:descriptor:enhance:preview:context.params.<param-name>
# API endpoint to fetch resource content
- urn:alm:descriptor:preview:resource:api:<api-url>
# JSONPath or JS expression to extract content from response
- urn:alm:descriptor:preview:resource:path:<jsonpath>
- urn:alm:descriptor:preview:resource:path.exp:<js-expression>
Example: Configure a dropdown to select a ConfigMap, with a preview button to view its YAML content.
- path: params.configmap
x-descriptors:
# Form label
- urn:alm:descriptor:label:en:ConfigMap
- urn:alm:descriptor:label:zh:ConfigMap
# Dropdown to select ConfigMap from current namespace
- urn:alm:descriptor:com.tectonic.ui:select
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/api/v1/namespaces/${context.namespace}/configmaps
- urn:alm:descriptor:expression:props.options:path:items
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
- urn:alm:descriptor:expression:props.options:value:path:metadata.name
- urn:alm:descriptor:com.tectonic.ui:validation:required
# Preview feature
- urn:alm:descriptor:enhance:preview:context.params.configmap
- urn:alm:descriptor:preview:resource:api:/kubernetes/${context.cluster}/api/v1/namespaces/${context.namespace}/configmaps?fieldSelector=metadata.name%3D${context.params.configmap}
- urn:alm:descriptor:preview:resource:path:items[0]
How it works in the UI:
- User selects a ConfigMap from the dropdown (e.g.,
my-config).
- A preview button appears next to the dropdown.
- Clicking the preview button fetches the selected ConfigMap from the Kubernetes API.
- The raw YAML content is displayed in a preview panel with syntax highlighting.
Usage Examples
Example: Provide a dropdown box for selecting namespaces for the image parameter of a Task.
Effect: When users orchestrate a Pipeline or TaskRun based on the UI, the namespace parameter of this Task supports dropdown selection of namespaces.
apiVersion: tekton.dev/v1
kind: Task
metadata:
name: demo-task
namespace: <your namespace>
annotations:
style.tekton.dev/descriptors: |
- path: params.namespace
x-descriptors:
- urn:alm:descriptor:label:en:namespace
- urn:alm:descriptor:label:zh:namespace(zh)
- urn:alm:descriptor:com.tectonic.ui:select
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/api/v1/namespaces
- urn:alm:descriptor:expression:props.options:path:items
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
- urn:alm:descriptor:expression:props.options:value:path:metadata.name
- urn:alm:descriptor:com.tectonic.ui:validation:required
spec:
params:
- name: namespace
type: string
steps:
- name: demo
image: ubuntu
script: |
#!/bin/sh
pwd
Example: Provide a dropdown box for selecting namespaces for the target-namespace parameter of a Pipeline.
Effect: When users trigger a Pipeline based on the UI or need to associate this pipeline in other resources (Trigger, TriggerTemplate) and configure runtime parameters, the target-namespace parameter supports dropdown selection of namespaces.
apiVersion: tekton.dev/v1
kind: Pipeline
metadata:
annotations:
style.tekton.dev/descriptors: |
- path: params.target-namespace
x-descriptors:
- urn:alm:descriptor:label:en:target-namespace
- urn:alm:descriptor:label:zh:target-namespace(zh)
- urn:alm:descriptor:com.tectonic.ui:select
- urn:alm:descriptor:expression:props.options:api:/kubernetes/${context.cluster}/api/v1/namespaces
- urn:alm:descriptor:expression:props.options:label:path:metadata.name
- urn:alm:descriptor:expression:props.options:value:path:metadata.name
- urn:alm:descriptor:expression:props.options:path:items
- urn:alm:descriptor:com.tectonic.ui:validation:required
name: demo-pipelines
namespace: <your namespace>
spec:
params:
- name: target-namespace
type: string
tasks:
- name: kubectl
taskRef:
resolver: hub
params:
- name: kind
value: task
- name: catalog
value: catalog
- name: name
value: kubectl
- name: version
value: "0.1"
Troubleshooting
- Form not displayed or configuration not taking effect: Check if the YAML format of
style.tekton.dev/descriptors is correct, confirm path is configured as params.parameter-name, verify each descriptor starts with - urn:alm:descriptor:
- Dynamic options not loading: Confirm the API path is correct, check if the current user has permission to access the specified API resource, verify
path, label and value configurations point to fields that exist in the returned data
- Default value not taking effect: Confirm fixed default value uses
expression:props.default:value format, expression default value uses === instead of =, check if variables (option, index, length, context) in the expression are spelled correctly
- Context variables not resolved: Verify variable names are correct (e.g.,
context.namespace not context.namespaces), confirm variables are used in descriptors that support expressions, check if JavaScript expression syntax has errors