-
Notifications
You must be signed in to change notification settings - Fork 39
154 lines (142 loc) · 6.23 KB
/
comment_build.yml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
name: Post build links to pull request
on:
workflow_run:
workflows: ['Create a test build']
types: [completed]
permissions:
actions: write
contents: write
pull-requests: write
jobs:
pr_comment:
if: github.event.workflow_run.event == 'pull_request' && github.event.workflow_run.conclusion == 'success'
runs-on: ubuntu-latest
steps:
- uses: actions/github-script@v7
with:
# This snippet is public-domain, combined from
# https://github.com/oprypin/nightly.link/blob/master/.github/workflows/pr-comment.yml
# https://github.com/AKSW/submission.d2r2.aksw.org/blob/main/.github/workflows/pr-comment.yml
script: |
// Function Definitions
/**
* Fetch PR details for a given commit SHA.
* @returns {Object} PR details containing prNumber, prRef, prRepoId.
* @throws {Error} If no matching PR is found.
*/
async function fetchPRDetails() {
const iterator = github.paginate.iterator(github.rest.pulls.list, {
owner: context.repo.owner,
repo: context.repo.repo,
});
for await (const { data } of iterator) {
for (const pull of data) {
if (pull.head.sha === '${{github.event.workflow_run.head_sha}}') {
return {
prNumber: pull.number,
prRef: pull.head.ref,
prRepoId: pull.head.repo.id
};
}
}
}
throw new Error("No matching PR found for the commit SHA");
}
/**
* Fetch all artifacts for a given workflow run.
* @returns {Object} All artifacts data.
* @throws {Error} If no artifacts are found.
*/
async function fetchAllArtifacts() {
const artifactsResponse = await github.rest.actions.listWorkflowRunArtifacts({
owner: context.repo.owner,
repo: context.repo.repo,
run_id: context.payload.workflow_run.id,
});
if (!(artifactsResponse.data && artifactsResponse.data.artifacts && artifactsResponse.data.artifacts.length)) {
throw new Error("No artifacts found for the workflow run");
}
return artifactsResponse.data.artifacts;
}
/**
* Create or update a comment on the PR.
* @param {number} prNumber - The PR number.
* @param {string} purpose - The purpose of the comment.
* @param {string} body - The comment body.
* @throws {Error} If the comment creation or update fails.
*/
async function upsertComment(prNumber, purpose, body) {
const { data: comments } = await github.rest.issues.listComments({
owner: context.repo.owner,
repo: context.repo.repo,
issue_number: prNumber,
});
const marker = `<!-- bot: ${purpose} -->`;
body = marker + "\n" + body;
const existing = comments.filter(c => c.body.includes(marker));
if (existing.length > 0) {
const last = existing[existing.length - 1];
core.info(`Updating comment ${last.id}`);
await github.rest.issues.updateComment({
owner: context.repo.owner,
repo: context.repo.repo,
body: body,
comment_id: last.id,
});
} else {
core.info(`Creating a comment in PR #${prNumber}`);
await github.rest.issues.createComment({
owner: context.repo.owner,
repo: context.repo.repo,
body: body,
issue_number: prNumber,
});
}
}
/**
* Handle and log errors with detailed context and exit the process.
* @param {Error} error - The error object.
* @param {string} description - Description of the context where the error occurred.
*/
function handleError(error, description, prNumber) {
core.error(`Failed to ${description}`);
core.error(`Message: ${error.message}`);
core.error(`Stack Trace: ${error.stack || 'No stack trace available'}`);
if (prNumber) {
core.error(`PR Number: ${prNumber}`);
}
core.error(`PRs: https://api.github.com/repos/${context.repo.owner}/${context.repo.repo}/pulls`);
core.error(`SHA: ${{github.event.workflow_run.head_sha}}`);
process.exit(1);
}
// Main Code Execution
let prNumber, prRef, prRepoId;
// Fetch PR details
try {
({ prNumber, prRef, prRepoId } = await fetchPRDetails());
core.info(`Found PR: #${prNumber}, Ref: ${prRef}, Repo ID: ${prRepoId}`);
} catch (error) {
handleError(error, 'fetch PR details', undefined);
}
// Fetch all artifacts
let allArtifacts;
try {
allArtifacts = await fetchAllArtifacts();
core.info(`Artifacts fetched successfully`);
} catch (error) {
handleError(error, 'fetch artifacts', prNumber);
}
// Construct the comment body
let body = 'Download the built assets for this pull request:\n' +
allArtifacts
.filter(item => item.name !== "assets")
.sort((a, b) => a.name.localeCompare(b.name))
.map(item => `* [${item.name}.zip](https://nightly.link/${context.repo.owner}/${context.repo.repo}/actions/artifacts/${item.id}.zip)`)
.join('\n');
// Upsert the comment on the PR
try {
await upsertComment(prNumber, "nightly-link", body);
core.info("Comment created/updated successfully");
} catch (error) {
handleError(error, 'create/update comment', prNumber);
}