
* In old version release run small package of most important tests * add to releases tests for stocks, warehouses and orders * Update .github/workflows/tests-nightly.yml Co-authored-by: Mika <6186720+NyanKiyoshi@users.noreply.github.com> * remove set-output command * Fix set tags step * Fix for url in release Co-authored-by: Mika <6186720+NyanKiyoshi@users.noreply.github.com>
257 lines
7.5 KiB
JavaScript
257 lines
7.5 KiB
JavaScript
const { Octokit } = require("@octokit/core");
|
|
const { Command } = require("commander");
|
|
const { GraphQLClient } = require("graphql-request");
|
|
|
|
const program = new Command();
|
|
const client = new GraphQLClient("https://dashboard.cypress.io/graphql");
|
|
|
|
const repo = "saleor-cloud-deployments";
|
|
const owner = "saleor";
|
|
|
|
program
|
|
.name("Approve PR")
|
|
.description("Approve and merge PR if patch release")
|
|
.option("--version <version>", "version of a project")
|
|
.option("--pull_request_number <pull_request_number>", "Pull Request number")
|
|
.option("--auto_release <auto_release>", "is auto release")
|
|
.option("--dashboard_url <dashboard_url>", "Cypress dashboard url")
|
|
.action(async options => {
|
|
const octokit = new Octokit({
|
|
auth: process.env.GITHUB_TOKEN,
|
|
});
|
|
|
|
const pullNumber = options.pull_request_number;
|
|
|
|
const pullRequest = await octokit.request(
|
|
"GET /repos/{owner}/{repo}/pulls/{pull_number}",
|
|
{
|
|
owner,
|
|
repo,
|
|
pull_number: pullNumber,
|
|
},
|
|
);
|
|
|
|
const commitId = pullRequest.data.merge_commit_sha;
|
|
|
|
const data = await getTestsStatusAndId(options.dashboard_url);
|
|
|
|
let testsStatus = data.status;
|
|
|
|
let requestBody = `Cypress tests passed. See results at ${options.dashboard_url}`;
|
|
|
|
if (testsStatus === "FAILED") {
|
|
const failedNewTests = [];
|
|
const listOfTestIssues = await getListOfTestsIssues(octokit);
|
|
const testCases = await getFailedTestCases(data.runId);
|
|
testCases.forEach(testCase => {
|
|
if (testCase.titleParts) {
|
|
const issue = issueOnGithub(listOfTestIssues, testCase.titleParts[1]);
|
|
if (issue) {
|
|
const knownBug = isIssueAKnownBugForReleaseVersion(
|
|
issue,
|
|
options.version,
|
|
);
|
|
if (!knownBug) {
|
|
failedNewTests.push({
|
|
title: testCase.titleParts[1],
|
|
url: issue.html_url,
|
|
spec: testCase.titleParts[0],
|
|
});
|
|
}
|
|
} else {
|
|
failedNewTests.push({
|
|
title: testCase.titleParts[1],
|
|
spec: testCase.titleParts[0],
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
if (failedNewTests.length === 0) {
|
|
requestBody = `All failed tests are known bugs, can be merged. See results at ${options.dashboard_url}`;
|
|
testsStatus = "PASSED";
|
|
} else if (failedNewTests.length > 10) {
|
|
//If there are more than 10 new bugs it's probably caused by something else. Server responses with 500, or test user was deleted, etc.
|
|
|
|
requestBody =
|
|
"There is more than 10 new bugs, check results manually and create issues for them if necessary";
|
|
} else {
|
|
requestBody = `New bugs found, results at: ${options.dashboard_url}. List of issues to check: `;
|
|
for (const newBug of failedNewTests) {
|
|
if (!newBug.url) {
|
|
const issueUrl = await createIssue(
|
|
newBug,
|
|
options.version,
|
|
octokit,
|
|
);
|
|
requestBody += `\n${newBug.title} - ${issueUrl}`;
|
|
} else {
|
|
requestBody += `\n${newBug.title} - ${newBug.url}`;
|
|
}
|
|
}
|
|
requestBody += `\nIf this bugs won't be fixed in next patch release for this version mark them as known issues`;
|
|
}
|
|
}
|
|
|
|
const event = "COMMENT";
|
|
|
|
await octokit.request(
|
|
"POST /repos/{owner}/{repo}/pulls/{pull_number}/reviews",
|
|
{
|
|
owner,
|
|
repo,
|
|
pull_number: pullNumber,
|
|
commit_id: commitId,
|
|
body: requestBody,
|
|
event,
|
|
comments: [],
|
|
},
|
|
);
|
|
|
|
if (
|
|
options.auto_release &&
|
|
isPatchRelease(options.version) &&
|
|
testsStatus === "PASSED"
|
|
) {
|
|
await octokit.request(
|
|
"PUT /repos/{owner}/{repo}/pulls/{pull_number}/merge",
|
|
{
|
|
owner,
|
|
repo,
|
|
pull_number: pullNumber,
|
|
merge_method: "squash",
|
|
},
|
|
);
|
|
}
|
|
})
|
|
.parse();
|
|
|
|
function isPatchRelease(version) {
|
|
const regex = /\d+\.\d+\.[1-9]/;
|
|
return version.match(regex) ? true : false;
|
|
}
|
|
|
|
async function getTestsStatusAndId(dashboardUrl) {
|
|
const getProjectRegex = /\/projects\/([^\/]*)/;
|
|
const getRunRegex = /\/runs\/([^\/]*)/;
|
|
|
|
const requestVariables = {
|
|
projectId: dashboardUrl.match(getProjectRegex)[1],
|
|
buildNumber: dashboardUrl.match(getRunRegex)[1],
|
|
};
|
|
|
|
const throwErrorAfterTimeout = setTimeout(function() {
|
|
throw new Error("Run have still running status, after all tests executed");
|
|
}, 1200000);
|
|
|
|
const data = await waitForTestsToFinish(requestVariables);
|
|
|
|
clearTimeout(throwErrorAfterTimeout);
|
|
return { status: data.status, runId: data.id };
|
|
}
|
|
|
|
async function waitForTestsToFinish(requestVariables) {
|
|
return new Promise((resolve, reject) => {
|
|
client
|
|
.request(
|
|
`query ($projectId: String!, $buildNumber: ID!) {
|
|
runByBuildNumber(buildNumber: $buildNumber, projectId: $projectId) {
|
|
status,
|
|
id
|
|
}
|
|
}`,
|
|
requestVariables,
|
|
)
|
|
.then(response => {
|
|
if (response.runByBuildNumber.status === "RUNNING") {
|
|
setTimeout(async function() {
|
|
resolve(await waitForTestsToFinish(requestVariables));
|
|
}, 10000);
|
|
} else {
|
|
resolve(response.runByBuildNumber);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
async function getFailedTestCases(runId) {
|
|
const requestVariables = {
|
|
input: {
|
|
runId,
|
|
testResultState: ["FAILED"],
|
|
},
|
|
};
|
|
|
|
return new Promise((resolve, reject) => {
|
|
client
|
|
.request(
|
|
`query RunTestResults($input: TestResultsTableInput!) {
|
|
testResults(input: $input) {
|
|
... on TestResult {
|
|
...RunTestResult
|
|
}
|
|
}
|
|
}
|
|
fragment RunTestResult on TestResult { id titleParts state}`,
|
|
requestVariables,
|
|
)
|
|
.then(response => {
|
|
resolve(response.testResults);
|
|
});
|
|
});
|
|
}
|
|
|
|
async function getListOfTestsIssues(octokit) {
|
|
const result = await octokit.request(
|
|
"GET /repos/{owner}/saleor-dashboard/issues?labels=tests",
|
|
{
|
|
owner,
|
|
},
|
|
);
|
|
return result.data;
|
|
}
|
|
|
|
function issueOnGithub(listOfTestIssues, testCaseTitle) {
|
|
if (listOfTestIssues.length > 0) {
|
|
return listOfTestIssues.find(issue => {
|
|
return issue.title.includes(testCaseTitle);
|
|
});
|
|
}
|
|
}
|
|
|
|
function isIssueAKnownBugForReleaseVersion(issue, releaseVersion) {
|
|
const issueBody = issue.body;
|
|
const regex = /Known bug for versions:([\s\S]*)Additional/;
|
|
const lines = issueBody.match(regex)[1].split("\n");
|
|
const lineContainReleaseVersionRegex = /v(\d{2,3}).*(true|false)/;
|
|
const releaseVersionLine = lines.find(line => {
|
|
if (line.match(lineContainReleaseVersionRegex)) {
|
|
const version = line.match(lineContainReleaseVersionRegex)[1];
|
|
if (version === getFormattedVersion(releaseVersion)) {
|
|
return line;
|
|
}
|
|
}
|
|
});
|
|
const knownBugOnReleaseVersion = releaseVersionLine
|
|
? releaseVersionLine.match(lineContainReleaseVersionRegex)[2]
|
|
: false;
|
|
return knownBugOnReleaseVersion === "true" ? true : false;
|
|
}
|
|
|
|
function getFormattedVersion(version) {
|
|
const regex = /^\d+\.\d+\./;
|
|
return version.match(regex)[0].replace(/\./g, "");
|
|
}
|
|
|
|
async function createIssue(newBug, version, octokit) {
|
|
const issue = await octokit.request("POST /repos/{owner}/{repo}/issues", {
|
|
owner,
|
|
repo: "saleor-dashboard",
|
|
title: `Cypress test fail: ${newBug.title}`,
|
|
body: `**Known bug for versions:**\nv${getFormattedVersion(
|
|
version,
|
|
)}: false\n**Additional Info:**\nSpec: ${newBug.spec}`,
|
|
labels: ["tests"],
|
|
});
|
|
return issue.data.html_url;
|
|
}
|