Skip to content

Commit 87375a4

Browse files
authored
Merge pull request #84 from dorny/v2.10
Version v2.10.0
2 parents 208adf4 + 24a7483 commit 87375a4

File tree

7 files changed

+179
-117
lines changed

7 files changed

+179
-117
lines changed

.github/workflows/pull-request-verification.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ on:
44
paths-ignore: [ '*.md' ]
55
branches:
66
- master
7-
- develop
7+
- '**'
88

99
jobs:
1010
build:

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
11
# Changelog
22

3+
## v2.10.0
4+
- [Add ref input parameter](https://github.com/dorny/paths-filter/pull/82)
5+
- [Fix change detection in PR when pullRequest.changed_files is incorrect](https://github.com/dorny/paths-filter/pull/83)
6+
37
## v2.9.3
48
- [Fix change detection when base is a tag](https://github.com/dorny/paths-filter/pull/78)
59

README.md

+8-3
Original file line numberDiff line numberDiff line change
@@ -66,13 +66,11 @@ For more scenarios see [examples](#examples) section.
6666

6767

6868
# What's New
69+
- Add `ref` input parameter
6970
- Add `list-files: csv` format
7071
- Configure matrix job to run for each folder with changes using `changes` output
7172
- Improved listing of matching files with `list-files: shell` and `list-files: escape` options
72-
- Support local changes
73-
- Fixed retrieval of all changes via Github API when there are 100+ changes
7473
- Paths expressions are now evaluated using [picomatch](https://github.com/micromatch/picomatch) library
75-
- Support workflows triggered by any event
7674

7775
For more information see [CHANGELOG](https://github.com/dorny/paths-filter/blob/master/CHANGELOG.md)
7876

@@ -111,6 +109,13 @@ For more information see [CHANGELOG](https://github.com/dorny/paths-filter/blob/
111109
# Default: repository default branch (e.g. master)
112110
base: ''
113111
112+
# Git reference (e.g. branch name) from which the changes will be detected.
113+
# Useful when workflow can be triggered only on default branch (e.g. repository_dispatch event)
114+
# but you want to get changes on different branch.
115+
# This option is ignored if action is triggered by pull_request event.
116+
# default: ${{ github.ref }}
117+
ref:
118+
114119
# How many commits are initially fetched from base branch.
115120
# If needed, each subsequent fetch doubles the
116121
# previously requested number of commits until the merge-base

action.yml

+6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,12 @@ inputs:
99
working-directory:
1010
description: 'Relative path under $GITHUB_WORKSPACE where the repository was checked out.'
1111
required: false
12+
ref:
13+
description: |
14+
Git reference (e.g. branch name) from which the changes will be detected.
15+
This option is ignored if action is triggered by pull_request event.
16+
default: ${{ github.ref }}
17+
required: false
1218
base:
1319
description: |
1420
Git reference (e.g. branch name) against which the changes will be detected. Defaults to repository default branch (e.g. master).

dist/index.js

+78-55
Original file line numberDiff line numberDiff line change
@@ -3865,34 +3865,44 @@ async function getChangesOnHead() {
38653865
return parseGitDiffOutput(output);
38663866
}
38673867
exports.getChangesOnHead = getChangesOnHead;
3868-
async function getChangesSinceMergeBase(base, ref, initialFetchDepth) {
3868+
async function getChangesSinceMergeBase(base, head, initialFetchDepth) {
38693869
let baseRef;
3870+
let headRef;
38703871
async function hasMergeBase() {
3871-
return (baseRef !== undefined && (await exec_1.default('git', ['merge-base', baseRef, ref], { ignoreReturnCode: true })).code === 0);
3872+
if (baseRef === undefined || headRef === undefined) {
3873+
return false;
3874+
}
3875+
return (await exec_1.default('git', ['merge-base', baseRef, headRef], { ignoreReturnCode: true })).code === 0;
38723876
}
38733877
let noMergeBase = false;
3874-
core.startGroup(`Searching for merge-base ${base}...${ref}`);
3878+
core.startGroup(`Searching for merge-base ${base}...${head}`);
38753879
try {
38763880
baseRef = await getFullRef(base);
3881+
headRef = await getFullRef(head);
38773882
if (!(await hasMergeBase())) {
3878-
await exec_1.default('git', ['fetch', '--no-tags', `--depth=${initialFetchDepth}`, 'origin', base, ref]);
3879-
if (baseRef === undefined) {
3880-
baseRef = await getFullRef(base);
3881-
if (baseRef === undefined) {
3882-
await exec_1.default('git', ['fetch', '--tags', '--depth=1', 'origin', base, ref], {
3883+
await exec_1.default('git', ['fetch', '--no-tags', `--depth=${initialFetchDepth}`, 'origin', base, head]);
3884+
if (baseRef === undefined || headRef === undefined) {
3885+
baseRef = baseRef !== null && baseRef !== void 0 ? baseRef : (await getFullRef(base));
3886+
headRef = headRef !== null && headRef !== void 0 ? headRef : (await getFullRef(head));
3887+
if (baseRef === undefined || headRef === undefined) {
3888+
await exec_1.default('git', ['fetch', '--tags', '--depth=1', 'origin', base, head], {
38833889
ignoreReturnCode: true // returns exit code 1 if tags on remote were updated - we can safely ignore it
38843890
});
3885-
baseRef = await getFullRef(base);
3891+
baseRef = baseRef !== null && baseRef !== void 0 ? baseRef : (await getFullRef(base));
3892+
headRef = headRef !== null && headRef !== void 0 ? headRef : (await getFullRef(head));
38863893
if (baseRef === undefined) {
38873894
throw new Error(`Could not determine what is ${base} - fetch works but it's not a branch or tag`);
38883895
}
3896+
if (headRef === undefined) {
3897+
throw new Error(`Could not determine what is ${head} - fetch works but it's not a branch or tag`);
3898+
}
38893899
}
38903900
}
38913901
let depth = initialFetchDepth;
38923902
let lastCommitCount = await getCommitCount();
38933903
while (!(await hasMergeBase())) {
38943904
depth = Math.min(depth * 2, Number.MAX_SAFE_INTEGER);
3895-
await exec_1.default('git', ['fetch', `--deepen=${depth}`, 'origin', base, ref]);
3905+
await exec_1.default('git', ['fetch', `--deepen=${depth}`, 'origin', base, head]);
38963906
const commitCount = await getCommitCount();
38973907
if (commitCount === lastCommitCount) {
38983908
core.info('No more commits were fetched');
@@ -3910,16 +3920,16 @@ async function getChangesSinceMergeBase(base, ref, initialFetchDepth) {
39103920
finally {
39113921
core.endGroup();
39123922
}
3913-
let diffArg = `${baseRef}...${ref}`;
3923+
// Three dots '...' change detection - finds merge-base and compares against it
3924+
let diffArg = `${baseRef}...${headRef}`;
39143925
if (noMergeBase) {
39153926
core.warning('No merge base found - change detection will use direct <commit>..<commit> comparison');
3916-
diffArg = `${baseRef}..${ref}`;
3927+
diffArg = `${baseRef}..${headRef}`;
39173928
}
39183929
// Get changes introduced on ref compared to base
39193930
core.startGroup(`Change detection ${diffArg}`);
39203931
let output = '';
39213932
try {
3922-
// Three dots '...' change detection - finds merge-base and compares against it
39233933
output = (await exec_1.default('git', ['diff', '--no-renames', '--name-status', '-z', diffArg])).stdout;
39243934
}
39253935
finally {
@@ -4690,6 +4700,7 @@ async function run() {
46904700
process.chdir(workingDirectory);
46914701
}
46924702
const token = core.getInput('token', { required: false });
4703+
const ref = core.getInput('ref', { required: false });
46934704
const base = core.getInput('base', { required: false });
46944705
const filtersInput = core.getInput('filters', { required: true });
46954706
const filtersYaml = isPathInput(filtersInput) ? getConfigFileContent(filtersInput) : filtersInput;
@@ -4700,7 +4711,8 @@ async function run() {
47004711
return;
47014712
}
47024713
const filter = new filter_1.Filter(filtersYaml);
4703-
const files = await getChangedFiles(token, base, initialFetchDepth);
4714+
const files = await getChangedFiles(token, base, ref, initialFetchDepth);
4715+
core.info(`Detected ${files.length} changed files`);
47044716
const results = filter.match(files);
47054717
exportResults(results, listFiles);
47064718
}
@@ -4720,7 +4732,7 @@ function getConfigFileContent(configPath) {
47204732
}
47214733
return fs.readFileSync(configPath, { encoding: 'utf8' });
47224734
}
4723-
async function getChangedFiles(token, base, initialFetchDepth) {
4735+
async function getChangedFiles(token, base, ref, initialFetchDepth) {
47244736
// if base is 'HEAD' only local uncommitted changes will be detected
47254737
// This is the simplest case as we don't need to fetch more commits or evaluate current/before refs
47264738
if (base === git.HEAD) {
@@ -4735,14 +4747,14 @@ async function getChangedFiles(token, base, initialFetchDepth) {
47354747
return await git.getChangesInLastCommit();
47364748
}
47374749
else {
4738-
return getChangedFilesFromGit(base, initialFetchDepth);
4750+
return getChangedFilesFromGit(base, ref, initialFetchDepth);
47394751
}
47404752
}
4741-
async function getChangedFilesFromGit(base, initialFetchDepth) {
4753+
async function getChangedFilesFromGit(base, head, initialFetchDepth) {
47424754
var _a;
47434755
const defaultRef = (_a = github.context.payload.repository) === null || _a === void 0 ? void 0 : _a.default_branch;
47444756
const beforeSha = github.context.eventName === 'push' ? github.context.payload.before : null;
4745-
const ref = git.getShortName(github.context.ref) ||
4757+
const ref = git.getShortName(head || github.context.ref) ||
47464758
(core.warning(`'ref' field is missing in event payload - using current branch, tag or commit SHA`),
47474759
await git.getCurrentRef());
47484760
const baseRef = git.getShortName(base) || defaultRef;
@@ -4781,47 +4793,58 @@ async function getChangedFilesFromGit(base, initialFetchDepth) {
47814793
// Uses github REST api to get list of files changed in PR
47824794
async function getChangedFilesFromApi(token, pullRequest) {
47834795
core.startGroup(`Fetching list of changed files for PR#${pullRequest.number} from Github API`);
4784-
core.info(`Number of changed_files is ${pullRequest.changed_files}`);
4785-
const client = new github.GitHub(token);
4786-
const pageSize = 100;
4787-
const files = [];
4788-
for (let page = 1; (page - 1) * pageSize < pullRequest.changed_files; page++) {
4789-
core.info(`Invoking listFiles(pull_number: ${pullRequest.number}, page: ${page}, per_page: ${pageSize})`);
4790-
const response = await client.pulls.listFiles({
4791-
owner: github.context.repo.owner,
4792-
repo: github.context.repo.repo,
4793-
pull_number: pullRequest.number,
4794-
page,
4795-
per_page: pageSize
4796-
});
4797-
for (const row of response.data) {
4798-
core.info(`[${row.status}] ${row.filename}`);
4799-
// There's no obvious use-case for detection of renames
4800-
// Therefore we treat it as if rename detection in git diff was turned off.
4801-
// Rename is replaced by delete of original filename and add of new filename
4802-
if (row.status === file_1.ChangeStatus.Renamed) {
4803-
files.push({
4804-
filename: row.filename,
4805-
status: file_1.ChangeStatus.Added
4806-
});
4807-
files.push({
4808-
// 'previous_filename' for some unknown reason isn't in the type definition or documentation
4809-
filename: row.previous_filename,
4810-
status: file_1.ChangeStatus.Deleted
4811-
});
4796+
try {
4797+
const client = new github.GitHub(token);
4798+
const per_page = 100;
4799+
const files = [];
4800+
for (let page = 1;; page++) {
4801+
core.info(`Invoking listFiles(pull_number: ${pullRequest.number}, page: ${page}, per_page: ${per_page})`);
4802+
const response = await client.pulls.listFiles({
4803+
owner: github.context.repo.owner,
4804+
repo: github.context.repo.repo,
4805+
pull_number: pullRequest.number,
4806+
per_page,
4807+
page
4808+
});
4809+
if (response.status !== 200) {
4810+
throw new Error(`Fetching list of changed files from GitHub API failed with error code ${response.status}`);
48124811
}
4813-
else {
4814-
// Github status and git status variants are same except for deleted files
4815-
const status = row.status === 'removed' ? file_1.ChangeStatus.Deleted : row.status;
4816-
files.push({
4817-
filename: row.filename,
4818-
status
4819-
});
4812+
core.info(`Received ${response.data.length} items`);
4813+
if (response.data.length === 0) {
4814+
core.info('All changed files has been fetched from GitHub API');
4815+
break;
4816+
}
4817+
for (const row of response.data) {
4818+
core.info(`[${row.status}] ${row.filename}`);
4819+
// There's no obvious use-case for detection of renames
4820+
// Therefore we treat it as if rename detection in git diff was turned off.
4821+
// Rename is replaced by delete of original filename and add of new filename
4822+
if (row.status === file_1.ChangeStatus.Renamed) {
4823+
files.push({
4824+
filename: row.filename,
4825+
status: file_1.ChangeStatus.Added
4826+
});
4827+
files.push({
4828+
// 'previous_filename' for some unknown reason isn't in the type definition or documentation
4829+
filename: row.previous_filename,
4830+
status: file_1.ChangeStatus.Deleted
4831+
});
4832+
}
4833+
else {
4834+
// Github status and git status variants are same except for deleted files
4835+
const status = row.status === 'removed' ? file_1.ChangeStatus.Deleted : row.status;
4836+
files.push({
4837+
filename: row.filename,
4838+
status
4839+
});
4840+
}
48204841
}
48214842
}
4843+
return files;
4844+
}
4845+
finally {
4846+
core.endGroup();
48224847
}
4823-
core.endGroup();
4824-
return files;
48254848
}
48264849
function exportResults(results, format) {
48274850
core.info('Results:');

src/git.ts

+23-15
Original file line numberDiff line numberDiff line change
@@ -54,38 +54,46 @@ export async function getChangesOnHead(): Promise<File[]> {
5454
return parseGitDiffOutput(output)
5555
}
5656

57-
export async function getChangesSinceMergeBase(base: string, ref: string, initialFetchDepth: number): Promise<File[]> {
57+
export async function getChangesSinceMergeBase(base: string, head: string, initialFetchDepth: number): Promise<File[]> {
5858
let baseRef: string | undefined
59+
let headRef: string | undefined
5960
async function hasMergeBase(): Promise<boolean> {
60-
return (
61-
baseRef !== undefined && (await exec('git', ['merge-base', baseRef, ref], {ignoreReturnCode: true})).code === 0
62-
)
61+
if (baseRef === undefined || headRef === undefined) {
62+
return false
63+
}
64+
return (await exec('git', ['merge-base', baseRef, headRef], {ignoreReturnCode: true})).code === 0
6365
}
6466

6567
let noMergeBase = false
66-
core.startGroup(`Searching for merge-base ${base}...${ref}`)
68+
core.startGroup(`Searching for merge-base ${base}...${head}`)
6769
try {
6870
baseRef = await getFullRef(base)
71+
headRef = await getFullRef(head)
6972
if (!(await hasMergeBase())) {
70-
await exec('git', ['fetch', '--no-tags', `--depth=${initialFetchDepth}`, 'origin', base, ref])
71-
if (baseRef === undefined) {
72-
baseRef = await getFullRef(base)
73-
if (baseRef === undefined) {
74-
await exec('git', ['fetch', '--tags', '--depth=1', 'origin', base, ref], {
73+
await exec('git', ['fetch', '--no-tags', `--depth=${initialFetchDepth}`, 'origin', base, head])
74+
if (baseRef === undefined || headRef === undefined) {
75+
baseRef = baseRef ?? (await getFullRef(base))
76+
headRef = headRef ?? (await getFullRef(head))
77+
if (baseRef === undefined || headRef === undefined) {
78+
await exec('git', ['fetch', '--tags', '--depth=1', 'origin', base, head], {
7579
ignoreReturnCode: true // returns exit code 1 if tags on remote were updated - we can safely ignore it
7680
})
77-
baseRef = await getFullRef(base)
81+
baseRef = baseRef ?? (await getFullRef(base))
82+
headRef = headRef ?? (await getFullRef(head))
7883
if (baseRef === undefined) {
7984
throw new Error(`Could not determine what is ${base} - fetch works but it's not a branch or tag`)
8085
}
86+
if (headRef === undefined) {
87+
throw new Error(`Could not determine what is ${head} - fetch works but it's not a branch or tag`)
88+
}
8189
}
8290
}
8391

8492
let depth = initialFetchDepth
8593
let lastCommitCount = await getCommitCount()
8694
while (!(await hasMergeBase())) {
8795
depth = Math.min(depth * 2, Number.MAX_SAFE_INTEGER)
88-
await exec('git', ['fetch', `--deepen=${depth}`, 'origin', base, ref])
96+
await exec('git', ['fetch', `--deepen=${depth}`, 'origin', base, head])
8997
const commitCount = await getCommitCount()
9098
if (commitCount === lastCommitCount) {
9199
core.info('No more commits were fetched')
@@ -103,17 +111,17 @@ export async function getChangesSinceMergeBase(base: string, ref: string, initia
103111
core.endGroup()
104112
}
105113

106-
let diffArg = `${baseRef}...${ref}`
114+
// Three dots '...' change detection - finds merge-base and compares against it
115+
let diffArg = `${baseRef}...${headRef}`
107116
if (noMergeBase) {
108117
core.warning('No merge base found - change detection will use direct <commit>..<commit> comparison')
109-
diffArg = `${baseRef}..${ref}`
118+
diffArg = `${baseRef}..${headRef}`
110119
}
111120

112121
// Get changes introduced on ref compared to base
113122
core.startGroup(`Change detection ${diffArg}`)
114123
let output = ''
115124
try {
116-
// Three dots '...' change detection - finds merge-base and compares against it
117125
output = (await exec('git', ['diff', '--no-renames', '--name-status', '-z', diffArg])).stdout
118126
} finally {
119127
fixStdOutNullTermination()

0 commit comments

Comments
 (0)