1
+ # See https://github.com/backstage/backstage/blob/master/.github/workflows/sync_renovate-changesets.yml
2
+ name : Sync Renovate changeset
3
+ on :
4
+ pull_request_target :
5
+ paths :
6
+ - ' .github/workflows/sync_renovate-changesets.yml'
7
+ - ' **/pnpm-lock.yaml'
8
+
9
+ jobs :
10
+ generate-changeset :
11
+ permissions :
12
+ contents : write
13
+ runs-on : ubuntu-latest
14
+ if : github.actor == 'renovate[bot]' && github.repository == 'PostgreSQL-Typed/PostgreSQL-Typed'
15
+ steps :
16
+ - name : Checkout
17
+ uses : actions/checkout@v3
18
+ with :
19
+ fetch-depth : 2
20
+ ref : ${{ github.head_ref }}
21
+ - name : Configure Git
22
+ run : |
23
+ git config --global user.email [email protected]
24
+ git config --global user.name 'Github changeset workflow'
25
+ - name : Generate changeset
26
+ uses : actions/github-script@v6
27
+ with :
28
+ script : |
29
+ const { promises: fs } = require("fs");
30
+ // Parses package.json files and returns the package names
31
+ async function getPackagesNames(files) {
32
+ const names = [];
33
+ for (const file of files) {
34
+ const data = JSON.parse(await fs.readFile(file, "utf8"));
35
+ if (!data.private) {
36
+ names.push(data.name);
37
+ }
38
+ }
39
+ return names;
40
+ }
41
+
42
+ async function createChangeset(fileName, packageBumps, packages) {
43
+ let message = "";
44
+ for (const [pkg, bump] of packageBumps) {
45
+ message = message + `Updated dependency \`${pkg}\` to \`${bump}\`.\n`;
46
+ }
47
+
48
+ const pkgs = packages.map((pkg) => `'${pkg}': patch`).join("\n");
49
+ const body = `---\n${pkgs}\n---\n\n${message.trim()}\n`;
50
+ await fs.writeFile(fileName, body);
51
+ }
52
+
53
+ async function getBumps(files) {
54
+ const bumps = new Map();
55
+ for (const file of files) {
56
+ const { stdout: changes } = await exec.getExecOutput("git", [
57
+ "show",
58
+ file,
59
+ ]);
60
+ for (const change of changes.split("\n")) {
61
+ if (!change.startsWith("+ ")) {
62
+ continue;
63
+ }
64
+ const match = change.match(/"(.*?)"/g);
65
+ bumps.set(match[0].replace(/"/g, ""), match[1].replace(/"/g, ""));
66
+ }
67
+ }
68
+ return bumps;
69
+ }
70
+
71
+ const branch = await exec.getExecOutput("git branch --show-current");
72
+ if (!branch.stdout.startsWith("renovate/")) {
73
+ console.log("Not a renovate branch, skipping");
74
+ return;
75
+ }
76
+ const diffOutput = await exec.getExecOutput("git diff --name-only HEAD~1");
77
+ const diffFiles = diffOutput.stdout.split("\n");
78
+ if (diffFiles.find((f) => f.startsWith(".changeset"))) {
79
+ console.log("Changeset already exists, skipping");
80
+ return;
81
+ }
82
+ const files = diffFiles
83
+ .filter((file) => file !== "package.json") // skip root package.json
84
+ .filter((file) => file.includes("package.json"));
85
+ const packageNames = await getPackagesNames(files);
86
+ if (!packageNames.length) {
87
+ console.log("No package.json changes to published packages, skipping");
88
+ return;
89
+ }
90
+ const { stdout: shortHash } = await exec.getExecOutput(
91
+ "git rev-parse --short HEAD"
92
+ );
93
+ const fileName = `.changeset/renovate-${shortHash.trim()}.md`;
94
+
95
+ const packageBumps = await getBumps(files);
96
+ await createChangeset(fileName, packageBumps, packageNames);
97
+ await exec.exec("git", ["add", fileName]);
98
+ await exec.exec("git commit -C HEAD --amend --no-edit");
99
+ await exec.exec("git push --force");
0 commit comments