Skip to content
Crow CI

Pipelines

Pipeline List View
Pipeline List View

Crow uses a three-level hierarchy: pipeline > workflow > step.

  • Pipeline — a single CI invocation. A pipeline is not defined in a file; it is created at runtime in response to one event (push, pull request, manual run, …) and bundles all workflows that match that event.
  • Workflow — a YAML file in .crow/ (or Jsonnet .jsonnet) defining a collection of steps. This is what you author to describe what should run.
  • Step — a single command sequence executed inside a container.

When an event reaches Crow, every workflow whose when block matches that event is collected into one pipeline and executed. A push that matches three workflow files results in one pipeline containing three workflows — not three pipelines. The pipeline is the run; the workflows are what runs inside it.

Each YAML file in .crow/ defines one workflow. A workflow that uses a matrix expands at runtime into one workflow instance per matrix combination, so a single matrix file can produce multiple workflows in the resulting pipeline.

The following file tree defines four workflows:

.crow/
├── build.yaml
├── deploy.yaml
├── lint.yaml
└── test.yaml

Each workflow can consist of an arbitrary number of steps. By default, workflows do not have a dependency to each other and are executed in parallel. Steps within a workflow are executed sequentially by their order of definition.

Both steps and workflows accept a depends_on: [] key which can be used to specify an explicit execution order.

By default, all workflows start in parallel if they have matching event triggers. An execution order can be enforced by using depends_on:

steps:
  - name: deploy
    image: <image>:<tag>
    commands:
      - some command

# these are names of other workflows
depends_on:
  - lint
  - build
  - test

This keyword also works for dependency resolution with steps.

Event triggers are mandatory and define under which conditions a workflow is executed. At the very least one even trigger must be specified, for example to execute the pipeline on a push event:

when:
  event: push

Typically, you want to use a more fine-grained logic including more events, for example triggering a workflow for pull_request events and pushes to the default branch of the repository:

when:
  - event: pull_request
  - event: push
    branch: ${CI_REPO_DEFAULT_BRANCH}

There are more ways to define event triggers using both list and map notation. See the when filters reference for all available options and the full list of events.

Matrix workflows execute a separate workflow for each combination in the specified matrix. This simplifies testing and building against multiple configurations without copying the full pipeline definition but only declare the variable parts.

Example:

matrix:
  GO_VERSION:
    - 1.4
    - 1.3
  REDIS_VERSION:
    - 2.6
    - 2.8
    - 3.0

Each definition can also be a combination of variables. In this case, nest the definitions below the include keyword:

matrix:
  include:
    - GO_VERSION: 1.4
      REDIS_VERSION: 2.8
    - GO_VERSION: 1.5
      REDIS_VERSION: 2.8

Matrix variables are interpolated in the YAML using the ${VARIABLE} syntax, before the YAML is parsed. This is an example YAML file before interpolating matrix parameters:

matrix:
  GO_VERSION:
    - 1.4
    - 1.3
  DATABASE:
    - mysql:8
    - mysql:5
    - mariadb:10.1

steps:
  - name: build
    image: golang:${GO_VERSION}
    commands:
      - go get
      - go build
      - go test

services:
  - name: database
    image: ${DATABASE}

And after:

steps:
  - name: build
    image: golang:1.4
    commands:
      - go get
      - go build
      - go test
    environment:
      - GO_VERSION=1.4
      - DATABASE=mysql:8

services:
  - name: database
    image: mysql:8
matrix:
  IMAGE:
    - golang:1.7
    - golang:1.8
    - golang:latest

steps:
  - name: build
    image: ${IMAGE}
    commands:
      - go build
      - go test
matrix:
  platform:
    - linux/amd64
    - linux/arm64

steps:
  - name: test
    image: <image>
    commands:
      - echo "Running on ${platform}"

  - name: test-arm
    image: <image>
    commands:
      - echo "Running on ${platform}"
    when:
      platform: linux/arm*

Commits can be prohibited from triggering a webhook by adding [SKIP CI] or [CI SKIP] (case-insensitive) to the commit message.

Crow supports configurable container naming schemes which determines the names of containers (and pods/services in Kubernetes) created during pipeline execution.

There are two supported schemes:

  1. Descriptive (default)

    • Format: <owner>-<repo name>-<pipeline id>-<workflow id>-<step name>
    • Example: myowner-myrepo-42-3-build
    • Matrix workflows: Each workflow instance gets a unique workflow number for proper identification
    • Single workflows: Workflow number is still included for consistency
  2. Hash-based (legacy)

    • Format: crow_<hash>
    • Example: crow_123e4567-e89b-12d3-a456-426614174000
    • This was previously the default (until 3.x) (as wp_<hash>), now updated to crow_ for clarity.

The naming scheme can be set via the server environment variable CROW_CONTAINER_NAME_SCHEME.

A pipeline can also be triggered manually from the UI or CLI rather than by an event like push or pull_request. To make a workflow eligible for manual triggering, add the manual event to its when block:

when:
  - event: manual

Or combined with other events:

when:
  - event: [push, manual]
    branch: main

When triggering a manual pipeline from the UI, the dialog has two columns: the available workflows on the left and their variables on the right. Only workflows with the manual event configured are listed. The selected workflows are bundled into a single new pipeline. You can:

  • Select workflows: Click anywhere on a workflow row (or its checkbox) to include it, then click “Run pipeline”.
  • Quick-start a single workflow: Click the play button next to a workflow to immediately start a pipeline containing just that workflow.
  • Select all: Use the “Select all” checkbox to include every available workflow.

Workflows that depend on others show a dependency icon; hover it to see the dependencies.

Manual pipeline trigger dialog
Manual pipeline trigger dialog

The right column shows the variables the selected workflows accept, grouped by workflow so it is clear which variable belongs where. Variables declared by more than one selected workflow appear once in a Shared group at the top; each remaining workflow’s variables follow under its own heading.

These variables are derived from each workflow’s declared variables: block and from the variables referenced in its when.evaluate expressions. Variables with a fixed set of values render as dropdowns; others as text inputs, both prefilled with their declared default. Required variables are marked and must be set before the pipeline can run.

Manual pipeline variables panel
Manual pipeline variables panel

Free-form variable entry is not available: only declared or evaluate-referenced variables can be set. Submitting a value outside a variable’s declared options, or omitting a required variable, is rejected.

When a selected workflow depends on other workflows (via depends_on), the system automatically includes those dependencies if they also have the manual event trigger.

For example, given these workflows:

# .crow/build.yaml
when:
  - event: manual

steps:
  - name: build
    image: golang
    commands:
      - go build
# .crow/deploy.yaml
when:
  - event: manual

depends_on:
  - build

steps:
  - name: deploy
    image: alpine
    commands:
      - ./deploy.sh

If you select only “deploy”, the “build” workflow will be automatically included because:

  1. “deploy” depends on “build”
  2. “build” has the manual event trigger

Manual pipelines can also be triggered via the CLI:

crow pipeline create --branch main --repo owner/repo

To run specific workflows, use the --workflow flag:

crow pipeline create --branch main --repo owner/repo --workflow build --workflow test

The evaluate condition can be used to further filter manual workflows. This is useful when you want to pass variables to control which steps run:

when:
  - event: manual
    evaluate: 'DEPLOY_ENV == "production"'

When triggering manually via the CLI, you can pass variables:

crow pipeline create --branch main --repo owner/repo --var DEPLOY_ENV=production