[![Ubuntu](https://img.shields.io/badge/Ubuntu-E95420?style=for-the-badge\&logo=ubuntu\&logoColor=white)](https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions#jobsjob_idruns-on) [![Mac OS](https://img.shields.io/badge/mac%20os-000000?style=for-the-badge\&logo=macos\&logoColor=F0F0F0)](https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions#jobsjob_idruns-on) [![Windows](https://img.shields.io/badge/Windows-0078D6?style=for-the-badge\&logo=windows\&logoColor=white)](https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions#jobsjob_idruns-on) [![Public workflows that use this action.](https://img.shields.io/endpoint?style=for-the-badge\&url=https%3A%2F%2Fused-by.vercel.app%2Fapi%2Fgithub-actions%2Fused-by%3Faction%3Dtj-actions%2Fchanged-files%26badge%3Dtrue)](https://github.com/search?o=desc\&q=tj-actions+changed-files+language%3AYAML\&s=\&type=Code) [![Codacy Badge](https://app.codacy.com/project/badge/Grade/4a625e9b62794b5b98e169c15c0e673c)](https://www.codacy.com/gh/tj-actions/changed-files/dashboard?utm_source=github.com\&utm_medium=referral\&utm_content=tj-actions/changed-files\&utm_campaign=Badge_Grade) [![CI](https://github.com/tj-actions/changed-files/actions/workflows/test.yml/badge.svg)](https://github.com/tj-actions/changed-files/actions/workflows/test.yml) [![Update release version.](https://github.com/tj-actions/changed-files/actions/workflows/sync-release-version.yml/badge.svg)](https://github.com/tj-actions/changed-files/actions/workflows/sync-release-version.yml) [![All Contributors](https://img.shields.io/badge/all_contributors-20-orange.svg?style=flat-square)](#contributors-) ## changed-files Effortlessly track all changed files and directories relative to a target branch, preceding commit or the last remote commit returning **relative paths** from the project root using this GitHub action. ## Table of contents * [Features ๐Ÿš€](#features-) * [Usage ๐Ÿ’ป](#usage-) * [On `pull_request` ๐Ÿ”€](#on-pull_request-) * [Using local .git history ๐Ÿ“](#using-local-git-history-) * [Using Github's API :octocat:](#using-githubs-api-octocat) * [On `push` โฌ†๏ธ](#on-push-๏ธ) * [Useful Acronyms ๐Ÿงฎ](#useful-acronyms-) * [Outputs ๐Ÿ“ค](#outputs-) * [Inputs โš™๏ธ](#inputs-๏ธ) * [Versioning ๐Ÿท๏ธ](#versioning-๏ธ) * [Examples ๐Ÿ“„](#examples-) * [Real-world usage ๐ŸŒ](#real-world-usage-) * [Open source projects ๐Ÿ“ฆ](#open-source-projects-) * [Scalability Example ๐Ÿ“ˆ](#scalability-example-) * [Known Limitation โš ๏ธ](#known-limitation-๏ธ) * [Migration guide ๐Ÿ”„](#migration-guide-) * [Credits ๐Ÿ‘](#credits-) * [Report Bugs ๐Ÿ›](#report-bugs-) * [Contributors โœจ](#contributors-) ## Features ๐Ÿš€ * Fast execution, averaging 0-10 seconds. * Leverages either [Github's REST API](https://docs.github.com/en/rest/reference/repos#list-commits) or [Git's native diff](https://git-scm.com/docs/git-diff) to determine changed files. * Facilitates easy debugging. * Scales to handle large repositories. * Supports Git submodules. * Supports [merge queues](https://docs.github.com/en/repositories/configuring-branches-and-merges-in-your-repository/configuring-pull-request-merges/managing-a-merge-queue) for pull requests. * Generates escaped [JSON output for running matrix jobs](https://github.com/tj-actions/changed-files/blob/main/.github/workflows/matrix-test.yml) based on changed files. * Lists changed directories. * Limits matching changed directories to a specified maximum depth. * Optionally excludes the current directory. * Writes outputs to a designated `.txt` or `.json` file for further processing. * Restores deleted files to their previous location or a newly specified location. * Supports Monorepos by fetching a fixed number of commits. * Compatible with all platforms (Linux, MacOS, Windows). * Supports [GitHub-hosted runners](https://docs.github.com/en/actions/using-github-hosted-runners/about-github-hosted-runners). * Supports [GitHub Enterprise Server](https://docs.github.com/en/enterprise-server@3.3/admin/github-actions/getting-started-with-github-actions-for-your-enterprise/getting-started-with-github-actions-for-github-enterprise-server). * Supports [self-hosted runners](https://docs.github.com/en/actions/hosting-your-own-runners/about-self-hosted-runners). * Lists all files and directories that have changed: * Between the current pull request branch and the last commit on the target branch. * Between the last commit and the current pushed change. * Between the last remote branch commit and the current HEAD. * Restricts change detection to a subset of files and directories: * Provides boolean output indicating changes in specific files. * Uses [Glob pattern](https://codepen.io/mrmlnc/pen/OXQjMe) matching. * Supports Globstar. * Supports brace expansion. * Supports negation. * Uses [YAML](https://yaml.org/) syntax for specifying patterns. * Supports [YAML anchors & aliases](https://www.educative.io/blog/advanced-yaml-syntax-cheatsheet#anchors). * Supports [YAML multi-line strings](https://learnxinyminutes.com/docs/yaml/). And many more... ## Usage ๐Ÿ’ป > **Warning** > > * For `push` events: When configuring [`actions/checkout`](https://github.com/actions/checkout#usage), make sure to set [`fetch-depth`](https://github.com/actions/checkout#usage) to either `0` or `2`, depending on your use case. > * For mono repositories where pulling all branch history might not be desired, you can still use the default [`fetch-depth`](https://github.com/actions/checkout#usage), which is set to `1` for `pull_request` events. > * Avoid using single or double quotes for multiline inputs, as the value is already a string separated by a newline character. See [Examples](#examples) for more information. > * If [`fetch-depth`](https://github.com/actions/checkout#usage) isn't set to `0`, ensure that `persist-credentials` is set to `true` when configuring [`actions/checkout`](https://github.com/actions/checkout#usage). > * For repositories that have PRs generated from forks, when configuring [`actions/checkout`](https://github.com/actions/checkout#usage), set the [`repository`](https://github.com/actions/checkout#usage) to `${{ github.event.pull_request.head.repo.full_name }}`. See [Example](https://github.com/tj-actions/changed-files/blob/main/.github/workflows/test.yml#L47-L51). Visit the [discussions for more information](https://github.com/tj-actions/changed-files/discussions) or [create a new discussion](https://github.com/tj-actions/changed-files/discussions/new/choose) for usage-related questions. ### On `pull_request` ๐Ÿ”€ #### Using local .git history ๐Ÿ“ ```yaml name: CI on: pull_request: branches: - main jobs: # ------------------------------------------------------------------------------------------------------------------------------------------------ # Event `pull_request`: Compare the last commit of the main branch or last remote commit of the PR branch -> to the current commit of a PR branch. # ------------------------------------------------------------------------------------------------------------------------------------------------ changed_files: runs-on: ubuntu-latest # windows-latest || macos-latest name: Test changed-files steps: - uses: actions/checkout@v3 with: fetch-depth: 0 # OR "2" -> To retrieve the preceding commit. # Example 1 - name: Get all test, doc and src files that have changed id: changed-files-yaml uses: tj-actions/changed-files@v38 with: files_yaml: | doc: - '**/*.md' - docs/** - README.md test: - test/** - '!test/**.md' src: - src/** # Optionally set `files_yaml_from_source_file` to read the YAML from a file. e.g `files_yaml_from_source_file: .github/changed-files.yml` - name: Run step if test file(s) change # NOTE: Ensure all outputs are prefixed by the same key used above e.g. `test_(...)` | `doc_(...)` | `src_(...)` when trying to access the `any_changed` output. if: steps.changed-files-yaml.outputs.test_any_changed == 'true' run: | echo "One or more test file(s) has changed." echo "List all the files that have changed: ${{ steps.changed-files-yaml.outputs.test_all_changed_files }}" - name: Run step if doc file(s) change if: steps.changed-files-yaml.outputs.doc_any_changed == 'true' run: | echo "One or more doc file(s) has changed." echo "List all the files that have changed: ${{ steps.changed-files-yaml.outputs.doc_all_changed_files }}" # Example 2 - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 # To compare changes between the current commit and the last pushed remote commit set `since_last_remote_commit: true`. e.g # with: # since_last_remote_commit: true - name: List all changed files run: | for file in ${{ steps.changed-files.outputs.all_changed_files }}; do echo "$file was changed" done # Example 3 - name: Get changed files in the docs folder id: changed-files-specific uses: tj-actions/changed-files@v38 with: files: docs/*.{js,html} # Alternatively using: `docs/**` or `docs` files_ignore: docs/static.js - name: Run step if any file(s) in the docs folder change if: steps.changed-files-specific.outputs.any_changed == 'true' run: | echo "One or more files in the docs folder has changed." echo "List all the files that have changed: ${{ steps.changed-files-specific.outputs.all_changed_files }}" ``` #### Using Github's API :octocat: ```yaml name: CI on: pull_request: branches: - main jobs: # ------------------------------------------------------------- # Event `pull_request`: Returns all changed pull request files. # -------------------------------------------------------------- changed_files: # NOTE: # - This is limited to pull_request* events and would raise an error for other events. # - A maximum of 3000 files can be returned. # - For more flexibility and no limitations see "Using local .git history" above. runs-on: ubuntu-latest # windows-latest || macos-latest name: Test changed-files permissions: pull-requests: read steps: - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 - name: List all changed files run: | for file in ${{ steps.changed-files.outputs.all_changed_files }}; do echo "$file was changed" done ``` ### On `push` โฌ†๏ธ ```yaml name: CI on: push: branches: - main # ------------------------------- # Optionally run on other events # ------------------------------- # schedule: # - cron: '0 0 * * *' # # release: # types: [...] # # workflow_dispatch: # # push: # tags: # - '**' # # merge_group: # # ...and many more jobs: # ------------------------------------------------------------- # Using GitHub's API is not supported for push events # ------------------------------------------------------------- # # ---------------------------------------------------------------------------------------------- # Using local .git history # ---------------------------------------------------------------------------------------------- # Event `push`: Compare the preceding remote commit -> to the current commit of the main branch # ---------------------------------------------------------------------------------------------- changed_files: runs-on: ubuntu-latest # windows-latest || macos-latest name: Test changed-files steps: - uses: actions/checkout@v3 with: fetch-depth: 0 # OR "2" -> To retrieve the preceding commit. # Example 1 - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 # NOTE: `since_last_remote_commit: true` is implied by default and falls back to the previous local commit. - name: List all changed files run: | for file in ${{ steps.changed-files.outputs.all_changed_files }}; do echo "$file was changed" done # Example 2: See above ... # Example 3: See above ... ``` To access more examples, navigate to the [Examples](#examples) section. If you feel generous and want to show some extra appreciation: Support this project with a :star: [![Buy me a coffee][buymeacoffee-shield]][buymeacoffee] [buymeacoffee]: https://www.buymeacoffee.com/jackton1 [buymeacoffee-shield]: https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png ## Useful Acronyms ๐Ÿงฎ | Acronym | Meaning | |:---------:|:------------:| | A | Added | | C | Copied | | M | Modified | | D | Deleted | | R | Renamed | | T | Type changed | | U | Unmerged | | X | Unknown | > **Warning** > > * When using `files_yaml*` inputs ensure all outputs are prefixed by the key `test_{...}` e.g. `test_added_files`, `test_any_changed` ## Outputs ๐Ÿ“ค | OUTPUT | TYPE | DESCRIPTION | |------------------------------------------------------------------------------------------------------------------------------------------------|--------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | [added\_files](#output_added_files) | string | Returns only files that are
Added (A). | | [added\_files\_count](#output_added_files_count) | string | Returns the number of `added_files` | | [all\_changed\_and\_modified\_files](#output_all_changed_and_modified_files) | string | Returns all changed and modified
files i.e. *a combination of (ACMRDTUX)* | | [all\_changed\_and\_modified\_files\_count](#output_all_changed_and_modified_files_count) | string | Returns the number of `all_changed_and_modified_files` | | [all\_changed\_files](#output_all_changed_files) | string | Returns all changed files i.e.
*a combination of all added, copied, modified and renamed files (ACMR)* | | [all\_changed\_files\_count](#output_all_changed_files_count) | string | Returns the number of `all_changed_files` | | [all\_modified\_files](#output_all_modified_files) | string | Returns all changed files i.e.
*a combination of all added, copied, modified, renamed and deleted files (ACMRD)*. | | [all\_modified\_files\_count](#output_all_modified_files_count) | string | Returns the number of `all_modified_files` | | [all\_old\_new\_renamed\_files](#output_all_old_new_renamed_files) | string | Returns only files that are
Renamed and lists their old
and new names. **NOTE:** This
requires setting `include_all_old_new_renamed_files` to `true`.
Also, keep in mind that
this output is global and
wouldn't be nested in outputs
generated when the `*_yaml_*` input
is used. (R) | | [all\_old\_new\_renamed\_files\_count](#output_all_old_new_renamed_files_count) | string | Returns the number of `all_old_new_renamed_files` | | [any\_changed](#output_any_changed) | string | Returns `true` when any of
the filenames provided using the
`files*` or `files_ignore*` inputs has changed. i.e.
*using a combination of all added, copied, modified and renamed files (ACMR)*. | | [any\_deleted](#output_any_deleted) | string | Returns `true` when any of
the filenames provided using the
`files*` or `files_ignore*` inputs has been deleted.
(D) | | [any\_modified](#output_any_modified) | string | Returns `true` when any of
the filenames provided using the
`files*` or `files_ignore*` inputs has been modified.
i.e. *using a combination of all added, copied, modified, renamed, and deleted files (ACMRD)*. | | [copied\_files](#output_copied_files) | string | Returns only files that are
Copied (C). | | [copied\_files\_count](#output_copied_files_count) | string | Returns the number of `copied_files` | | [deleted\_files](#output_deleted_files) | string | Returns only files that are
Deleted (D). | | [deleted\_files\_count](#output_deleted_files_count) | string | Returns the number of `deleted_files` | | [modified\_files](#output_modified_files) | string | Returns only files that are
Modified (M). | | [modified\_files\_count](#output_modified_files_count) | string | Returns the number of `modified_files` | | [only\_changed](#output_only_changed) | string | Returns `true` when only files
provided using the `files*` or `files_ignore*` inputs
has changed. i.e. *using a combination of all added, copied, modified and renamed files (ACMR)*. | | [only\_deleted](#output_only_deleted) | string | Returns `true` when only files
provided using the `files*` or `files_ignore*` inputs
has been deleted. (D) | | [only\_modified](#output_only_modified) | string | Returns `true` when only files
provided using the `files*` or `files_ignore*` inputs
has been modified. (ACMRD). | | [other\_changed\_files](#output_other_changed_files) | string | Returns all other changed files
not listed in the files
input i.e. *using a combination of all added, copied, modified and renamed files (ACMR)*. | | [other\_changed\_files\_count](#output_other_changed_files_count) | string | Returns the number of `other_changed_files` | | [other\_deleted\_files](#output_other_deleted_files) | string | Returns all other deleted files
not listed in the files
input i.e. *a combination of all deleted files (D)* | | [other\_deleted\_files\_count](#output_other_deleted_files_count) | string | Returns the number of `other_deleted_files` | | [other\_modified\_files](#output_other_modified_files) | string | Returns all other modified files
not listed in the files
input i.e. *a combination of all added, copied, modified, and deleted files (ACMRD)* | | [other\_modified\_files\_count](#output_other_modified_files_count) | string | Returns the number of `other_modified_files` | | [renamed\_files](#output_renamed_files) | string | Returns only files that are
Renamed (R). | | [renamed\_files\_count](#output_renamed_files_count) | string | Returns the number of `renamed_files` | | [type\_changed\_files](#output_type_changed_files) | string | Returns only files that have
their file type changed (T). | | [type\_changed\_files\_count](#output_type_changed_files_count) | string | Returns the number of `type_changed_files` | | [unknown\_files](#output_unknown_files) | string | Returns only files that are
Unknown (X). | | [unknown\_files\_count](#output_unknown_files_count) | string | Returns the number of `unknown_files` | | [unmerged\_files](#output_unmerged_files) | string | Returns only files that are
Unmerged (U). | | [unmerged\_files\_count](#output_unmerged_files_count) | string | Returns the number of `unmerged_files` | ## Inputs โš™๏ธ | INPUT | TYPE | REQUIRED | DEFAULT | DESCRIPTION | |----------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------|----------|---------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | [api\_url](#input_api_url) | string | false | `"${{ github.api_url }}"` | Github API URL. | | [base\_sha](#input_base_sha) | string | false | | Specify a different base commit
SHA used for comparing changes | | [diff\_relative](#input_diff_relative) | string | false | `"true"` | Exclude changes outside the current
directory and show path names
relative to it. **NOTE:** This
requires you to specify the
top level directory via the
`path` input. | | [dir\_names](#input_dir_names) | string | false | `"false"` | Output unique changed directories instead
of filenames. **NOTE:** This returns
`.` for changed files located
in the current working directory
which defaults to `$GITHUB_WORKSPACE`. | | [dir\_names\_exclude\_current\_dir](#input_dir_names_exclude_current_dir) | string | false | `"false"` | Exclude the current directory represented
by `.` from the output
when `dir_names` is set to
`true`. | | [dir\_names\_include\_files](#input_dir_names_include_files) | string | false | | Include files in the output
when `dir_names` is set to
`true`. **NOTE:** This returns only
the matching files and also
the directory names. | | [dir\_names\_include\_files\_separator](#input_dir_names_include_files_separator) | string | false | `"\n"` | Separator used to split the
`dir_names_include_files` input | | [dir\_names\_max\_depth](#input_dir_names_max_depth) | string | false | | Limit the directory output to
a maximum depth e.g `test/test1/test2`
with max depth of `2`
returns `test/test1`. | | [escape\_json](#input_escape_json) | string | false | `"true"` | Escape JSON output. | | [fail\_on\_initial\_diff\_error](#input_fail_on_initial_diff_error) | string | false | `"false"` | Fail when the initial diff
fails. | | [fail\_on\_submodule\_diff\_error](#input_fail_on_submodule_diff_error) | string | false | `"false"` | Fail when the submodule diff
fails. | | [fetch\_additional\_submodule\_history](#input_fetch_additional_submodule_history) | string | false | `"false"` | Fetch additional history for submodules. | | [fetch\_depth](#input_fetch_depth) | string | false | `"50"` | Depth of additional branch history
fetched. **NOTE**: This can be
adjusted to resolve errors with
insufficient history. | | [files](#input_files) | string | false | | File and directory patterns used
to detect changes (Defaults to the entire repo if unset) **NOTE:**
Multiline file/directory patterns should not
include quotes. | | [files\_from\_source\_file](#input_files_from_source_file) | string | false | | Source file(s) used to populate
the `files` input. | | [files\_from\_source\_file\_separator](#input_files_from_source_file_separator) | string | false | `"\n"` | Separator used to split the
`files_from_source_file` input | | [files\_ignore](#input_files_ignore) | string | false | | Ignore changes to these file(s)
**NOTE:** Multiline file/directory patterns should
not include quotes. | | [files\_ignore\_from\_source\_file](#input_files_ignore_from_source_file) | string | false | | Source file(s) used to populate
the `files_ignore` input | | [files\_ignore\_from\_source\_file\_separator](#input_files_ignore_from_source_file_separator) | string | false | `"\n"` | Separator used to split the
`files_ignore_from_source_file` input | | [files\_ignore\_separator](#input_files_ignore_separator) | string | false | `"\n"` | Separator used to split the
`files_ignore` input | | [files\_ignore\_yaml](#input_files_ignore_yaml) | string | false | | YAML used to define a
set of file patterns to
ignore changes | | [files\_ignore\_yaml\_from\_source\_file](#input_files_ignore_yaml_from_source_file) | string | false | | Source file(s) used to populate
the `files_ignore_yaml` input. [Example](https://github.com/tj-actions/changed-files/blob/main/test/changed-files.yml) | | [files\_ignore\_yaml\_from\_source\_file\_separator](#input_files_ignore_yaml_from_source_file_separator) | string | false | `"\n"` | Separator used to split the
`files_ignore_yaml_from_source_file` input | | [files\_separator](#input_files_separator) | string | false | `"\n"` | Separator used to split the
`files` input | | [files\_yaml](#input_files_yaml) | string | false | | YAML used to define a
set of file patterns to
detect changes | | [files\_yaml\_from\_source\_file](#input_files_yaml_from_source_file) | string | false | | Source file(s) used to populate
the `files_yaml` input. [Example](https://github.com/tj-actions/changed-files/blob/main/test/changed-files.yml) | | [files\_yaml\_from\_source\_file\_separator](#input_files_yaml_from_source_file_separator) | string | false | `"\n"` | Separator used to split the
`files_yaml_from_source_file` input | | [include\_all\_old\_new\_renamed\_files](#input_include_all_old_new_renamed_files) | string | false | `"false"` | Include `all_old_new_renamed_files` output. Note this
can generate a large output
See: [#501](https://github.com/tj-actions/changed-files/issues/501). | | [json](#input_json) | string | false | `"false"` | Output list of changed files
in a JSON formatted string
which can be used for
matrix jobs. | | [old\_new\_files\_separator](#input_old_new_files_separator) | string | false | `" "` | Split character for old and
new renamed filename pairs. | | [old\_new\_separator](#input_old_new_separator) | string | false | `","` | Split character for old and
new filename pairs. | | [output\_dir](#input_output_dir) | string | false | `".github/outputs"` | Directory to store output files. | | [output\_renamed\_files\_as\_deleted\_and\_added](#input_output_renamed_files_as_deleted_and_added) | string | false | `"false"` | Output renamed files as deleted
and added files. | | [path](#input_path) | string | false | `"."` | Specify a relative path under
`$GITHUB_WORKSPACE` to locate the repository. | | [quotepath](#input_quotepath) | string | false | `"true"` | Use non-ascii characters to match
files and output the filenames
completely verbatim by setting this
to `false` | | [recover\_deleted\_files](#input_recover_deleted_files) | string | false | `"false"` | Recover deleted files. | | [recover\_deleted\_files\_to\_destination](#input_recover_deleted_files_to_destination) | string | false | | Recover deleted files to a
new destination directory, defaults to
the original location. | | [recover\_files](#input_recover_files) | string | false | | File and directory patterns used
to recover deleted files, defaults
to the patterns provided via
the `files`, `files_from_source_file`, `files_ignore` and
`files_ignore_from_source_file` inputs or all deleted
files if no patterns are
provided. | | [recover\_files\_ignore](#input_recover_files_ignore) | string | false | | File and directory patterns to
ignore when recovering deleted files. | | [recover\_files\_ignore\_separator](#input_recover_files_ignore_separator) | string | false | `"\n"` | Separator used to split the
`recover_files_ignore` input | | [recover\_files\_separator](#input_recover_files_separator) | string | false | `"\n"` | Separator used to split the
`recover_files` input | | [separator](#input_separator) | string | false | `" "` | Split character for output strings | | [sha](#input_sha) | string | false | | Specify a different commit SHA
used for comparing changes | | [since](#input_since) | string | false | | Get changed files for commits
whose timestamp is older than
the given time. | | [since\_last\_remote\_commit](#input_since_last_remote_commit) | string | false | `"false"` | Use the last commit on
the remote branch as the
`base_sha`. Defaults to the last
non-merge commit on the target
branch for pull request events
and the previous remote commit
of the current branch for
push events. | | [skip\_initial\_fetch](#input_skip_initial_fetch) | string | false | `"false"` | Skip the initial fetch to
improve performance for shallow repositories.
**NOTE**: This could lead to
errors with missing history and
the intended use is limited
to when you've fetched the
history necessary to perform the
diff. | | [token](#input_token) | string | false | `"${{ github.token }}"` | Github token used to fetch
changed files from Github's API. | | [until](#input_until) | string | false | | Get changed files for commits
whose timestamp is earlier than
the given time. | | [write\_output\_files](#input_write_output_files) | string | false | `"false"` | Write outputs to the `output_dir`
defaults to `.github/outputs` folder. **NOTE:**
This creates a `.txt` file
by default and a `.json`
file if `json` is set
to `true`. | ## Versioning ๐Ÿท๏ธ This GitHub Action follows the principles of [Semantic Versioning](https://semver.org) for versioning releases. The format of the version string is as follows: * major: indicates significant changes or new features that may not be backward compatible. * minor: indicates minor changes or new features that are backward compatible. * patch: indicates bug fixes or other small changes that are backward compatible. ## Examples ๐Ÿ“„
Get all changed files in the current branch ```yaml ... - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 ... ```
Get all changed files and use a comma separator ```yaml ... - name: Get all changed files and use a comma separator in the output id: changed-files uses: tj-actions/changed-files@v38 with: separator: "," ... ``` See [inputs](#inputs) for more information.
Get all changed files and list all added files ```yaml ... - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 - name: List all added files run: | for file in ${{ steps.changed-files.outputs.added_files }}; do echo "$file was added" done ... ``` See [outputs](#outputs) for a list of all available outputs.
Get all changed files and optionally run a step if a file was modified ```yaml ... - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 - name: Run a step if my-file.txt was modified if: contains(steps.changed-files.outputs.modified_files, 'my-file.txt') run: | echo "my-file.txt file has been modified." ... ``` See [outputs](#outputs) for a list of all available outputs.
Get all changed files and write the outputs to a txt file ```yaml ... - name: Get changed files and write the outputs to a Txt file id: changed-files-write-output-files-txt uses: ./ with: write_output_files: true - name: Verify the contents of the .github/outputs/added_files.txt file run: | cat .github/outputs/added_files.txt ... ```
Get all changed files and write the outputs to a json file ```yaml ... - name: Get changed files and write the outputs to a JSON file id: changed-files-write-output-files-json uses: ./ with: json: true write_output_files: true - name: Verify the contents of the .github/outputs/added_files.json file run: | cat .github/outputs/added_files.json ... ```
Get all changed files using a list of files ```yaml ... - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 with: files: | my-file.txt *.sh *.png !*.md test_directory **/*.sql ... ``` See [inputs](#inputs) for more information.
Get all changed files using a list of files and take action based on the changes ```yaml ... - name: Get changed files id: changed-files-specific uses: tj-actions/changed-files@v38 with: files: | my-file.txt *.sh *.png !*.md test_directory **/*.sql - name: Run step if any of the listed files above change if: steps.changed-files-specific.outputs.any_changed == 'true' run: | echo "One or more files listed above has changed." - name: Run step if only the files listed above change if: steps.changed-files-specific.outputs.only_changed == 'true' run: | echo "Only files listed above have changed." - name: Run step if any of the listed files above is deleted if: steps.changed-files-specific.outputs.any_deleted == 'true' run: | for file in ${{ steps.changed-files-specific.outputs.deleted_files }}; do echo "$file was deleted" done - name: Run step if all listed files above have been deleted if: steps.changed-files-specific.outputs.only_deleted == 'true' run: | for file in ${{ steps.changed-files-specific.outputs.deleted_files }}; do echo "$file was deleted" done ... ``` See [outputs](#outputs) for a list of all available outputs.
Get all changed files using a source file or list of file(s) to populate to files input ```yaml ... - name: Get changed files using a source file or list of file(s) to populate to files input. id: changed-files-specific-source-file uses: tj-actions/changed-files@v38 with: files_from_source_file: test/changed-files-list.txt ... ``` See [inputs](#inputs) for more information.
Get changed files using a source file or list of file(s) to populate to files input and optionally specify more files ```yaml ... - name: Get changed files using a source file or list of file(s) to populate to files input and optionally specify more files. id: changed-files-specific-source-file-and-specify-files uses: tj-actions/changed-files@v38 with: files_from_source_file: | test/changed-files-list.txt files: | test.txt ... ``` See [inputs](#inputs) for more information.
Get all changed files using a different SHA ```yaml ... - name: Get changed files using a different SHA id: changed-files uses: tj-actions/changed-files@v38 with: sha: ${{ github.event.pull_request.head.sha }} ... ``` See [inputs](#inputs) for more information.
Get all changed files using a different base SHA ```yaml ... - name: Get changed files using a different base SHA id: changed-files uses: tj-actions/changed-files@v38 with: base_sha: ${{ github.event.pull_request.base.sha }} ... ``` See [inputs](#inputs) for more information.
Get all changed files between the previous tag and the current tag ```yaml ... on: push: tags: - 'v*' jobs: release: name: Release runs-on: ubuntu-latest steps: - name: Checkout uses: actions/checkout@v3 with: fetch-depth: 0 - name: Get changed files id: changed-files uses: tj-actions/changed-files@v38 - name: Get changed files in the .github folder id: changed-files-specific uses: tj-actions/changed-files@v38 with: base_sha: ${{ steps.get-base-sha.outputs.base_sha }} files: .github/** - name: Run step if any file(s) in the .github folder change if: steps.changed-files-specific.outputs.any_changed == 'true' run: | echo "One or more files in the .github folder has changed." echo "List all the files that have changed: ${{ steps.changed-files-specific.outputs.all_changed_files }}" ... ``` See [inputs](#inputs) for more information.
Get all changed files for a repository located in a different path ```yaml ... - name: Checkout into dir1 uses: actions/checkout@v3 with: fetch-depth: 0 path: dir1 - name: Run changed-files with defaults in dir1 id: changed-files-for-dir1 uses: tj-actions/changed-files@v38 with: path: dir1 - name: List all added files in dir1 run: | for file in ${{ steps.changed-files-for-dir1.outputs.added_files }}; do echo "$file was added" done ... ``` See [inputs](#inputs) for more information.
Get all changed files with non-รคลกฤ‡ฤฏรญ characters i.e (Filename in other languages) ```yaml ... - name: Run changed-files with quotepath disabled id: changed-files-quotepath uses: tj-actions/changed-files@v38 with: quotepath: "false" - name: Run changed-files with quotepath disabled for a specified list of file(s) id: changed-files-quotepath-specific uses: ./ with: files: test/test-รจ.txt quotepath: "false" ... ``` See [inputs](#inputs) for more information.
Get all changed files using the last successful commit of the base branch > **Warning** > > This setting overrides the commit sha used by setting `since_last_remote_commit` to true. > It is recommended to use either solution that works for your use case. See [inputs](#inputs) for more information.
Get all changed files but only return the directory names ```yaml ... - name: Run changed-files with dir_names id: changed-files-dir-names uses: tj-actions/changed-files@v38 with: dir_names: "true" ... ``` See [inputs](#inputs) for more information.
Get all changed files and return JSON formatted outputs ```yaml ... - name: Run changed-files with JSON output id: changed-files-json uses: tj-actions/changed-files@v38 with: json: "true" ... ``` See [inputs](#inputs) for more information.
Get all changed files by commits pushed in the past ```yaml ... - name: Get changed-files since 2022-08-19 id: changed-files-since uses: tj-actions/changed-files@v38 with: since: "2022-08-19" - name: Get changed-files until 2022-08-20 id: changed-files-until uses: tj-actions/changed-files@v38 with: until: "2022-08-20" ... ``` See [inputs](#inputs) for more information.
## Real-world usage ๐ŸŒ ### Open source projects ๐Ÿ“ฆ * [vitejs/vite: uses tj-actions/changed-files to automate testing](https://github.com/vitejs/vite/blob/8da04227d6f818a8ad9efc0056101968037c2e36/.github/workflows/ci.yml#L61) * [qgis/QGIS: uses tj-actions/changed-files to automate spell checking](https://github.com/qgis/QGIS/blob/a5333497e90ac9de4ca70463d8e0b64c3f294d63/.github/workflows/code_layout.yml#L147) * [coder/code-server: uses tj-actions/changed-files to automate detecting changes and run steps based on the outcome](https://github.com/coder/code-server/blob/c32a31d802f679846876b8ad9aacff6cf7b5361d/.github/workflows/build.yaml#L48) * [tldr-pages/tldr: uses tj-actions/changed-files to automate detecting spelling errors](https://github.com/tldr-pages/tldr/blob/main/.github/workflows/codespell.yml#L14) * [nodejs/docker-node: uses tj-actions/changed-files to generate matrix jobs based on changes detected](https://github.com/nodejs/docker-node/blob/3c4fa6daf06a4786d202f2f610351837806a0380/.github/workflows/build-test.yml#L29) * [refined-github: uses tj-actions/changed-files to automate test URL validation in added/edited files](https://github.com/refined-github/refined-github/blob/b754bfe58904da8a599d7876fdaaf18302785629/.github/workflows/features.yml#L35) * [aws-doc-sdk-examples: uses tj-actions/changed-files to automate testing](https://github.com/awsdocs/aws-doc-sdk-examples/blob/2393723ef6b0cad9502f4852f5c72f7be58ca89d/.github/workflows/javascript.yml#L22) * [nhost: uses tj-actions/changed-files to automate testing based on changes detected](https://github.com/nhost/nhost/blob/main/.github/workflows/ci.yaml#L44-L48) * [qmk\_firmware uses tj-actions/changed-files to run linters](https://github.com/qmk/qmk_firmware/blob/7a737235ffd49c32d2c5561e8fe53fd96baa7f96/.github/workflows/lint.yml#L30) And many more... ### Scalability Example ๐Ÿ“ˆ ![image](https://github.com/tj-actions/changed-files/assets/17484350/23767413-4c51-42fb-ab1c-39ef72c44904) ## Known Limitation โš ๏ธ > **Warning** > > * Spaces in file names can introduce bugs when using bash loops. See: [#216](https://github.com/tj-actions/changed-files/issues/216) > However, this action will handle spaces in file names, with a recommendation of using a separator to prevent any hidden issues. > > ![Screen Shot 2021-10-23 at 9 37 34 AM](https://user-images.githubusercontent.com/17484350/138558767-b13c90bf-a1ae-4e86-9520-70a6a4624f41.png) ## Migration guide ๐Ÿ”„ With the switch from using grep's Extended regex to match files to the natively supported workflow glob pattern matching syntax introduced in [v13](https://github.com/tj-actions/changed-files/releases/tag/v13) you'll need to modify patterns used to match `files`. ```diff ... - name: Get specific changed files id: changed-files-specific uses: tj-actions/changed-files@v24 with: files: | - \.sh$ - .(sql|py)$ - ^(dir1|dir2) + **/*.{sh,sql,py} + {dir1,dir2}/** ``` * Free software: [MIT license](LICENSE) ## Credits ๐Ÿ‘ This package was created with [cookiecutter-action](https://github.com/tj-actions/cookiecutter-action). * [tj-actions/auto-doc](https://github.com/tj-actions/auto-doc) * [tj-actions/verify-changed-files](https://github.com/tj-actions/verify-changed-files) * [tj-actions/demo](https://github.com/tj-actions/demo) * [tj-actions/demo2](https://github.com/tj-actions/demo2) * [tj-actions/demo3](https://github.com/tj-actions/demo3) * [tj-actions/release-tagger](https://github.com/tj-actions/release-tagger) ## Report Bugs ๐Ÿ› Report bugs at https://github.com/tj-actions/changed-files/issues. If you are reporting a bug, please include: * Your operating system name and version. * Any details about your workflow that might be helpful in troubleshooting. (**NOTE**: Ensure that you include full log outputs with debugging enabled) * Detailed steps to reproduce the bug. ## Contributors โœจ Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
Josh Soref
Josh Soref

๐Ÿ“–
Nick Landers
Nick Landers

๐Ÿ’ป
Krasimir Nikolov
Krasimir Nikolov

๐Ÿ’ป ๐Ÿ“–
Ivan Pizhenko
Ivan Pizhenko

๐Ÿ’ป ๐Ÿ“–
talva-tr
talva-tr

๐Ÿ’ป
Ikko Ashimine
Ikko Ashimine

๐Ÿ“–
James
James

๐Ÿ“–
James Cheng
James Cheng

๐Ÿ“–
Masaya Suzuki
Masaya Suzuki

๐Ÿ’ป
fagai
fagai

๐Ÿ“–
Constantine Peresypkin
Constantine Peresypkin

๐Ÿ’ป
Mathieu Dupuy
Mathieu Dupuy

๐Ÿ“–
Joe Moggridge
Joe Moggridge

๐Ÿ“–
Charles Santos
Charles Santos

๐Ÿ’ป
Kostiantyn Korniienko
Kostiantyn Korniienko

๐Ÿ“–
Logan Pulley
Logan Pulley

๐Ÿ’ป
Kenji Miyake
Kenji Miyake

๐Ÿ’ป
adonisgarciac
adonisgarciac

๐Ÿ’ป ๐Ÿ“–
Chiel Fernhout
Chiel Fernhout

๐Ÿ“–
Alberto Perdomo
Alberto Perdomo

๐Ÿ“–
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!