diff --git a/.github/workflows/pull-request-verification.yml b/.github/workflows/pull-request-verification.yml index 27627a94..b18d7ba2 100644 --- a/.github/workflows/pull-request-verification.yml +++ b/.github/workflows/pull-request-verification.yml @@ -29,6 +29,9 @@ jobs: - name: filter-test if: steps.filter.outputs.any != 'true' || steps.filter.outputs.error == 'true' run: exit 1 + - name: changes-test + if: contains(fromJSON(steps.filter.outputs.changes), 'error') || !contains(fromJSON(steps.filter.outputs.changes), 'any') + run: exit 1 test-external: runs-on: ubuntu-latest diff --git a/README.md b/README.md index 145ecc08..448a7425 100644 --- a/README.md +++ b/README.md @@ -58,7 +58,7 @@ For more scenarios see [examples](#examples) section. ## Notes: - Paths expressions are evaluated using [picomatch](https://github.com/micromatch/picomatch) library. Documentation for path expression format can be found on project github page. -- Micromatch [dot](https://github.com/micromatch/picomatch#options) option is set to true. +- Picomatch [dot](https://github.com/micromatch/picomatch#options) option is set to true. Globbing will match also paths where file or folder name starts with a dot. - It's recommended to quote your path expressions with `'` or `"`. Otherwise you will get an error if it starts with `*`. - Local execution with [act](https://github.com/nektos/act) works only with alternative runner image. Default runner doesn't have `git` binary. @@ -66,6 +66,7 @@ For more scenarios see [examples](#examples) section. # What's New +- Configure matrix job to run for each folder with changes using `changes` output - Improved listing of matching files with `list-files: shell` and `list-files: escape` options - Support local changes - Fixed retrieval of all changes via Github API when there are 100+ changes @@ -122,9 +123,9 @@ For more information see [CHANGELOG](https://github.com/actions/checkout/blob/ma # Enables listing of files matching the filter: # 'none' - Disables listing of matching files (default). # 'json' - Matching files paths are formatted as JSON array. - # 'shell' - Space delimited list usable as command line argument list in linux shell. + # 'shell' - Space delimited list usable as command line argument list in Linux shell. # If needed it uses single or double quotes to wrap filename with unsafe characters. - # 'escape'- Space delimited list usable as command line argument list in linux shell. + # 'escape'- Space delimited list usable as command line argument list in Linux shell. # Backslash escapes every potentially unsafe character. # Default: none list-files: '' @@ -147,6 +148,7 @@ For more information see [CHANGELOG](https://github.com/actions/checkout/blob/ma - `'true'` - if **any** of changed files matches any of filter rules - `'false'` - if **none** of changed files matches any of filter rules - If enabled, for each filter it sets output variable with name `${FILTER_NAME}_files`. It will contain list of all files matching the filter. +- `changes` - JSON array with names of all filters matching any of changed files. # Examples @@ -230,6 +232,41 @@ jobs: ``` +
+Use change detection to configure matrix job + +```yaml +jobs: + # JOB to run change detection + changes: + runs-on: ubuntu-latest + outputs: + # Expose matched filters as job 'packages' output variable + packages: ${{ steps.filter.outputs.changes }} + steps: + # For pull requests it's not necessary to checkout the code + - uses: dorny/paths-filter@v2 + id: filter + with: + filters: | + package1: src/package1 + package2: src/package2 + + # JOB to build and test each of modified packages + build: + needs: changes + strategy: + matrix: + # Parse JSON array containing names of all filters matching any of changed files + # e.g. ['package1', 'package2'] if both package folders contains changes + package: ${{ fromJson(needs.changes.outputs.packages) }} + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - ... +``` +
+ ## Change detection workflows
@@ -406,7 +443,7 @@ jobs: # Enable listing of files matching each filter. # Paths to files will be available in `${FILTER_NAME}_files` output variable. # Paths will be escaped and space-delimited. - # Output is usable as command line argument list in linux shell + # Output is usable as command line argument list in Linux shell list-files: shell # In this example changed files will be checked by linter. diff --git a/action.yml b/action.yml index d1321d69..41c15ba9 100644 --- a/action.yml +++ b/action.yml @@ -37,6 +37,9 @@ inputs: This option takes effect only when changes are detected using git against different base branch. required: false default: '10' +outputs: + changes: + description: JSON array with names of all filters matching any of changed files runs: using: 'node12' main: 'dist/index.js' diff --git a/dist/index.js b/dist/index.js index 953026bf..3b17e87e 100644 --- a/dist/index.js +++ b/dist/index.js @@ -4791,10 +4791,12 @@ async function getChangedFilesFromApi(token, pullRequest) { } function exportResults(results, format) { core.info('Results:'); + const changes = []; for (const [key, files] of Object.entries(results)) { const value = files.length > 0; core.startGroup(`Filter ${key} = ${value}`); if (files.length > 0) { + changes.push(key); core.info('Matching files:'); for (const file of files) { core.info(`${file.filename} [${file.status}]`); @@ -4809,6 +4811,14 @@ function exportResults(results, format) { core.setOutput(`${key}_files`, filesValue); } } + if (results['changes'] === undefined) { + const changesJson = JSON.stringify(changes); + core.info(`Changes output set to ${changesJson}`); + core.setOutput('changes', changesJson); + } + else { + core.info('Cannot set changes output variable - name already used by filter output'); + } core.endGroup(); } function serializeExport(files, format) { diff --git a/src/main.ts b/src/main.ts index 176dafc9..e2ccb65a 100644 --- a/src/main.ts +++ b/src/main.ts @@ -175,10 +175,12 @@ async function getChangedFilesFromApi( function exportResults(results: FilterResults, format: ExportFormat): void { core.info('Results:') + const changes = [] for (const [key, files] of Object.entries(results)) { const value = files.length > 0 core.startGroup(`Filter ${key} = ${value}`) if (files.length > 0) { + changes.push(key) core.info('Matching files:') for (const file of files) { core.info(`${file.filename} [${file.status}]`) @@ -193,6 +195,14 @@ function exportResults(results: FilterResults, format: ExportFormat): void { core.setOutput(`${key}_files`, filesValue) } } + + if (results['changes'] === undefined) { + const changesJson = JSON.stringify(changes) + core.info(`Changes output set to ${changesJson}`) + core.setOutput('changes', changesJson) + } else { + core.info('Cannot set changes output variable - name already used by filter output') + } core.endGroup() }