Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

#62221 GitHub Actions workflow hardening #8007

Open
wants to merge 46 commits into
base: trunk
Choose a base branch
from

Conversation

johnbillion
Copy link
Member

@johnbillion johnbillion commented Dec 16, 2024

What this does:

  • Removes all usage of inline GitHub Actions expressions in scripts, either replacing them with environment variables or refactoring the affected code. Actions expressions are not safe to use inline inside scripts.
  • Addresses all instances of missing quotes around variables in scripts to prevent word splitting.
  • Adds some missing permissions: {} declarations to workflows and jobs.
  • Introduces Actionlint for static analysis of workflow files. It detects syntax errors, semantic errors with expressions, inputs, outputs, and secrets, and includes several security checks.
  • Fixes a few issues detected by Actionlint that don't relate to security, for example references to non-existent matrix values and inputs.
  • Standardises on using maps for the environment properties in docker-compose.yml.
  • Refactors and reformats a few areas of code for legibility and clarity.

Notes

This PR previously included additional static analysis of workflow files using Octoscan, Zizmor, and Poutine, but I've removed these and I'll be proposing them separately at a later date.


Trac ticket: https://core.trac.wordpress.org/ticket/62221

This comment was marked as off-topic.

@github-advanced-security

This comment was marked as outdated.

github-advanced-security[bot]

This comment was marked as outdated.

@johnbillion johnbillion marked this pull request as ready for review January 21, 2025 12:32
Copy link

The following accounts have interacted with this PR and/or linked issues. I will continue to update these lists as activity occurs. You can also manually ask me to refresh this list by adding the props-bot label.

Core Committers: Use this line as a base for the props when committing in SVN:

Props johnbillion.

To understand the WordPress project's expectations around crediting contributors, please review the Contributor Attribution page in the Core Handbook.

@johnbillion johnbillion requested review from swissspidy, felixarntz, aaronjorbin and desrosj and removed request for felixarntz January 21, 2025 13:17
Copy link
Member

@felixarntz felixarntz left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@johnbillion Based on my limited GH Actions knowledge, this looks reasonable to me.

Since it covers all the different workflows, I think we need to do some extra post-commit verification though. Obviously all workflows should pass, but there are other aspects that we also need to test that wouldn't cause a workflow to fail - such as ensuring the performance data for each commit is still sent to the Code Vitals dashboard as expected.

Comment on lines -335 to -339
- name: Set commit details
# Only needed when publishing results.
if: ${{ github.event_name == 'push' && github.ref == 'refs/heads/trunk' && ! inputs.memcached && ! inputs.multisite }}
# Write to an environment variable to have the output available in later steps of the job.
run: echo "COMMITTED_AT=$(git show -s $GITHUB_SHA --format='%cI')" >> $GITHUB_ENV
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was just recently modified to fix a problem in https://core.trac.wordpress.org/changeset/59582. Your change looks right, but let's double-check that the commit data is still sent correctly after this has been committed to Core.

@@ -64,6 +64,10 @@ env:
LOCAL_PHP: ${{ inputs.php-version }}${{ 'latest' != inputs.php-version && '-fpm' || '' }}
LOCAL_MULTISITE: ${{ inputs.multisite }}

# Disable permissions for all available scopes by default.
# Any needed permissions should be configured at the job level.
permissions: {}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not quite familiar with how this permissions element works in GH Actions. Makes sense that it's an empty default for security. But I'm curious, for what kind of steps or jobs would one need to override this?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Typically for any write action, like pushing changes to the repository or writing PR comments or changing labels.

By disabling that by default if not needed, this reduces risk of privilege escalation, where for example one can suddenly push changes to a repository through a workflow.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There are some really fine-grained permissions available, so there's a good number of possibilities.

We also use contents: read permissions for the Slack notifications workflow so that the steps can read previous workflow runs and determine which message to send to Slack.

Copy link
Contributor

@desrosj desrosj left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for this, @johnbillion! Looks really great. I left mostly questions, some inline documentation additions, and a few suggestions. Feel free to commit after going through them!

push:
branches:
- trunk
- '[0-9].[0-9]'
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In the past when introducing new workflows, I've tried to use only future facing patterns starting with when the workflow was introduced. I don't feel strongly about not doing that here, but may be nice for consistency.

Suggested change
- '[0-9].[0-9]'
- '6.[8-9]'
- '[7-9].[0-9]'

This would require an update once we get to version 10.0, but based on 3 releases per year, that's not for at least 7 years from now.

branches:
- trunk
- '[0-9].[0-9]'
tags:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should this be run on tagging? Since this is only a build tool facing linter, I'm not sure we need to run it on a tag. We don't currently run the Docker environment or theme build test workflows on tag.


# https://github.com/rhysd/actionlint
- name: Run actionlint
uses: docker://rhysd/actionlint:1.7.7
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you know if this get picked up by Dependabot as currently configured? It's not clear in the docs.

with:
persist-credentials: false

# https://github.com/rhysd/actionlint
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
# https://github.com/rhysd/actionlint
# actionlint is static checker for GitHub Actions workflow files.
# See https://github.com/rhysd/actionlint.

runs-on: ubuntu-latest
permissions:
contents: read
timeout-minutes: 1
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would we want to pad this to 5 to guard against poor network connectivity or service level issues that cause delays?


jobs:
actionlint:
name: Actionlint
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
name: Actionlint
name: Run actionlint

permissions: {}

jobs:
actionlint:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
actionlint:
# Runs the actionlint GitHub Action workflow file linter.
#
# This helps guard against common mistakes including strong type checking for expressions (${{ }}), security checks,
# `run:` script checking, glob syntax validation, and more.
#
# Performs the following steps:
# - Checks out the repository.
# - Runs actionlint.
actionlint:

- name: Checkout repository
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
with:
persist-credentials: false
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
persist-credentials: false
persist-credentials: false
show-progress: ${{ runner.debug == '1' && 'true' || 'false' }}

We have this in other workflows calling checkout to limit the output when debug mode is disabled.


jobs:
lint:
name: Lint
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
name: Lint
name: Lint GitHub Action files

Makes the job name a bit less generic.

@@ -37,6 +37,10 @@ on:
type: 'string'
default: '5.7'

# Disable permissions for all available scopes by default.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks like we're now disabling permissions for all jobs in this callable workflow, but the upugrade-testing.yml file is passing contents: read. Unless I'm missing something, that can be changed to {}.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants