feat: support checkout on a commit in addition to a ref
This commit is contained in:
parent
fc687f898b
commit
095c53659b
|
@ -1,4 +1,8 @@
|
||||||
import {createOrUpdateBranch, tryFetch} from '../lib/create-or-update-branch'
|
import {
|
||||||
|
createOrUpdateBranch,
|
||||||
|
tryFetch,
|
||||||
|
getWorkingBaseAndType
|
||||||
|
} from '../lib/create-or-update-branch'
|
||||||
import * as fs from 'fs'
|
import * as fs from 'fs'
|
||||||
import {GitCommandManager} from '../lib/git-command-manager'
|
import {GitCommandManager} from '../lib/git-command-manager'
|
||||||
import * as path from 'path'
|
import * as path from 'path'
|
||||||
|
@ -193,6 +197,21 @@ describe('create-or-update-branch tests', () => {
|
||||||
expect(await tryFetch(git, REMOTE_NAME, NOT_EXIST_BRANCH)).toBeFalsy()
|
expect(await tryFetch(git, REMOTE_NAME, NOT_EXIST_BRANCH)).toBeFalsy()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('tests getWorkingBaseAndType on a checked out ref', async () => {
|
||||||
|
const [workingBase, workingBaseType] = await getWorkingBaseAndType(git)
|
||||||
|
expect(workingBase).toEqual(BASE)
|
||||||
|
expect(workingBaseType).toEqual('branch')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('tests getWorkingBaseAndType on a checked out commit', async () => {
|
||||||
|
// Checkout the HEAD commit SHA
|
||||||
|
const headSha = await git.revParse('HEAD')
|
||||||
|
await git.exec(['checkout', headSha])
|
||||||
|
const [workingBase, workingBaseType] = await getWorkingBaseAndType(git)
|
||||||
|
expect(workingBase).toEqual(headSha)
|
||||||
|
expect(workingBaseType).toEqual('commit')
|
||||||
|
})
|
||||||
|
|
||||||
it('tests no changes resulting in no new branch being created', async () => {
|
it('tests no changes resulting in no new branch being created', async () => {
|
||||||
const commitMessage = uuidv4()
|
const commitMessage = uuidv4()
|
||||||
const result = await createOrUpdateBranch(
|
const result = await createOrUpdateBranch(
|
||||||
|
@ -1450,4 +1469,133 @@ describe('create-or-update-branch tests', () => {
|
||||||
await gitLogMatches([_commitMessage, INIT_COMMIT_MESSAGE])
|
await gitLogMatches([_commitMessage, INIT_COMMIT_MESSAGE])
|
||||||
).toBeTruthy()
|
).toBeTruthy()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Working Base is Not a Ref (WBNR)
|
||||||
|
// A commit is checked out leaving the repository in a "detached HEAD" state
|
||||||
|
|
||||||
|
it('tests create and update in detached HEAD state (WBNR)', async () => {
|
||||||
|
// Checkout the HEAD commit SHA
|
||||||
|
const headSha = await git.revParse('HEAD')
|
||||||
|
await git.checkout(headSha)
|
||||||
|
|
||||||
|
// Create tracked and untracked file changes
|
||||||
|
const changes = await createChanges()
|
||||||
|
const commitMessage = uuidv4()
|
||||||
|
const result = await createOrUpdateBranch(
|
||||||
|
git,
|
||||||
|
commitMessage,
|
||||||
|
BASE,
|
||||||
|
BRANCH,
|
||||||
|
REMOTE_NAME,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
expect(result.action).toEqual('created')
|
||||||
|
expect(await getFileContent(TRACKED_FILE)).toEqual(changes.tracked)
|
||||||
|
expect(await getFileContent(UNTRACKED_FILE)).toEqual(changes.untracked)
|
||||||
|
expect(
|
||||||
|
await gitLogMatches([commitMessage, INIT_COMMIT_MESSAGE])
|
||||||
|
).toBeTruthy()
|
||||||
|
|
||||||
|
// Push pull request branch to remote
|
||||||
|
await git.push([
|
||||||
|
'--force-with-lease',
|
||||||
|
REMOTE_NAME,
|
||||||
|
`HEAD:refs/heads/${BRANCH}`
|
||||||
|
])
|
||||||
|
|
||||||
|
await afterTest(false)
|
||||||
|
await beforeTest()
|
||||||
|
|
||||||
|
// Checkout the HEAD commit SHA
|
||||||
|
const _headSha = await git.revParse('HEAD')
|
||||||
|
await git.checkout(_headSha)
|
||||||
|
|
||||||
|
// Create tracked and untracked file changes
|
||||||
|
const _changes = await createChanges()
|
||||||
|
const _commitMessage = uuidv4()
|
||||||
|
const _result = await createOrUpdateBranch(
|
||||||
|
git,
|
||||||
|
_commitMessage,
|
||||||
|
BASE,
|
||||||
|
BRANCH,
|
||||||
|
REMOTE_NAME,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
expect(_result.action).toEqual('updated')
|
||||||
|
expect(_result.hasDiffWithBase).toBeTruthy()
|
||||||
|
expect(await getFileContent(TRACKED_FILE)).toEqual(_changes.tracked)
|
||||||
|
expect(await getFileContent(UNTRACKED_FILE)).toEqual(_changes.untracked)
|
||||||
|
expect(
|
||||||
|
await gitLogMatches([_commitMessage, INIT_COMMIT_MESSAGE])
|
||||||
|
).toBeTruthy()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('tests create and update with commits on the base inbetween, in detached HEAD state (WBNR)', async () => {
|
||||||
|
// Checkout the HEAD commit SHA
|
||||||
|
const headSha = await git.revParse('HEAD')
|
||||||
|
await git.checkout(headSha)
|
||||||
|
|
||||||
|
// Create tracked and untracked file changes
|
||||||
|
const changes = await createChanges()
|
||||||
|
const commitMessage = uuidv4()
|
||||||
|
const result = await createOrUpdateBranch(
|
||||||
|
git,
|
||||||
|
commitMessage,
|
||||||
|
BASE,
|
||||||
|
BRANCH,
|
||||||
|
REMOTE_NAME,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
expect(result.action).toEqual('created')
|
||||||
|
expect(await getFileContent(TRACKED_FILE)).toEqual(changes.tracked)
|
||||||
|
expect(await getFileContent(UNTRACKED_FILE)).toEqual(changes.untracked)
|
||||||
|
expect(
|
||||||
|
await gitLogMatches([commitMessage, INIT_COMMIT_MESSAGE])
|
||||||
|
).toBeTruthy()
|
||||||
|
|
||||||
|
// Push pull request branch to remote
|
||||||
|
await git.push([
|
||||||
|
'--force-with-lease',
|
||||||
|
REMOTE_NAME,
|
||||||
|
`HEAD:refs/heads/${BRANCH}`
|
||||||
|
])
|
||||||
|
|
||||||
|
await afterTest(false)
|
||||||
|
await beforeTest()
|
||||||
|
|
||||||
|
// Create commits on the base
|
||||||
|
const commitsOnBase = await createCommits(git)
|
||||||
|
await git.push([
|
||||||
|
'--force',
|
||||||
|
REMOTE_NAME,
|
||||||
|
`HEAD:refs/heads/${DEFAULT_BRANCH}`
|
||||||
|
])
|
||||||
|
|
||||||
|
// Checkout the HEAD commit SHA
|
||||||
|
const _headSha = await git.revParse('HEAD')
|
||||||
|
await git.checkout(_headSha)
|
||||||
|
|
||||||
|
// Create tracked and untracked file changes
|
||||||
|
const _changes = await createChanges()
|
||||||
|
const _commitMessage = uuidv4()
|
||||||
|
const _result = await createOrUpdateBranch(
|
||||||
|
git,
|
||||||
|
_commitMessage,
|
||||||
|
BASE,
|
||||||
|
BRANCH,
|
||||||
|
REMOTE_NAME,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
expect(_result.action).toEqual('updated')
|
||||||
|
expect(_result.hasDiffWithBase).toBeTruthy()
|
||||||
|
expect(await getFileContent(TRACKED_FILE)).toEqual(_changes.tracked)
|
||||||
|
expect(await getFileContent(UNTRACKED_FILE)).toEqual(_changes.untracked)
|
||||||
|
expect(
|
||||||
|
await gitLogMatches([
|
||||||
|
_commitMessage,
|
||||||
|
...commitsOnBase.commitMsgs,
|
||||||
|
INIT_COMMIT_MESSAGE
|
||||||
|
])
|
||||||
|
).toBeTruthy()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
59
dist/index.js
vendored
59
dist/index.js
vendored
|
@ -2932,10 +2932,30 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
exports.createOrUpdateBranch = exports.tryFetch = void 0;
|
exports.createOrUpdateBranch = exports.tryFetch = exports.getWorkingBaseAndType = exports.WorkingBaseType = void 0;
|
||||||
const core = __importStar(__webpack_require__(186));
|
const core = __importStar(__webpack_require__(186));
|
||||||
const uuid_1 = __webpack_require__(840);
|
const uuid_1 = __webpack_require__(840);
|
||||||
const CHERRYPICK_EMPTY = 'The previous cherry-pick is now empty, possibly due to conflict resolution.';
|
const CHERRYPICK_EMPTY = 'The previous cherry-pick is now empty, possibly due to conflict resolution.';
|
||||||
|
var WorkingBaseType;
|
||||||
|
(function (WorkingBaseType) {
|
||||||
|
WorkingBaseType["Branch"] = "branch";
|
||||||
|
WorkingBaseType["Commit"] = "commit";
|
||||||
|
})(WorkingBaseType = exports.WorkingBaseType || (exports.WorkingBaseType = {}));
|
||||||
|
function getWorkingBaseAndType(git) {
|
||||||
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
|
const symbolicRefResult = yield git.exec(['symbolic-ref', 'HEAD', '--short'], true);
|
||||||
|
if (symbolicRefResult.exitCode == 0) {
|
||||||
|
// A ref is checked out
|
||||||
|
return [symbolicRefResult.stdout.trim(), WorkingBaseType.Branch];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// A commit is checked out (detached HEAD)
|
||||||
|
const headSha = yield git.revParse('HEAD');
|
||||||
|
return [headSha, WorkingBaseType.Commit];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
exports.getWorkingBaseAndType = getWorkingBaseAndType;
|
||||||
function tryFetch(git, remote, branch) {
|
function tryFetch(git, remote, branch) {
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
try {
|
try {
|
||||||
|
@ -2983,8 +3003,14 @@ function splitLines(multilineString) {
|
||||||
}
|
}
|
||||||
function createOrUpdateBranch(git, commitMessage, base, branch, branchRemoteName, signoff) {
|
function createOrUpdateBranch(git, commitMessage, base, branch, branchRemoteName, signoff) {
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
// Get the working base. This may or may not be the actual base.
|
// Get the working base.
|
||||||
const workingBase = yield git.symbolicRef('HEAD', ['--short']);
|
// When a ref, it may or may not be the actual base.
|
||||||
|
// When a commit, we must rebase onto the actual base.
|
||||||
|
const [workingBase, workingBaseType] = yield getWorkingBaseAndType(git);
|
||||||
|
core.info(`Working base is ${workingBaseType} '${workingBase}'`);
|
||||||
|
if (workingBaseType == WorkingBaseType.Commit && !base) {
|
||||||
|
throw new Error(`When in 'detached HEAD' state, 'base' must be supplied.`);
|
||||||
|
}
|
||||||
// If the base is not specified it is assumed to be the working base.
|
// If the base is not specified it is assumed to be the working base.
|
||||||
base = base ? base : workingBase;
|
base = base ? base : workingBase;
|
||||||
const baseRemote = 'origin';
|
const baseRemote = 'origin';
|
||||||
|
@ -3009,10 +3035,14 @@ function createOrUpdateBranch(git, commitMessage, base, branch, branchRemoteName
|
||||||
}
|
}
|
||||||
// Perform fetch and reset the working base
|
// Perform fetch and reset the working base
|
||||||
// Commits made during the workflow will be removed
|
// Commits made during the workflow will be removed
|
||||||
yield git.fetch([`${workingBase}:${workingBase}`], baseRemote, ['--force']);
|
if (workingBaseType == WorkingBaseType.Branch) {
|
||||||
|
core.info(`Resetting working base branch '${workingBase}' to its remote`);
|
||||||
|
yield git.fetch([`${workingBase}:${workingBase}`], baseRemote, ['--force']);
|
||||||
|
}
|
||||||
// If the working base is not the base, rebase the temp branch commits
|
// If the working base is not the base, rebase the temp branch commits
|
||||||
|
// This will also be true if the working base type is a commit
|
||||||
if (workingBase != base) {
|
if (workingBase != base) {
|
||||||
core.info(`Rebasing commits made to branch '${workingBase}' on to base branch '${base}'`);
|
core.info(`Rebasing commits made to ${workingBaseType} '${workingBase}' on to base branch '${base}'`);
|
||||||
// Checkout the actual base
|
// Checkout the actual base
|
||||||
yield git.fetch([`${base}:${base}`], baseRemote, ['--force']);
|
yield git.fetch([`${base}:${base}`], baseRemote, ['--force']);
|
||||||
yield git.checkout(base);
|
yield git.checkout(base);
|
||||||
|
@ -6927,19 +6957,14 @@ function createPullRequest(inputs) {
|
||||||
yield gitAuthHelper.configureToken(inputs.token);
|
yield gitAuthHelper.configureToken(inputs.token);
|
||||||
core.endGroup();
|
core.endGroup();
|
||||||
}
|
}
|
||||||
// Determine if the checked out ref is a valid base for a pull request
|
core.startGroup('Checking the base repository state');
|
||||||
// The action needs the checked out HEAD ref to be a branch
|
const [workingBase, workingBaseType] = yield create_or_update_branch_1.getWorkingBaseAndType(git);
|
||||||
// This check will fail in the following cases:
|
core.info(`Working base is ${workingBaseType} '${workingBase}'`);
|
||||||
// - HEAD is detached
|
// When in detached HEAD state (checked out on a commit), we need to
|
||||||
// - HEAD is a merge commit (pull_request events)
|
// know the 'base' branch in order to rebase changes.
|
||||||
// - HEAD is a tag
|
if (workingBaseType == create_or_update_branch_1.WorkingBaseType.Commit && !inputs.base) {
|
||||||
core.startGroup('Checking the checked out ref');
|
throw new Error(`When the repository is checked out on a commit instead of a branch, the 'base' input must be supplied.`);
|
||||||
const symbolicRefResult = yield git.exec(['symbolic-ref', 'HEAD', '--short'], true);
|
|
||||||
if (symbolicRefResult.exitCode != 0) {
|
|
||||||
core.debug(`${symbolicRefResult.stderr}`);
|
|
||||||
throw new Error('The checked out ref is not a valid base for a pull request. Unable to continue.');
|
|
||||||
}
|
}
|
||||||
const workingBase = symbolicRefResult.stdout.trim();
|
|
||||||
// If the base is not specified it is assumed to be the working base.
|
// If the base is not specified it is assumed to be the working base.
|
||||||
const base = inputs.base ? inputs.base : workingBase;
|
const base = inputs.base ? inputs.base : workingBase;
|
||||||
// Throw an error if the base and branch are not different branches
|
// Throw an error if the base and branch are not different branches
|
||||||
|
|
|
@ -5,6 +5,28 @@ import {v4 as uuidv4} from 'uuid'
|
||||||
const CHERRYPICK_EMPTY =
|
const CHERRYPICK_EMPTY =
|
||||||
'The previous cherry-pick is now empty, possibly due to conflict resolution.'
|
'The previous cherry-pick is now empty, possibly due to conflict resolution.'
|
||||||
|
|
||||||
|
export enum WorkingBaseType {
|
||||||
|
Branch = 'branch',
|
||||||
|
Commit = 'commit'
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getWorkingBaseAndType(
|
||||||
|
git: GitCommandManager
|
||||||
|
): Promise<[string, WorkingBaseType]> {
|
||||||
|
const symbolicRefResult = await git.exec(
|
||||||
|
['symbolic-ref', 'HEAD', '--short'],
|
||||||
|
true
|
||||||
|
)
|
||||||
|
if (symbolicRefResult.exitCode == 0) {
|
||||||
|
// A ref is checked out
|
||||||
|
return [symbolicRefResult.stdout.trim(), WorkingBaseType.Branch]
|
||||||
|
} else {
|
||||||
|
// A commit is checked out (detached HEAD)
|
||||||
|
const headSha = await git.revParse('HEAD')
|
||||||
|
return [headSha, WorkingBaseType.Commit]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export async function tryFetch(
|
export async function tryFetch(
|
||||||
git: GitCommandManager,
|
git: GitCommandManager,
|
||||||
remote: string,
|
remote: string,
|
||||||
|
@ -80,8 +102,15 @@ export async function createOrUpdateBranch(
|
||||||
branchRemoteName: string,
|
branchRemoteName: string,
|
||||||
signoff: boolean
|
signoff: boolean
|
||||||
): Promise<CreateOrUpdateBranchResult> {
|
): Promise<CreateOrUpdateBranchResult> {
|
||||||
// Get the working base. This may or may not be the actual base.
|
// Get the working base.
|
||||||
const workingBase = await git.symbolicRef('HEAD', ['--short'])
|
// When a ref, it may or may not be the actual base.
|
||||||
|
// When a commit, we must rebase onto the actual base.
|
||||||
|
const [workingBase, workingBaseType] = await getWorkingBaseAndType(git)
|
||||||
|
core.info(`Working base is ${workingBaseType} '${workingBase}'`)
|
||||||
|
if (workingBaseType == WorkingBaseType.Commit && !base) {
|
||||||
|
throw new Error(`When in 'detached HEAD' state, 'base' must be supplied.`)
|
||||||
|
}
|
||||||
|
|
||||||
// If the base is not specified it is assumed to be the working base.
|
// If the base is not specified it is assumed to be the working base.
|
||||||
base = base ? base : workingBase
|
base = base ? base : workingBase
|
||||||
const baseRemote = 'origin'
|
const baseRemote = 'origin'
|
||||||
|
@ -109,12 +138,16 @@ export async function createOrUpdateBranch(
|
||||||
|
|
||||||
// Perform fetch and reset the working base
|
// Perform fetch and reset the working base
|
||||||
// Commits made during the workflow will be removed
|
// Commits made during the workflow will be removed
|
||||||
await git.fetch([`${workingBase}:${workingBase}`], baseRemote, ['--force'])
|
if (workingBaseType == WorkingBaseType.Branch) {
|
||||||
|
core.info(`Resetting working base branch '${workingBase}' to its remote`)
|
||||||
|
await git.fetch([`${workingBase}:${workingBase}`], baseRemote, ['--force'])
|
||||||
|
}
|
||||||
|
|
||||||
// If the working base is not the base, rebase the temp branch commits
|
// If the working base is not the base, rebase the temp branch commits
|
||||||
|
// This will also be true if the working base type is a commit
|
||||||
if (workingBase != base) {
|
if (workingBase != base) {
|
||||||
core.info(
|
core.info(
|
||||||
`Rebasing commits made to branch '${workingBase}' on to base branch '${base}'`
|
`Rebasing commits made to ${workingBaseType} '${workingBase}' on to base branch '${base}'`
|
||||||
)
|
)
|
||||||
// Checkout the actual base
|
// Checkout the actual base
|
||||||
await git.fetch([`${base}:${base}`], baseRemote, ['--force'])
|
await git.fetch([`${base}:${base}`], baseRemote, ['--force'])
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
import * as core from '@actions/core'
|
import * as core from '@actions/core'
|
||||||
import {createOrUpdateBranch} from './create-or-update-branch'
|
import {
|
||||||
|
createOrUpdateBranch,
|
||||||
|
getWorkingBaseAndType,
|
||||||
|
WorkingBaseType
|
||||||
|
} from './create-or-update-branch'
|
||||||
import {GitHubHelper} from './github-helper'
|
import {GitHubHelper} from './github-helper'
|
||||||
import {GitCommandManager} from './git-command-manager'
|
import {GitCommandManager} from './git-command-manager'
|
||||||
import {GitAuthHelper} from './git-auth-helper'
|
import {GitAuthHelper} from './git-auth-helper'
|
||||||
|
@ -81,24 +85,16 @@ export async function createPullRequest(inputs: Inputs): Promise<void> {
|
||||||
core.endGroup()
|
core.endGroup()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine if the checked out ref is a valid base for a pull request
|
core.startGroup('Checking the base repository state')
|
||||||
// The action needs the checked out HEAD ref to be a branch
|
const [workingBase, workingBaseType] = await getWorkingBaseAndType(git)
|
||||||
// This check will fail in the following cases:
|
core.info(`Working base is ${workingBaseType} '${workingBase}'`)
|
||||||
// - HEAD is detached
|
// When in detached HEAD state (checked out on a commit), we need to
|
||||||
// - HEAD is a merge commit (pull_request events)
|
// know the 'base' branch in order to rebase changes.
|
||||||
// - HEAD is a tag
|
if (workingBaseType == WorkingBaseType.Commit && !inputs.base) {
|
||||||
core.startGroup('Checking the checked out ref')
|
|
||||||
const symbolicRefResult = await git.exec(
|
|
||||||
['symbolic-ref', 'HEAD', '--short'],
|
|
||||||
true
|
|
||||||
)
|
|
||||||
if (symbolicRefResult.exitCode != 0) {
|
|
||||||
core.debug(`${symbolicRefResult.stderr}`)
|
|
||||||
throw new Error(
|
throw new Error(
|
||||||
'The checked out ref is not a valid base for a pull request. Unable to continue.'
|
`When the repository is checked out on a commit instead of a branch, the 'base' input must be supplied.`
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
const workingBase = symbolicRefResult.stdout.trim()
|
|
||||||
// If the base is not specified it is assumed to be the working base.
|
// If the base is not specified it is assumed to be the working base.
|
||||||
const base = inputs.base ? inputs.base : workingBase
|
const base = inputs.base ? inputs.base : workingBase
|
||||||
// Throw an error if the base and branch are not different branches
|
// Throw an error if the base and branch are not different branches
|
||||||
|
|
Loading…
Reference in a new issue