PromucFlow_constructor/app/client/cypress/scripts/cypress-split-static.ts
Sagar Khalasi 828c6aeefe
chore: Adding new command for running test case (#35419)
## Description

It is for running a spec file with given run_count with PR comment.

Fixes #`34956`  

## Automation

/ok-to-test tags="@tag.Sanity"

### 🔍 Cypress test results
<!-- This is an auto-generated comment: Cypress test results  -->
> [!TIP]
> 🟢 🟢 🟢 All cypress tests have passed! 🎉 🎉 🎉
> Workflow run:
<https://github.com/appsmithorg/appsmith/actions/runs/10250647978>
> Commit: 69556d529719ff00d6d0b6961edb83a2a6112541
> <a
href="https://internal.appsmith.com/app/cypress-dashboard/rundetails-65890b3c81d7400d08fa9ee5?branch=master&workflowId=10250647978&attempt=1"
target="_blank">Cypress dashboard</a>.
> Tags: `@tag.Sanity`
> Spec:
> <hr>Mon, 05 Aug 2024 14:49:54 UTC
<!-- end of auto-generated comment: Cypress test results  -->


## Communication
Should the DevRel and Marketing teams inform users about this change?
- [ ] Yes
- [x] No


<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit


- **New Features**
- Introduced a CI/CD workflow to manage client and server builds with
Cypress testing.
- Added a job to enforce TypeScript file checks, providing feedback for
corrections on pull requests.
- Expanded testing capabilities with a new job for managing testing
limits.

- **Improvements**
	- Enhanced logging for better visibility into test execution attempts.

<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2024-08-05 21:04:45 +05:30

239 lines
7.1 KiB
TypeScript

import util from "./util";
export class staticSplit {
util = new util();
dbClient = this.util.configureDbClient();
private async getSpecsWithTime(specs: string[], attemptId: number) {
const client = await this.dbClient.connect();
const defaultDuration = 180000;
const specsMap = new Map();
try {
const queryRes = await client.query(
'SELECT * FROM public."spec_avg_duration" ORDER BY duration DESC',
);
queryRes.rows.forEach((obj) => {
specsMap.set(obj.name, obj);
});
const allSpecsWithDuration = specs.map((spec) => {
const match = specsMap.get(spec);
return match ? match : { name: spec, duration: defaultDuration };
});
return await this.util.divideSpecsIntoBalancedGroups(
allSpecsWithDuration,
Number(this.util.getVars().totalRunners),
);
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
// This function will finally get the specs as a comma separated string to pass the specs to the command
private async getSpecsToRun(
specPattern: string | string[] = "cypress/e2e/**/**/*.{js,ts}",
ignorePattern: string | string[],
attemptId: number,
): Promise<string[]> {
try {
const specFilePaths = await this.util.getSpecFilePaths(
specPattern,
ignorePattern,
);
if (this.util.getVars().cypressRerun === "true") {
return specFilePaths;
} else {
const specsToRun = await this.getSpecsWithTime(
specFilePaths,
attemptId,
);
return specsToRun === undefined || specsToRun.length === 0
? []
: specsToRun[Number(this.util.getVars().thisRunner)].map(
(spec) => spec.name,
);
}
} catch (err) {
console.error(err);
process.exit(1);
}
}
private async createAttempt() {
const client = await this.dbClient.connect();
try {
const runResponse = await client.query(
`INSERT INTO public."attempt" ("workflowId", "attempt", "repo", "committer", "type", "commitMsg", "branch")
VALUES ($1, $2, $3, $4, $5, $6, $7)
ON CONFLICT ("workflowId", attempt) DO NOTHING
RETURNING id;`,
[
this.util.getVars().runId,
this.util.getVars().attempt_number,
this.util.getVars().repository,
this.util.getVars().committer,
this.util.getVars().tag,
this.util.getVars().commitMsg,
this.util.getVars().branch,
],
);
if (runResponse.rows.length > 0) {
return runResponse.rows[0].id;
} else {
const res = await client.query(
`SELECT id FROM public."attempt" WHERE "workflowId" = $1 AND attempt = $2`,
[this.util.getVars().runId, this.util.getVars().attempt_number],
);
return res.rows[0].id;
}
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
private async createMatrix(attemptId: number) {
const client = await this.dbClient.connect();
try {
const matrixResponse = await client.query(
`INSERT INTO public."matrix" ("workflowId", "matrixId", "status", "attemptId")
VALUES ($1, $2, $3, $4)
ON CONFLICT ("matrixId", "attemptId") DO NOTHING
RETURNING id;`,
[
this.util.getVars().runId,
this.util.getVars().thisRunner,
"started",
attemptId,
],
);
return matrixResponse.rows[0].id;
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
private async getFailedSpecsFromPreviousRun(
runnerId = Number(this.util.getVars().thisRunner),
workflowId = Number(this.util.getVars().runId),
attempt_number = Number(this.util.getVars().attempt_number) - 1,
) {
const client = await this.dbClient.connect();
try {
const dbRes = await client.query(
`SELECT name FROM public."specs"
WHERE "matrixId" =
(SELECT id FROM public."matrix"
WHERE "attemptId" = (
SELECT id FROM public."attempt" WHERE "workflowId" = $1 and "attempt" = $2
) AND "matrixId" = $3
) AND status IN ('fail', 'queued', 'in-progress')`,
[workflowId, attempt_number, runnerId],
);
const specs: string[] =
dbRes.rows.length > 0 ? dbRes.rows.map((row) => row.name) : [];
return specs;
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
private async addSpecsToMatrix(matrixId: number, specs: string[]) {
const client = await this.dbClient.connect();
try {
for (const spec of specs) {
const res = await client.query(
`INSERT INTO public."specs" ("name", "matrixId", "status") VALUES ($1, $2, $3) RETURNING id`,
[spec, matrixId, "queued"],
);
}
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
private async updateTheSpecsForMatrix(attemptId: number, specs: string[]) {
const client = await this.dbClient.connect();
try {
if (specs.length > 0) {
const matrixRes = await this.createMatrix(attemptId);
await this.addSpecsToMatrix(matrixRes, specs);
}
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
private async getFlakySpecs() {
const client = await this.dbClient.connect();
try {
const dbRes = await client.query(
`SELECT spec FROM public."flaky_specs" WHERE skip=true`,
);
const specs: string[] =
dbRes.rows.length > 0 ? dbRes.rows.map((row) => row.spec) : [];
return specs;
} catch (err) {
console.log(err);
} finally {
client.release();
}
}
public async splitSpecs(config: Cypress.PluginConfigOptions) {
try {
let specPattern = config.specPattern;
let ignorePattern: string | string[] = config.excludeSpecPattern;
const cypressSpecs = this.util.getVars().cypressSpecs;
const defaultSpec = "cypress/scripts/no_spec.ts";
if (cypressSpecs != "") {
specPattern = cypressSpecs?.split(",").filter((val) => val !== "");
}
if (this.util.getVars().cypressRerun === "true") {
specPattern =
(await this.getFailedSpecsFromPreviousRun()) ?? defaultSpec;
}
if (this.util.getVars().cypressSkipFlaky === "true") {
let specsToSkip = (await this.getFlakySpecs()) ?? [];
ignorePattern = [...ignorePattern, ...specsToSkip];
}
const attempt = await this.createAttempt();
const specs =
(await this.getSpecsToRun(specPattern, ignorePattern, attempt)) ?? [];
console.log("SPECS TO RUN ----------> :", specs);
console.log("attempt ID ----------> :", attempt);
if (specs.length > 0 && !specs.includes(defaultSpec)) {
config.specPattern = specs.length == 1 ? specs[0] : specs;
} else {
config.specPattern = defaultSpec;
}
await this.updateTheSpecsForMatrix(attempt, specs);
return config;
} catch (err) {
console.log(err);
} finally {
this.dbClient.end();
}
}
}