summaryrefslogtreecommitdiff
path: root/.github/workflows/android-merge.js
diff options
context:
space:
mode:
Diffstat (limited to '.github/workflows/android-merge.js')
-rw-r--r--.github/workflows/android-merge.js318
1 files changed, 0 insertions, 318 deletions
diff --git a/.github/workflows/android-merge.js b/.github/workflows/android-merge.js
deleted file mode 100644
index 315f81ba0..000000000
--- a/.github/workflows/android-merge.js
+++ /dev/null
@@ -1,318 +0,0 @@
-// SPDX-FileCopyrightText: 2023 yuzu Emulator Project
-// SPDX-License-Identifier: GPL-2.0-or-later
-
-// Note: This is a GitHub Actions script
-// It is not meant to be executed directly on your machine without modifications
-
-const fs = require("fs");
-// which label to check for changes
-const CHANGE_LABEL_MAINLINE = 'android-merge';
-const CHANGE_LABEL_EA = 'android-ea-merge';
-// how far back in time should we consider the changes are "recent"? (default: 24 hours)
-const DETECTION_TIME_FRAME = (parseInt(process.env.DETECTION_TIME_FRAME)) || (24 * 3600 * 1000);
-const BUILD_EA = process.env.BUILD_EA == 'true';
-const MAINLINE_TAG = process.env.MAINLINE_TAG;
-
-async function checkBaseChanges(github) {
- // query the commit date of the latest commit on this branch
- const query = `query($owner:String!, $name:String!, $ref:String!) {
- repository(name:$name, owner:$owner) {
- ref(qualifiedName:$ref) {
- target {
- ... on Commit { id pushedDate oid }
- }
- }
- }
- }`;
- const variables = {
- owner: 'yuzu-emu',
- name: 'yuzu',
- ref: 'refs/heads/master',
- };
- const result = await github.graphql(query, variables);
- const pushedAt = result.repository.ref.target.pushedDate;
- console.log(`Last commit pushed at ${pushedAt}.`);
- const delta = new Date() - new Date(pushedAt);
- if (delta <= DETECTION_TIME_FRAME) {
- console.info('New changes detected, triggering a new build.');
- return true;
- }
- console.info('No new changes detected.');
- return false;
-}
-
-async function checkAndroidChanges(github) {
- if (checkBaseChanges(github)) return true;
- const pulls = getPulls(github, false);
- for (let i = 0; i < pulls.length; i++) {
- let pull = pulls[i];
- if (new Date() - new Date(pull.headRepository.pushedAt) <= DETECTION_TIME_FRAME) {
- console.info(`${pull.number} updated at ${pull.headRepository.pushedAt}`);
- return true;
- }
- }
- console.info("No changes detected in any tagged pull requests.");
- return false;
-}
-
-async function tagAndPush(github, owner, repo, execa, commit=false) {
- let altToken = process.env.ALT_GITHUB_TOKEN;
- if (!altToken) {
- throw `Please set ALT_GITHUB_TOKEN environment variable. This token should have write access to ${owner}/${repo}.`;
- }
- const query = `query ($owner:String!, $name:String!) {
- repository(name:$name, owner:$owner) {
- refs(refPrefix: "refs/tags/", orderBy: {field: TAG_COMMIT_DATE, direction: DESC}, first: 10) {
- nodes { name }
- }
- }
- }`;
- const variables = {
- owner: owner,
- name: repo,
- };
- const tags = await github.graphql(query, variables);
- const tagList = tags.repository.refs.nodes;
- let lastTag = 'android-1';
- for (let i = 0; i < tagList.length; ++i) {
- if (tagList[i].name.includes('android-')) {
- lastTag = tagList[i].name;
- break;
- }
- }
- const tagNumber = /\w+-(\d+)/.exec(lastTag)[1] | 0;
- const channel = repo.split('-')[1];
- const newTag = `${channel}-${tagNumber + 1}`;
- console.log(`New tag: ${newTag}`);
- if (commit) {
- let channelName = channel[0].toUpperCase() + channel.slice(1);
- console.info(`Committing pending commit as ${channelName} ${tagNumber + 1}`);
- await execa("git", ['commit', '-m', `${channelName} ${tagNumber + 1}`]);
- }
- console.info('Pushing tags to GitHub ...');
- await execa("git", ['tag', newTag]);
- await execa("git", ['remote', 'add', 'target', `https://${altToken}@github.com/${owner}/${repo}.git`]);
- await execa("git", ['push', 'target', 'master', '-f']);
- await execa("git", ['push', 'target', 'master', '--tags']);
- console.info('Successfully pushed new changes.');
-}
-
-async function tagAndPushEA(github, owner, repo, execa) {
- let altToken = process.env.ALT_GITHUB_TOKEN;
- if (!altToken) {
- throw `Please set ALT_GITHUB_TOKEN environment variable. This token should have write access to ${owner}/${repo}.`;
- }
- const query = `query ($owner:String!, $name:String!) {
- repository(name:$name, owner:$owner) {
- refs(refPrefix: "refs/tags/", orderBy: {field: TAG_COMMIT_DATE, direction: DESC}, first: 10) {
- nodes { name }
- }
- }
- }`;
- const variables = {
- owner: owner,
- name: repo,
- };
- const tags = await github.graphql(query, variables);
- const tagList = tags.repository.refs.nodes;
- let lastTag = 'ea-1';
- for (let i = 0; i < tagList.length; ++i) {
- if (tagList[i].name.includes('ea-')) {
- lastTag = tagList[i].name;
- break;
- }
- }
- const tagNumber = /\w+-(\d+)/.exec(lastTag)[1] | 0;
- const newTag = `ea-${tagNumber + 1}`;
- console.log(`New tag: ${newTag}`);
- console.info('Pushing tags to GitHub ...');
- await execa("git", ["remote", "add", "android", "https://github.com/yuzu-emu/yuzu-android.git"]);
- await execa("git", ["fetch", "android"]);
-
- await execa("git", ['tag', newTag]);
- await execa("git", ['push', 'android', `${newTag}`]);
-
- fs.writeFile('tag-name.txt', newTag, (err) => {
- if (err) throw 'Could not write tag name to file!'
- })
-
- console.info('Successfully pushed new changes.');
-}
-
-async function generateReadme(pulls, context, mergeResults, execa) {
- let baseUrl = `https://github.com/${context.repo.owner}/${context.repo.repo}/`;
- let output =
- "| Pull Request | Commit | Title | Author | Merged? |\n|----|----|----|----|----|\n";
- for (let pull of pulls) {
- let pr = pull.number;
- let result = mergeResults[pr];
- output += `| [${pr}](${baseUrl}/pull/${pr}) | [\`${result.rev || "N/A"}\`](${baseUrl}/pull/${pr}/files) | ${pull.title} | [${pull.author.login}](https://github.com/${pull.author.login}/) | ${result.success ? "Yes" : "No"} |\n`;
- }
- output +=
- "\n\nEnd of merge log. You can find the original README.md below the break.\n\n-----\n\n";
- output += fs.readFileSync("./README.md");
- fs.writeFileSync("./README.md", output);
- await execa("git", ["add", "README.md"]);
-}
-
-async function fetchPullRequests(pulls, repoUrl, execa) {
- console.log("::group::Fetch pull requests");
- for (let pull of pulls) {
- let pr = pull.number;
- console.info(`Fetching PR ${pr} ...`);
- await execa("git", [
- "fetch",
- "-f",
- "--no-recurse-submodules",
- repoUrl,
- `pull/${pr}/head:pr-${pr}`,
- ]);
- }
- console.log("::endgroup::");
-}
-
-async function mergePullRequests(pulls, execa) {
- let mergeResults = {};
- console.log("::group::Merge pull requests");
- await execa("git", ["config", "--global", "user.name", "yuzubot"]);
- await execa("git", [
- "config",
- "--global",
- "user.email",
- "yuzu\x40yuzu-emu\x2eorg", // prevent email harvesters from scraping the address
- ]);
- let hasFailed = false;
- for (let pull of pulls) {
- let pr = pull.number;
- console.info(`Merging PR ${pr} ...`);
- try {
- const process1 = execa("git", [
- "merge",
- "--squash",
- "--no-edit",
- `pr-${pr}`,
- ]);
- process1.stdout.pipe(process.stdout);
- await process1;
-
- const process2 = execa("git", ["commit", "-m", `Merge yuzu-emu#${pr}`]);
- process2.stdout.pipe(process.stdout);
- await process2;
-
- const process3 = await execa("git", ["rev-parse", "--short", `pr-${pr}`]);
- mergeResults[pr] = {
- success: true,
- rev: process3.stdout,
- };
- } catch (err) {
- console.log(
- `::error title=#${pr} not merged::Failed to merge pull request: ${pr}: ${err}`
- );
- mergeResults[pr] = { success: false };
- hasFailed = true;
- await execa("git", ["reset", "--hard"]);
- }
- }
- console.log("::endgroup::");
- if (hasFailed) {
- throw 'There are merge failures. Aborting!';
- }
- return mergeResults;
-}
-
-async function resetBranch(execa) {
- console.log("::group::Reset master branch");
- let hasFailed = false;
- try {
- await execa("git", ["remote", "add", "source", "https://github.com/yuzu-emu/yuzu.git"]);
- await execa("git", ["fetch", "source"]);
- const process1 = await execa("git", ["rev-parse", "source/master"]);
- const headCommit = process1.stdout;
-
- await execa("git", ["reset", "--hard", headCommit]);
- } catch (err) {
- console.log(`::error title=Failed to reset master branch`);
- hasFailed = true;
- }
- console.log("::endgroup::");
- if (hasFailed) {
- throw 'Failed to reset the master branch. Aborting!';
- }
-}
-
-async function getPulls(github) {
- const query = `query ($owner:String!, $name:String!, $label:String!) {
- repository(name:$name, owner:$owner) {
- pullRequests(labels: [$label], states: OPEN, first: 100) {
- nodes {
- number title author { login }
- }
- }
- }
- }`;
- const mainlineVariables = {
- owner: 'yuzu-emu',
- name: 'yuzu',
- label: CHANGE_LABEL_MAINLINE,
- };
- const mainlineResult = await github.graphql(query, mainlineVariables);
- const pulls = mainlineResult.repository.pullRequests.nodes;
- if (BUILD_EA) {
- const eaVariables = {
- owner: 'yuzu-emu',
- name: 'yuzu',
- label: CHANGE_LABEL_EA,
- };
- const eaResult = await github.graphql(query, eaVariables);
- const eaPulls = eaResult.repository.pullRequests.nodes;
- return pulls.concat(eaPulls);
- }
- return pulls;
-}
-
-async function getMainlineTag(execa) {
- console.log(`::group::Getting mainline tag android-${MAINLINE_TAG}`);
- let hasFailed = false;
- try {
- await execa("git", ["remote", "add", "mainline", "https://github.com/yuzu-emu/yuzu-android.git"]);
- await execa("git", ["fetch", "mainline", "--tags"]);
- await execa("git", ["checkout", `tags/android-${MAINLINE_TAG}`]);
- await execa("git", ["submodule", "update", "--init", "--recursive"]);
- } catch (err) {
- console.log('::error title=Failed pull tag');
- hasFailed = true;
- }
- console.log("::endgroup::");
- if (hasFailed) {
- throw 'Failed pull mainline tag. Aborting!';
- }
-}
-
-async function mergebot(github, context, execa) {
- // Reset our local copy of master to what appears on yuzu-emu/yuzu - master
- await resetBranch(execa);
-
- const pulls = await getPulls(github);
- let displayList = [];
- for (let i = 0; i < pulls.length; i++) {
- let pull = pulls[i];
- displayList.push({ PR: pull.number, Title: pull.title });
- }
- console.info("The following pull requests will be merged:");
- console.table(displayList);
- await fetchPullRequests(pulls, "https://github.com/yuzu-emu/yuzu", execa);
- const mergeResults = await mergePullRequests(pulls, execa);
-
- if (BUILD_EA) {
- await tagAndPushEA(github, 'yuzu-emu', `yuzu-android`, execa);
- } else {
- await generateReadme(pulls, context, mergeResults, execa);
- await tagAndPush(github, 'yuzu-emu', `yuzu-android`, execa, true);
- }
-}
-
-module.exports.mergebot = mergebot;
-module.exports.checkAndroidChanges = checkAndroidChanges;
-module.exports.tagAndPush = tagAndPush;
-module.exports.checkBaseChanges = checkBaseChanges;
-module.exports.getMainlineTag = getMainlineTag;