Skip to content

Commit 642fb77

Browse files
authored
Add base arg for commitChangesFromRepo (#13)
1 parent b127b6e commit 642fb77

File tree

7 files changed

+220
-62
lines changed

7 files changed

+220
-62
lines changed

.changeset/tiny-days-rescue.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@s0/ghcommit": minor
3+
---
4+
5+
Allow for base commit to be specified with commitChangesFromRepo

.github/workflows/ci.yaml

+2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@ jobs:
99
runs-on: ubuntu-latest
1010
steps:
1111
- uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7
12+
with:
13+
fetch-depth: 2
1214
- name: Use Node.js
1315
uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4.0.2
1416
with:

README.md

+27-2
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,14 @@ In addition to `CommitFilesBasedArgs`, this function has the following arguments
7474

7575
```ts
7676
{
77+
/**
78+
* The base commit to build your changes on-top of
79+
*
80+
* @default HEAD
81+
*/
82+
base?: {
83+
commit: string;
84+
};
7785
/**
7886
* The root of the repository.
7987
*
@@ -86,7 +94,7 @@ In addition to `CommitFilesBasedArgs`, this function has the following arguments
8694
Example:
8795

8896
```ts
89-
import { getOctokit } from "@actions/github";
97+
import { context, getOctokit } from "@actions/github";
9098
import { commitChangesFromRepo } from "@s0/ghcommit/git";
9199

92100
const octokit = getOctokit(process.env.GITHUB_TOKEN);
@@ -103,7 +111,7 @@ await commitChangesFromRepo({
103111
},
104112
});
105113

106-
// Commit & push the files from ta specific directory
114+
// Commit & push the files from a specific directory
107115
// where we've cloned a repo, and made changes to files
108116
await commitChangesFromRepo({
109117
octokit,
@@ -115,6 +123,23 @@ await commitChangesFromRepo({
115123
},
116124
repoDirectory: "/tmp/some-repo",
117125
});
126+
127+
// Commit & push the files from the current directory,
128+
// but ensure changes from any locally-made commits are also included
129+
await commitChangesFromRepo({
130+
octokit,
131+
owner: "my-org",
132+
repository: "my-repo",
133+
branch: "another-new-branch-to-create",
134+
message: {
135+
headline: "[chore] do something else",
136+
},
137+
base: {
138+
// This will be the original sha from the workflow run,
139+
// even if we've made commits locally
140+
commit: context.sha,
141+
},
142+
});
118143
```
119144

120145
### `commitFilesFromDirectory`

src/git.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -8,21 +8,23 @@ import {
88
} from "./interface";
99

1010
export const commitChangesFromRepo = async ({
11+
base,
1112
repoDirectory = process.cwd(),
1213
log,
1314
...otherArgs
1415
}: CommitChangesFromRepoArgs): Promise<CommitFilesResult> => {
16+
const ref = base?.commit ?? "HEAD";
1517
const gitLog = await git.log({
1618
fs,
1719
dir: repoDirectory,
18-
ref: "HEAD",
20+
ref,
1921
depth: 1,
2022
});
2123

2224
const oid = gitLog[0]?.oid;
2325

2426
if (!oid) {
25-
throw new Error("Could not determine oid for current branch");
27+
throw new Error(`Could not determine oid for ${ref}`);
2628
}
2729

2830
// Determine changed files

src/github/graphql/queries.ts

+5
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,11 @@ const GET_REF_TREE = /* GraphQL */ `
100100
tree {
101101
oid
102102
}
103+
parents(first: 10) {
104+
nodes {
105+
oid
106+
}
107+
}
103108
file(path: $path) {
104109
oid
105110
}

src/interface.ts

+18
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,24 @@ export interface CommitFilesFromDirectoryArgs
8484
}
8585

8686
export interface CommitChangesFromRepoArgs extends CommitFilesBasedArgs {
87+
/**
88+
* The base commit to build your changes on-top of.
89+
*
90+
* By default, this commit will be the HEAD of the local repository,
91+
* meaning that if any commits have been made locally and not pushed,
92+
* this command will fail.
93+
*
94+
* To include all changes, this should be set to a commit that is known
95+
* to be in the remote repository (such as the default branch).
96+
*
97+
* If you want to base the changes on a different commit to one checked out,
98+
* make sure that you also pull this commit from the remote.
99+
*
100+
* @default HEAD
101+
*/
102+
base?: {
103+
commit: string;
104+
};
87105
/**
88106
* The root of the repository.
89107
*

src/test/integration/git.test.ts

+159-58
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import { getOctokit } from "@actions/github";
1212
import { commitChangesFromRepo } from "../../git";
1313
import { getRefTreeQuery } from "../../github/graphql/queries";
1414
import { deleteBranches } from "./util";
15+
import git from "isomorphic-git";
1516

1617
const octokit = getOctokit(ENV.GITHUB_TOKEN);
1718

@@ -57,6 +58,96 @@ const expectBranchHasFile = async ({
5758
}
5859
};
5960

61+
const expectParentHasOid = async ({
62+
branch,
63+
oid,
64+
}: {
65+
branch: string;
66+
oid: string;
67+
}) => {
68+
const commit = (
69+
await getRefTreeQuery(octokit, {
70+
owner: REPO.owner,
71+
name: REPO.repository,
72+
ref: `refs/heads/${branch}`,
73+
path: "README.md",
74+
})
75+
).repository?.ref?.target;
76+
77+
if (!commit || !("parents" in commit)) {
78+
throw new Error("Expected commit to have a parent");
79+
}
80+
81+
expect(commit.parents.nodes).toEqual([{ oid }]);
82+
};
83+
84+
const makeFileChanges = async (repoDirectory: string) => {
85+
// Update an existing file
86+
await fs.promises.writeFile(
87+
path.join(repoDirectory, "LICENSE"),
88+
"This is a new license",
89+
);
90+
// Remove a file
91+
await fs.promises.rm(path.join(repoDirectory, "package.json"));
92+
// Remove a file nested in a directory
93+
await fs.promises.rm(path.join(repoDirectory, "src", "index.ts"));
94+
// Add a new file
95+
await fs.promises.writeFile(
96+
path.join(repoDirectory, "new-file.txt"),
97+
"This is a new file",
98+
);
99+
// Add a new file nested in a directory
100+
await fs.promises.mkdir(path.join(repoDirectory, "nested"), {
101+
recursive: true,
102+
});
103+
await fs.promises.writeFile(
104+
path.join(repoDirectory, "nested", "nested-file.txt"),
105+
"This is a nested file",
106+
);
107+
// Add files that should be ignored
108+
await fs.promises.writeFile(
109+
path.join(repoDirectory, ".env"),
110+
"This file should be ignored",
111+
);
112+
await fs.promises.mkdir(path.join(repoDirectory, "coverage", "foo"), {
113+
recursive: true,
114+
});
115+
await fs.promises.writeFile(
116+
path.join(repoDirectory, "coverage", "foo", "bar"),
117+
"This file should be ignored",
118+
);
119+
};
120+
121+
const makeFileChangeAssertions = async (branch: string) => {
122+
// Expect the deleted files to not exist
123+
await expectBranchHasFile({ branch, path: "package.json", oid: null });
124+
await expectBranchHasFile({ branch, path: "src/index.ts", oid: null });
125+
// Expect updated file to have new oid
126+
await expectBranchHasFile({
127+
branch,
128+
path: "LICENSE",
129+
oid: "8dd03bb8a1d83212f3667bd2eb8b92746120ab8f",
130+
});
131+
// Expect new files to have correct oid
132+
await expectBranchHasFile({
133+
branch,
134+
path: "new-file.txt",
135+
oid: "be5b944ff55ca7569cc2ae34c35b5bda8cd5d37e",
136+
});
137+
await expectBranchHasFile({
138+
branch,
139+
path: "nested/nested-file.txt",
140+
oid: "60eb5af9a0c03dc16dc6d0bd9a370c1aa4e095a3",
141+
});
142+
// Expect ignored files to not exist
143+
await expectBranchHasFile({ branch, path: ".env", oid: null });
144+
await expectBranchHasFile({
145+
branch,
146+
path: "coverage/foo/bar",
147+
oid: null,
148+
});
149+
};
150+
60151
describe("git", () => {
61152
const branches: string[] = [];
62153

@@ -72,7 +163,7 @@ describe("git", () => {
72163
await fs.promises.mkdir(testDir, { recursive: true });
73164
const repoDirectory = path.join(testDir, "repo-1");
74165

75-
// Clone the git repo locally usig the git cli and child-process
166+
// Clone the git repo locally using the git cli and child-process
76167
await new Promise<void>((resolve, reject) => {
77168
const p = exec(
78169
`git clone ${process.cwd()} repo-1`,
@@ -89,40 +180,7 @@ describe("git", () => {
89180
p.stderr?.pipe(process.stderr);
90181
});
91182

92-
// Update an existing file
93-
await fs.promises.writeFile(
94-
path.join(repoDirectory, "LICENSE"),
95-
"This is a new license",
96-
);
97-
// Remove a file
98-
await fs.promises.rm(path.join(repoDirectory, "package.json"));
99-
// Remove a file nested in a directory
100-
await fs.promises.rm(path.join(repoDirectory, "src", "index.ts"));
101-
// Add a new file
102-
await fs.promises.writeFile(
103-
path.join(repoDirectory, "new-file.txt"),
104-
"This is a new file",
105-
);
106-
// Add a new file nested in a directory
107-
await fs.promises.mkdir(path.join(repoDirectory, "nested"), {
108-
recursive: true,
109-
});
110-
await fs.promises.writeFile(
111-
path.join(repoDirectory, "nested", "nested-file.txt"),
112-
"This is a nested file",
113-
);
114-
// Add files that should be ignored
115-
await fs.promises.writeFile(
116-
path.join(repoDirectory, ".env"),
117-
"This file should be ignored",
118-
);
119-
await fs.promises.mkdir(path.join(repoDirectory, "coverage", "foo"), {
120-
recursive: true,
121-
});
122-
await fs.promises.writeFile(
123-
path.join(repoDirectory, "coverage", "foo", "bar"),
124-
"This file should be ignored",
125-
);
183+
await makeFileChanges(repoDirectory);
126184

127185
// Push the changes
128186
await commitChangesFromRepo({
@@ -137,33 +195,76 @@ describe("git", () => {
137195
log,
138196
});
139197

140-
// Expect the deleted files to not exist
141-
await expectBranchHasFile({ branch, path: "package.json", oid: null });
142-
await expectBranchHasFile({ branch, path: "src/index.ts", oid: null });
143-
// Expect updated file to have new oid
144-
await expectBranchHasFile({
145-
branch,
146-
path: "LICENSE",
147-
oid: "8dd03bb8a1d83212f3667bd2eb8b92746120ab8f",
148-
});
149-
// Expect new files to have correct oid
150-
await expectBranchHasFile({
151-
branch,
152-
path: "new-file.txt",
153-
oid: "be5b944ff55ca7569cc2ae34c35b5bda8cd5d37e",
198+
await makeFileChangeAssertions(branch);
199+
200+
// Expect the OID to be the HEAD commit
201+
const oid =
202+
(
203+
await git.log({
204+
fs,
205+
dir: repoDirectory,
206+
ref: "HEAD",
207+
depth: 1,
208+
})
209+
)[0]?.oid ?? "NO_OID";
210+
211+
await expectParentHasOid({ branch, oid });
212+
});
213+
214+
it("should correctly be able to base changes off specific commit", async () => {
215+
const branch = `${TEST_BRANCH_PREFIX}-specific-base`;
216+
217+
await fs.promises.mkdir(testDir, { recursive: true });
218+
const repoDirectory = path.join(testDir, "repo-2");
219+
220+
// Clone the git repo locally usig the git cli and child-process
221+
await new Promise<void>((resolve, reject) => {
222+
const p = exec(
223+
`git clone ${process.cwd()} repo-2`,
224+
{ cwd: testDir },
225+
(error) => {
226+
if (error) {
227+
reject(error);
228+
} else {
229+
resolve();
230+
}
231+
},
232+
);
233+
p.stdout?.pipe(process.stdout);
234+
p.stderr?.pipe(process.stderr);
154235
});
155-
await expectBranchHasFile({
156-
branch,
157-
path: "nested/nested-file.txt",
158-
oid: "60eb5af9a0c03dc16dc6d0bd9a370c1aa4e095a3",
236+
237+
makeFileChanges(repoDirectory);
238+
239+
// Determine the previous commit hash
240+
const gitLog = await git.log({
241+
fs,
242+
dir: repoDirectory,
243+
ref: "HEAD",
244+
depth: 2,
159245
});
160-
// Expect ignored files to not exist
161-
await expectBranchHasFile({ branch, path: ".env", oid: null });
162-
await expectBranchHasFile({
246+
247+
const oid = gitLog[1]?.oid ?? "";
248+
249+
// Push the changes
250+
await commitChangesFromRepo({
251+
octokit,
252+
...REPO,
163253
branch,
164-
path: "coverage/foo/bar",
165-
oid: null,
254+
message: {
255+
headline: "Test commit",
256+
body: "This is a test commit",
257+
},
258+
repoDirectory,
259+
log,
260+
base: {
261+
commit: oid,
262+
},
166263
});
264+
265+
await makeFileChangeAssertions(branch);
266+
267+
await expectParentHasOid({ branch, oid });
167268
});
168269
});
169270

0 commit comments

Comments
 (0)