How to create dynamic GitHub Actions based on the contents of a repository

If you are here, you probably know what GitHub Actions are and understand the basics of YAML Dynamic GitHub Actions GitHub Actions allow you to create jobs programmatically by using something called matrix strategy. In this guide, we'll show how we use matrix strategies at Diploi to generate dynamic GitHub Action jobs, based on the configuration that applications generated using Diploi have For context, all applications generated on Diploi, have a mono-repo folder structure and every time a user launches a deployment, Diploi uses Kubernetes to create isolated Docker containers for each component (eg. Nextjs, Node, Svelte, etc) that the application created has. Every time a user pushes a new update to GitHub, we need to update the Docker images that Kubernetes will be running For an application on Diploi, we build two Docker images per component in the stack, one image for development, which has a cloud development environment and another image for staging/production environments without a development environment. The components of an application built using Diploi come from our Stack Builder, where you can configure a new application Walkthrough For this guide, we'll use an application with Bun and React+Vite In the backend, each component that's part of the stack of this application is sent to an API that generates the application's folders and creates a diploi.yaml file which serves as the blueprint for our implementation of Kubernetes and Docker, plus Build.yaml file for our GitHub Action The output from the API is a new application with a folder structure that looks like this: root/ .github/ workflow/ Build.yaml bun/ vite/ diploi.yaml We'll focus on the Build.yaml file, which it has all the instructions that will be run by the GitHub Action runner and it's the same file we generate for all applications created using Diploi's Stack Builder. Let's look at Build.yaml closer: name: Build Components on: push: branches: - '*' jobs: define-components: name: Define Components runs-on: ubuntu-latest outputs: components: ${{ steps.diploi-meta.outputs.components }} steps: - name: Checkout code uses: actions/checkout@v3 - id: diploi-meta name: Diploi meta uses: diploi/action-components@v1.6-alpha run-builds: name: Build ${{ matrix.name }} ${{ matrix.stage }} runs-on: ubuntu-latest needs: define-components strategy: fail-fast: false matrix: include: ${{ fromJSON(needs.define-components.outputs.components) }} steps: - name: Checkout code uses: actions/checkout@v3 - name: Diploi build uses: diploi/action-build@v1.7-alpha with: ${{ matrix }} env: project: ${{ secrets.DIPLOI_REGISTRY_PROJECT }} registry: ${{ secrets.DIPLOI_REGISTRY_HOSTNAME }} username: ${{ secrets.DIPLOI_REGISTRY_USERNAME }} password: ${{ secrets.DIPLOI_REGISTRY_PASSWORD }} Deep dive In a nutshell, our GitHub Action is divided in two jobs, define-components and run-builds, where define-components will generate the components based on our choices from the Stack Builder, which then will be passed to run-builds to create the builds for each component Let's breakdown these two jobs in detail: define-components - https://github.com/diploi/action-components/tree/main This job is in charge of creating an array of objects with the metadata from each component and add-on that will be part of our application. The action is stored on a separate repository, which houses an action.yml file which is the entry point for the GitHub Action runner and a node script with the code that takes in the components and add-ons details by reading the diploi.yaml file in the root of our monorepo. name: 'Diploi Component Metadata Action' description: 'An action that collects component metadata from a Diploi stack' branding: icon: 'package' color: 'purple' outputs: components: description: 'List of components (identifier, name, folder, type, ref) that need to be built' runs: using: 'node20' main: 'index.js' The define-components job executes a file called index.js that when the job is completed, will generate an output array which then will be used by the next job run-builds, //... const componentOutput = { identifier: component.identifier, name: component.name || component.identifier, folder: component.identifier, type: isDevImageAvailable ? 'main' : 'main-dev', }; //... core.setOutput('components', JSON.stringify(componentsOutput)); Let's take a look at the output of the job [{"identifier":"bun","name":"Bun","folder":"bun","type":"main"}, {"identifier":"bun","name":"Bun Dev","folder":"bun","type":"dev"}, {"identifier":"react-vite","name":"React + Vite","folder":"react-vite","type":"main"}, {"identifier":"react-vite","na

Apr 2, 2025 - 13:28
 0
How to create dynamic GitHub Actions based on the contents of a repository
If you are here, you probably know what GitHub Actions are and understand the basics of YAML

Dynamic GitHub Actions

GitHub Actions allow you to create jobs programmatically by using something called matrix strategy. In this guide, we'll show how we use matrix strategies at Diploi to generate dynamic GitHub Action jobs, based on the configuration that applications generated using Diploi have

For context, all applications generated on Diploi, have a mono-repo folder structure and every time a user launches a deployment, Diploi uses Kubernetes to create isolated Docker containers for each component (eg. Nextjs, Node, Svelte, etc) that the application created has. Every time a user pushes a new update to GitHub, we need to update the Docker images that Kubernetes will be running

For an application on Diploi, we build two Docker images per component in the stack, one image for development, which has a cloud development environment and another image for staging/production environments without a development environment. The components of an application built using Diploi come from our Stack Builder, where you can configure a new application

Diploi Stack Builder

Walkthrough

For this guide, we'll use an application with Bun and React+Vite

In the backend, each component that's part of the stack of this application is sent to an API that generates the application's folders and creates a diploi.yaml file which serves as the blueprint for our implementation of Kubernetes and Docker, plus Build.yaml file for our GitHub Action

The output from the API is a new application with a folder structure that looks like this:

root/
    .github/
        workflow/
            Build.yaml
    bun/
    vite/
    diploi.yaml

We'll focus on the Build.yaml file, which it has all the instructions that will be run by the GitHub Action runner and it's the same file we generate for all applications created using Diploi's Stack Builder. Let's look at Build.yaml closer:

name: Build Components

on:
  push:
    branches:
      - '*'

jobs:
  define-components:
    name: Define Components
    runs-on: ubuntu-latest
    outputs:
      components: ${{ steps.diploi-meta.outputs.components }}
    steps:
      - name: Checkout code
        uses: actions/checkout@v3
      - id: diploi-meta
        name: Diploi meta
        uses: diploi/action-components@v1.6-alpha
  run-builds:
    name: Build ${{ matrix.name }} ${{ matrix.stage }}
    runs-on: ubuntu-latest
    needs: define-components
    strategy:
      fail-fast: false
      matrix:
        include: ${{ fromJSON(needs.define-components.outputs.components) }}
    steps:
      - name: Checkout code
        uses: actions/checkout@v3
      - name: Diploi build
        uses: diploi/action-build@v1.7-alpha
        with: ${{ matrix }}
        env:
          project: ${{ secrets.DIPLOI_REGISTRY_PROJECT }}
          registry: ${{ secrets.DIPLOI_REGISTRY_HOSTNAME }}
          username: ${{ secrets.DIPLOI_REGISTRY_USERNAME }}
          password: ${{ secrets.DIPLOI_REGISTRY_PASSWORD }}

Deep dive

In a nutshell, our GitHub Action is divided in two jobs, define-components and run-builds, where define-components will generate the components based on our choices from the Stack Builder, which then will be passed to run-builds to create the builds for each component

Let's breakdown these two jobs in detail:

define-components - https://github.com/diploi/action-components/tree/main

This job is in charge of creating an array of objects with the metadata from each component and add-on that will be part of our application. The action is stored on a separate repository, which houses an action.yml file which is the entry point for the GitHub Action runner and a node script with the code that takes in the components and add-ons details by reading the diploi.yaml file in the root of our monorepo.

name: 'Diploi Component Metadata Action'
description: 'An action that collects component metadata from a Diploi stack'
branding:
icon: 'package'
color: 'purple'

outputs:
components:
    description: 'List of components (identifier, name, folder, type, ref) that need to be built'

runs:
using: 'node20'
main: 'index.js'

The define-components job executes a file called index.js that when the job is completed, will generate an output array which then will be used by the next job run-builds,

//...

const componentOutput = {
    identifier: component.identifier,
    name: component.name || component.identifier,
    folder: component.identifier,
    type: isDevImageAvailable ? 'main' : 'main-dev',
    };

//...

core.setOutput('components', JSON.stringify(componentsOutput));

Let's take a look at the output of the job

[{"identifier":"bun","name":"Bun","folder":"bun","type":"main"},
{"identifier":"bun","name":"Bun Dev","folder":"bun","type":"dev"},
{"identifier":"react-vite","name":"React + Vite","folder":"react-vite","type":"main"},
{"identifier":"react-vite","name":"React + Vite Dev","folder":"react-vite","type":"dev"}]

This output will be used as arguments that will dynamically generate the next job steps

run-builds - https://github.com/diploi/action-build/tree/main

Until this point, I talked about the run-builds job in singular, but as you might have noticed, the output from define-components will be an array which will then be used to generate an instance of the run-builds job for each object in the array

In run-builds we use the matrix strategy to make the job into dynamically generated jobs. This makes our process scalable since we don't need to have a unique action per component, and instead we can have a single action that takes each output object properties and runs a separate job dynamically

GitHub Action tree

Now that you have seen matrix strategies in action, you might have an idea as to how to generate jobs programmatically when running GitHub Actions

Go out there and make us proud!