Skip to main content

Compile bash to pipeline yaml- Gitlab, GitHub and more

Project description

bash2yaml

Compile pure Bash scripts into your CI/CD YAML. Get IDE support for your scripts while keeping them centralized.

Tired of writing Bash inside YAML strings with no syntax highlighting, linting, or testing? bash2yaml lets you develop your CI logic in .sh files and then compiles them into your CI/CD configuration, giving you the best of both worlds.

Supported targets: GitLab CI, GitHub Actions, CircleCI, AWS CodeBuild, Bitbucket Pipelines, and Semaphore CI.

Bash in YAML is Bash without quality gates. Also, includes support for inlining a large number of scripts from other language, from Python to PHP.

tests pre-commit.ci status Downloads Python Version Release


Before

Surely Gitlab has a solution for this? Not as far as I can tell. Here are some of my best workarounds.

Your IDE sees a single YAML string, and your scripts are trapped in one file.

.gitlab-ci.yml:

build-job:
  script:
    - echo "Building project"
    - make build

After

Your IDE provides full support for Bash, and your scripts can be versioned, shared, and tested independently.

scripts/build.sh:

#!/usr/bin/env bash
set -eo pipefail

echo "Building project"
make build

uncompiled.gitlab-ci.yml:

build-job:
  script:
    - ./scripts/build.sh

Run bash2yaml compile, and the final, valid .gitlab-ci.yml is generated for you.


Who is this for?

This tool is for you if:

  • You manage CI/CD templates in a centralized repository and include: them in many other projects.
  • Your .gitlab-ci.yml files contain thousands of lines of shell scripts.
  • You want to write, test, and debug your CI scripts locally without involving Docker or a full pipeline simulation.
  • Your IDE's lack of Bash support in YAML files is slowing you down.
  • You want to be able to put Python or other non-Bash scripts into your shared templates.

If your CI/CD configuration is simple or contained entirely within a single repository, you might not need this tool.

Installation

bash2yaml is a standalone command-line tool. Installation with pipx is recommended to avoid dependency conflicts.

Install [all] extras for all commands. On your build server install just bash2yaml for the core libraries which allows you to run compile, decompile on server. This minimizes supply chain risks.

# Recommended
pipx install bash2yaml[all]

# Or via pip
pip install bash2yaml[all]

Getting Started: A Quick Tour

  1. Initialize Your Project Run bash2yaml init to create a configuration file (.bash2yaml.toml) and directories to organize your source files.

  2. Decompile an Existing Configuration If you have an existing .gitlab-ci.yml with inline scripts, you can extract them automatically:

bash2yaml decompile --in-file .gitlab-ci.yml --out my-project/
  1. Write and Edit Your Scripts Create or edit your .sh files in the scripts directory. Write standard, clean Bash—your IDE will thank you. In your source YAML (uncompiled.yml):
my-job:
  script:
    - ./scripts/my-script.sh
  1. Compile Compile your source YAML and scripts into a final, GitLab-ready configuration:
bash2yaml compile --in my-project/ --out compiled/

This generates a compiled/.gitlab-ci.yml file, ready to be deployed to your project's root.

Usage and Commands

bash2yaml provides a few core commands to manage your workflow.

Run with

  • bash2yaml for CLI
  • bash2yaml-interactive for CLI question and answer
  • bash2yaml-tui for Terminal UI
  • bash2yaml-gui for GUI

Core Compile/Decompile

Command Description
compile Compiles source YAML and .sh files into final CI/CD YAML for your target.
decompile Extracts inline scripts from CI/CD YAML into separate .sh files.

Debugging from remote repo

Command Description
copy2local Copies compiled files from a central repo to a local project for testing.
map-deploy Copies compiled files from a central repo to a many local project for testing.
commit-map Copies intential changes in local projects back to the central repo.

Setup

Command Description
init Initializes a new bash2yaml project and config file.
clean Carefully delete output in target folder.
install-precommit Add git hook to compile before commit
uninstall-precommit Remove precommit hook

Diagnostics

Command Description
lint Call GitLab APIs to lint your YAML
detect-drift Report what unexpected changes were made to the generated files.
show-config Display config after cascade
doctor Look for environment problems
graph Generate graph of inline relationships
detect-uncompiled Detect if you forgot to compile
validate Validate JSON schema of all YAML in input and output

Other

Command Description
check-pins Analyze GitLab CI include: statements and suggest pinning to tags
trigger-pipelines Trigger pipelines in GitLab projects and optionally wait for completion
autogit Manually trigger autogit process (stage, commit, and/or push changes)

Simulate Gitlab Pipeline Locally

Command Description
run Best efforts to run bash in a .gitlab-ci.yml file in similar order as a real runner

For detailed options on any command, run bash2yaml <command> --help.


Advanced Topics

Bash Completion

Enable tab completion in your shell by running the global activation command once:

activate-global-python-argcomplete

Global Variables

To define variables that should be inlined into the global variables: block of your .gitlab-ci.yml, create a global_variables.sh file in your input directory.

Limitations

  • No include: Inlining: This tool only inlines .sh file references. It does not process or merge GitLab's include: statements for other YAML files.
  • Single Statement Invocations: The script invocation must be on its own line. Multi-statement lines like echo "hello" && script.sh are not supported.
  • Comments: Comments in the source YAML may not be preserved in the final compiled output.

How It Compares

  • gitlab-ci-local: This is an excellent tool for running your entire GitLab pipeline in local Docker containers. bash2yaml is different—it focuses on the "unit testing" of your Bash logic itself, assuming you can and want to execute your scripts on your local machine without the overhead of Docker.
  • **GitHub Actions ** GitHub composite actions do not have this problem. A shared GitHub action can reference a script in the shared action's repo. A GitHub "reusable" workflow is a single yaml file and might suffer from the same problem as Gitlab pipelines.
  • Git Submodules Build runners will need permissions to clone and git is more complicated to use.
  • Base image holds all bash You can only have one base image, so if you are using it for bash and yaml, you can't use other base images.
  • Trigger remote pipeline A remote pipeline has access to the shell files in its own repo.

Project Health & Info

Metric Health Metric Info
Tests Tests License License
Coverage Codecov PyPI PyPI
Lint / Pre-commit pre-commit.ci status Python Versions Python Version
Quality Gate Quality Gate Status Docs Docs
CI Build Build Downloads Downloads
Maintainability Maintainability Rating Last Commit Last Commit
Category Health
Open Issues GitHub issues
Stars GitHub Repo stars

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

bash2yaml-0.11.1.tar.gz (177.9 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

bash2yaml-0.11.1-py3-none-any.whl (222.8 kB view details)

Uploaded Python 3

File details

Details for the file bash2yaml-0.11.1.tar.gz.

File metadata

  • Download URL: bash2yaml-0.11.1.tar.gz
  • Upload date:
  • Size: 177.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for bash2yaml-0.11.1.tar.gz
Algorithm Hash digest
SHA256 7eea64efe9c1eaba98c4adb6a07bd94b93b3ea11d749e9f239ef950e9d790aed
MD5 fa2f16dd728a9f9418c4c78cb1462d2e
BLAKE2b-256 8afe813856cc698edc0bcf8a2b0000ace0f55ca7566dc57db87a4647cb8fa3fc

See more details on using hashes here.

Provenance

The following attestation bundles were made for bash2yaml-0.11.1.tar.gz:

Publisher: publish_to_pypi.yml on matthewdeanmartin/bash2yaml

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file bash2yaml-0.11.1-py3-none-any.whl.

File metadata

  • Download URL: bash2yaml-0.11.1-py3-none-any.whl
  • Upload date:
  • Size: 222.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for bash2yaml-0.11.1-py3-none-any.whl
Algorithm Hash digest
SHA256 6b797b540d8779e213315a74a8acada430ad6e9f9150b392d75d40000ba4a7f3
MD5 dba93b13e505550d833a6a1614e982ee
BLAKE2b-256 c20a3e1c0ce7f3b3c59355a98f2a3224c5eb7a8d3ecab51c92ba62d10b8ecd91

See more details on using hashes here.

Provenance

The following attestation bundles were made for bash2yaml-0.11.1-py3-none-any.whl:

Publisher: publish_to_pypi.yml on matthewdeanmartin/bash2yaml

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page