In Ruby development, maintaining code quality across versions is crucial. This post demonstrates how to set up a workflow to test your code across versions and generate detailed coverage reports, keeping your project robust as Ruby evolves.
If you don’t already have a working setup for Ruby code coverage, I recommend checking out my Ruby Code Coverage Setup Guide, where I explain how to configure coverage reports for different dependencies and runtimes via test runners. You can also find the example at ryancyq/ruby-code-coverage.
Prerequisites
These are the tools being used:
- CodeCov : A code coverage reporting and tracking tool.
- GitHub Actions : A CI/CD platform integrated with GitHub.
Create a GitHub Actions Workflow
Let’s create a GitHub Actions workflow for code coverage analysis and run the same job for different Ruby versions. This will be done by leveraging GitHub Actions matrix strategies.
# coverage.yml
jobs:
coverage:
name: "Ruby ${{ matrix.ruby-version }}"
runs-on: ubuntu-latest
strategy:
fail-fast: false
matrix:
ruby-version:
- "2.7"
- "3.0"
- "3.1"
- "3.2"
- "3.3"
- "head"
steps:
- uses: actions/checkout@v4
- uses: ruby/setup-ruby@v1
with:
ruby-version: ${{ matrix.ruby-version }}
rubygems: 3.4.22 # last version to support Ruby 2.7
Next, we need to install the gems required for generating the coverage report as described in Ruby Code Coverage setup guide.
# coverage.yml
jobs:
coverage:
# .. more
- run: gem install rake rspec simplecov simplecov-html simplecov-cobertura
- run: rake coverage:run
- run: sudo apt install tree
- run: tree -a coverage
- uses: actions/upload-artifact@v4
with:
name: "coverage-ruby-${{ matrix.ruby-version }}"
path: coverage/ruby-*
include-hidden-files: true
retention-days: 1
The tree
command is included in the example for troubleshooting purposes. In the case of Ruby 3.3
, we should see the following output after executing the tree
command:
coverage
└── ruby-3.3.5
├── .last_run.json
├── .resultset.json
├── .resultset.json.lock
└── coverage.xml
1 directory, 4 files
The coverage result from each Ruby version will be uploaded to GitHub Actions Artifacts.
data:image/s3,"s3://crabby-images/c6d5d/c6d5d0459d8f8b75bbae302ec91bc9025f5e98a4" alt="Code Coverage Artifacts"
Collate Coverage Reports from Test Runners
Next, we need to add another report
job to collate the coverage results into a single coverage result.
# coverage.yml
jobs:
# .. more
report:
name: "Report to CodeCov"
needs: [coverage]
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/download-artifact@v4
with:
pattern: coverage-ruby-*
path: coverage-results
merge-multiple: true
- uses: ruby/setup-ruby@v1
with:
ruby-version: "3"
- run: gem install rake rspec simplecov simplecov-html simplecov-cobertura
- run: rake coverage:report
env:
COV_DIR: coverage-results
- run: sudo apt install tree
- run: tree -a coverage-results
- run: tree -a coverage
The outputs from the tree
commands above should look something like:
tree -a coverage-results
coverage-results
├── ruby-2.7.8
│ ├── .last_run.json
│ ├── .resultset.json
│ ├── .resultset.json.lock
│ └── coverage.xml
├── ruby-3.0.7
│ ├── .last_run.json
│ ├── .resultset.json
│ ├── .resultset.json.lock
│ └── coverage.xml
├── ruby-3.1.6
│ ├── .last_run.json
│ ├── .resultset.json
│ ├── .resultset.json.lock
│ └── coverage.xml
├── ruby-3.2.5
│ ├── .last_run.json
│ ├── .resultset.json
│ ├── .resultset.json.lock
│ └── coverage.xml
├── ruby-3.3.5
│ ├── .last_run.json
│ ├── .resultset.json
│ ├── .resultset.json.lock
│ └── coverage.xml
└── ruby-3.4.0
├── .last_run.json
├── .resultset.json
├── .resultset.json.lock
└── coverage.xml
6 directories, 24 files
tree -a coverage
coverage
├── .last_run.json
├── .resultset.json
├── .resultset.json.lock
└── coverage.xml
0 directories, 4 files
Report to CodeCov
In the next step, we will pass the coverage
directory (which contains the collated coverage result) to CodeCov Action.
# coverage.yml
jobs:
# .. more
report:
steps:
# .. more
- name: Upload coverage to Codecov
uses: codecov/codecov-action@v4
with:
directory: coverage
token: ${{ secrets.CODECOV_TOKEN }}
fail_ci_if_error: true
Report to CodeCov with Flags
Alternatively, if you want to leverage CodeCov Flags to have a better overview of the coverage result for each Ruby version, we could skip coverage result collation and upload individual coverage results to CodeCov with the corresponding flags.
# coverage.yml
jobs:
# .. more
report:
name: "Report to CodeCov"
needs: [coverage]
runs-on: ubuntu-latest
strategy:
fail-fast: false
matrix:
ruby-version:
- "2.7"
- "3.0"
- "3.1"
- "3.2"
- "3.3"
- "head"
steps:
- uses: actions/checkout@v4
- uses: actions/download-artifact@v4
with:
pattern: "coverage-ruby-${{ matrix.ruby-version }}*"
path: coverage
- run: sudo apt install tree
- run: tree -a coverage
- name: Upload coverage to Codecov
uses: codecov/codecov-action@v4
with:
directory: coverage
token: ${{ secrets.CODECOV_TOKEN }}
flags: "ruby-${{ matrix.ruby-version }}"
fail_ci_if_error: true
In the modified report
job, we’ve removed the need for ruby-setup
to collate coverage results, and artifacts are now downloaded one at a time.
data:image/s3,"s3://crabby-images/e582f/e582f1bddd501f511f9a29332b7bffdf7ed15a64" alt="CodeCov with flags"
The coverage report for each Ruby version remains around 91% due to the if-else
statement on RUBY_VERSION
.
data:image/s3,"s3://crabby-images/52003/52003a424b7f8b5010e96f666c8eca8e96d4be6d" alt="CodeCov Report for flags"
However, the overall code coverage stays at 100% as the result of merging the individual coverage reports.
data:image/s3,"s3://crabby-images/6b6bf/6b6bf6db531944252b9f4c98469d3d3ff9c1668e" alt="Overall CodeCov Report"