Only run linting jobs if the files changed make sense for the job

Created on 3 June 2024, 10 months ago
Updated 12 July 2024, 9 months ago

Problem/Motivation

Currently all jobs are run under any changes to the code, but it doesn't always make sense. For example, if no PHP files were changed at all we wouldn't need to run phpstan, or if no CSS files were changed, we don't need stylelint...

Steps to reproduce

Make any change to a contrib project and see how the pipelines run all the jobs.

Proposed resolution

For MRs:

  • Only run stylelint if CSS files were changed
  • Only run eslint if JS or YML files were changed
  • Only run phpstan if PHP files were changed

We might want to keep the current behaviour for scheduled pipelines.

Remaining tasks

MR and tests.

User interface changes

Some jobs won't run if they are not needed in the context of the MR.

Note

This issue is similar but not the same as #3367511: All files in the project are checked for coding standards โ†’ - which is about only running the linting jobs on the subset of files that have been changed in the MR

๐Ÿ“Œ Task
Status

Active

Component

gitlab-ci

Created by

๐Ÿ‡ช๐Ÿ‡ธSpain fjgarlin

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

  • Issue created by @fjgarlin
  • ๐Ÿ‡ฌ๐Ÿ‡งUnited Kingdom jonathan1055

    This is a good idea, and just to confirm, the new behavior should be limited to MR branches only. Scheduled and Commit pipelines should run everything, I think.

    We also might want a new option variable to turn off this filtering and run all jobs in a MR just like now. There are cases when, say, Coder has introduced a new sniff, or you are working on fixing coding standards that the project ignores. There needs to be a way to override the new behavior and find all places where the new standard fails. Using the scheduled pipeline would not be enough in this scenario as the sniff would be ignored in the committed code phpcs.xml so would not find the failures.

    I have updated the issue summary for eslint because we currently also check YML files. That job has

      rules:
        - *skip-eslint-rule
        - exists:
            - "**/*.js"
            - "**/*.yml"
    

    So I think we should leave the skip-...-rule unaltered, but add a - changes: to the actual job. Or maybe the exists: can just become changes:

    Could this also be extended to the composer-lint and phpcs jobs? The set of files is larger for those jobs, but the principle is still the same.

  • ๐Ÿ‡ช๐Ÿ‡ธSpain fjgarlin

    the new behavior should be limited to MR branches only. Scheduled and Commit pipelines should run everything, I think

    Correct, that's the way I see it too.

    We also might want a new option variable to turn off this filtering

    Happy with that approach.

    Could this also be extended to the composer-lint and phpcs jobs? The set of files is larger for those jobs, but the principle is still the same.

    Yup, 100%. I think all linting jobs where it makes sense.

    --

    It won't be as straightforward as replacing "exists" with "changes" as "changes" is MR specific, but we will start discovering those things once the work on this issue starts.

  • ๐Ÿ‡ง๐Ÿ‡ชBelgium wim leers Ghent ๐Ÿ‡ง๐Ÿ‡ช๐Ÿ‡ช๐Ÿ‡บ

    For example for the eslint job we should match core's behavior:

    '๐Ÿงน JavaScript linting (eslint)':
    โ€ฆ
      # Run on push, or on MRs if CSS files have changed, or manually.
      rules:
        - if: $CI_PIPELINE_SOURCE == "push" && $CI_PROJECT_ROOT_NAMESPACE == "project"
        - if: $CI_PIPELINE_SOURCE == "merge_request_event"
          changes:
            - core/.eslint*
            - core/.prettier*
            - core/package.json
            - core/yarn.lock
            - "**/*.js"
            - "**/*.yml"
        - when: manual
          allow_failure: true
    

    โ€ฆ but adjust for contrib:

      # Run on push, or on MRs if CSS files have changed, or manually.
      rules:
        - if: $CI_PIPELINE_SOURCE == "push" && $CI_PROJECT_ROOT_NAMESPACE == "project"
        - if: $CI_PIPELINE_SOURCE == "merge_request_event"
          changes:
            - "**/.eslint*"
            - "**/.prettier*"
            - "**/package.json"
            - "**/yarn.lock"
            - "**/*.js"
            - "**/*.yml"
        - when: manual
          allow_failure: true
    
  • ๐Ÿ‡ง๐Ÿ‡ชBelgium wim leers Ghent ๐Ÿ‡ง๐Ÿ‡ช๐Ÿ‡ช๐Ÿ‡บ

    I've implemented this for the contrib module I'm currently working on: #3460778-3: CI: do not run PHP-only jobs if an MR only changes the UI โ†’ .

    Works well ๐Ÿ‘ Speeds up CI runs, reduces CI resource consumption, and improves signal-to-noise ratio of CI.

  • ๐Ÿ‡ง๐Ÿ‡ชBelgium wim leers Ghent ๐Ÿ‡ง๐Ÿ‡ช๐Ÿ‡ช๐Ÿ‡บ

    Crucially:

    • Merge request pipelines, rules:changes compares the changes with the target MR branch.
    • Branch pipelines, rules:changes compares the changes with the previous commit on the branch.

    โ€” https://docs.gitlab.com/ee/ci/yaml/#ruleschanges

  • ๐Ÿ‡ง๐Ÿ‡ชBelgium wim leers Ghent ๐Ÿ‡ง๐Ÿ‡ช๐Ÿ‡ช๐Ÿ‡บ
Production build 0.71.5 2024