Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

System tests should cleanup after themselves #2216

Merged
merged 77 commits into from
Oct 21, 2024
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
77 commits
Select commit Hold shift + click to select a range
d993866
keeping track of resources created during tests and adding cleanup
ATorrise Aug 6, 2024
3b01d28
tests that ive been preoccupied with
ATorrise Aug 14, 2024
fb156de
fixing these was hardgit add .
ATorrise Aug 15, 2024
960777b
ensuring more tests delete jobs from mf, also fixing delete operations
ATorrise Aug 15, 2024
68a7fdd
Merge remote-tracking branch 'origin/next' into system-test-cleanup
ATorrise Aug 15, 2024
4f059e6
finished zosjobs sdk tests and starting zosfiles sdk tests
ATorrise Aug 16, 2024
81048a7
modifying upload.system.test took the longest... removing some test f…
ATorrise Aug 21, 2024
a505c0d
took awhile to see the difference in regex between v1 and v2 that i w…
ATorrise Aug 22, 2024
56f0e9a
providing more clarity in a comment about calculating previous jobid
ATorrise Aug 22, 2024
c458433
fixing cli.zos-jobs.download and its accompanying scripts
ATorrise Aug 22, 2024
e356ac8
more cli sytem tests and ammended shell scripts
ATorrise Aug 22, 2024
123c7f9
finishin clizosjobslistspoolfiles
ATorrise Aug 22, 2024
d7b2840
fixing submit with a different, less intensive cleanup approach
ATorrise Aug 22, 2024
e1f2ca2
realizing need to keep createZosmfSession for uss files..
ATorrise Aug 23, 2024
4ccaad5
finishing cliZosjobsSubmit
ATorrise Aug 23, 2024
c5995dd
fixing error in variable name across files and finishing view spool t…
ATorrise Aug 23, 2024
a0fa936
unable to modify file like others
ATorrise Aug 26, 2024
f746a0b
deletes remaining ds from mf
ATorrise Aug 26, 2024
885a28a
Merge branch 'next' into system-test-cleanup
ATorrise Aug 26, 2024
f334c76
Update cli.zos-jobs.submit.stdin.system.test.ts to remove semicolon
ATorrise Aug 26, 2024
e11726f
Update cli.zos-jobs.submit.stdin.system.test.ts removing accidental line
ATorrise Aug 26, 2024
70cb522
whoops
ATorrise Aug 26, 2024
d87f789
Merge branch 'next' into system-test-cleanup
ATorrise Aug 26, 2024
0b475ac
changelog
ATorrise Aug 27, 2024
3853c8a
linting
ATorrise Aug 27, 2024
0159d2c
Merge branch 'next' into system-test-cleanup
ATorrise Aug 28, 2024
c7de80c
no circ dependency error?
ATorrise Aug 28, 2024
54baa2d
updating changelog
ATorrise Aug 28, 2024
ebe3570
moving things around with andrew's help
ATorrise Aug 28, 2024
663340f
moving around things in testEnv and ITestEnv
ATorrise Aug 28, 2024
9d59e1a
about to make a really big change
ATorrise Aug 29, 2024
fe2c72b
restoring clitestutils cleanup()
ATorrise Aug 29, 2024
46677f7
Merge branch 'next' into system-test-cleanup
ATorrise Aug 29, 2024
21bf9aa
fixing other testEnv
ATorrise Aug 29, 2024
a9d2865
updating imports
ATorrise Aug 29, 2024
52361c2
fixing imports
ATorrise Aug 29, 2024
309be62
hopefully not breaking
ATorrise Aug 29, 2024
585bb75
accidentally left 1 thing out of if statement
ATorrise Aug 29, 2024
a83444a
whoops
ATorrise Aug 29, 2024
bc1ffa2
making a wait util method and implementing it everywhere
ATorrise Aug 29, 2024
8ccb8f4
addressing one of fernandos comments
ATorrise Aug 30, 2024
842ee6b
timeout corrections
ATorrise Aug 30, 2024
9e8f22e
removing change to a unit test
ATorrise Aug 30, 2024
1c7a5b2
changing system test to work with 'any' job name
ATorrise Aug 30, 2024
83faac9
wiping testEnvironment.resources in cleanup
ATorrise Sep 4, 2024
6c7064b
changing the way the test works but need testing mini to be active an…
ATorrise Sep 4, 2024
deb5cfa
Merge branch 'next' into system-test-cleanup
ATorrise Sep 4, 2024
cefbff1
hopefully good to merge now
ATorrise Sep 4, 2024
aaa43b8
Merge branch 'system-test-cleanup' of https://github.com/zowe/zowe-cl…
ATorrise Sep 4, 2024
201ecc1
fixin - initializing resources
ATorrise Sep 4, 2024
4a37668
addressing timothy's requests
ATorrise Sep 19, 2024
36a19ca
Merge remote-tracking branch 'origin/next' into system-test-cleanup
ATorrise Sep 19, 2024
5e40b67
Merge branch 'master' of https://github.com/zowe/zowe-cli into system…
ATorrise Sep 19, 2024
2d1c9be
fixing runCliScrip import and TestEnvironment.cleanup
ATorrise Sep 19, 2024
d89d3b1
createSession() to createZosmfSession()
ATorrise Sep 19, 2024
ae20141
createsession to createzosmfsession
ATorrise Sep 19, 2024
54e8d99
adding import back into test
ATorrise Sep 20, 2024
9de05a2
getting things deleted
ATorrise Sep 20, 2024
f9f3b16
Merge branch 'master' of https://github.com/zowe/zowe-cli into system…
zFernand0 Sep 26, 2024
3b008cb
review: update a few files based on comments :yum:
zFernand0 Sep 27, 2024
4a96709
chore: address Zos limitation on dataset name length
zFernand0 Sep 27, 2024
04b92c8
midway through editing file to utilize testResources
ATorrise Sep 30, 2024
5f1d974
midway changes.. tests failing
ATorrise Oct 1, 2024
6a1dc9f
test fixin
ATorrise Oct 2, 2024
a628144
1 error left in download test
ATorrise Oct 3, 2024
7f50352
Fix remaining uss download system tests
t1m0thyj Oct 3, 2024
66f5e11
Clean up resources in download system tests
t1m0thyj Oct 7, 2024
73e9201
Clean up remaining resources for zosfiles
t1m0thyj Oct 8, 2024
0a32123
Revert changes to cli-test-utils package
t1m0thyj Oct 8, 2024
e3805fd
Clean up local files from vsam tests
t1m0thyj Oct 8, 2024
b2a3bac
Fix failing zosjobs system tests
t1m0thyj Oct 11, 2024
336d3e4
Fix remaining errors in system tests
t1m0thyj Oct 14, 2024
7068b6c
Address unused variables found by CodeQL
t1m0thyj Oct 14, 2024
a2cb59e
Merge branch 'master' into system-test-cleanup
t1m0thyj Oct 14, 2024
f7f6696
Fix system tests for cancel and search jobs
t1m0thyj Oct 14, 2024
8229c76
Merge branch 'master' into system-test-cleanup
t1m0thyj Oct 18, 2024
cacdd3a
Fix failing TSO system tests
t1m0thyj Oct 18, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
59 changes: 58 additions & 1 deletion __tests__/__packages__/cli-test-utils/src/TestUtils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,64 @@
import * as fs from "fs";
import { spawnSync, SpawnSyncReturns, ExecFileException } from "child_process";
import { ITestEnvironment } from "./environment/doc/response/ITestEnvironment";
import { CommandProfiles, ICommandDefinition, IHandlerParameters } from "@zowe/imperative";
import { AbstractSession, CommandProfiles, ICommandDefinition, IHandlerParameters, IO } from "@zowe/imperative";
import { DeleteJobs, ICommonJobParms, IDeleteJobParms, IJob } from "@zowe/zos-jobs-for-zowe-sdk";
import { Delete } from "@zowe/zos-files-for-zowe-sdk"
import { posix } from "path";

/**
* Delete a local testing file after use
* @param {string} filePath - File path of temporary file
*/
export function deleteLocalFile(filePath: string): void {
ATorrise marked this conversation as resolved.
Show resolved Hide resolved
try {
fs.unlinkSync(filePath);
} catch {
// If fs.unlinkSync fails, try to delete it with IO.deleteFile
try {
IO.deleteFile(posix.basename(filePath));
} catch {
throw new Error(`Error deleting local file: ${filePath}`);
}
}
}

/**
* Delete a uss file from the mainframe
* @param {AbstractSession} session - z/OSMF connection info
* @param {string} fileName - The name of the USS file
*/
export function deleteFiles(session: AbstractSession, fileName: string): void {
Delete.ussFile(session, fileName)
}

/**
* Delete a dataset from the mainframe
* @param {AbstractSession} session - z/OSMF connection info
* @param {string} datasetName - The name of the dataset
*/
export function deleteDataset(session: AbstractSession, dataSetName: string): void {
Delete.dataSet(session, dataSetName);
}

/**
* Delete a job from the mainframe using Zowe SDKs - IJob
* @param {AbstractSession} session - z/OSMF connection info
* @param {IJob} job - the job that you want to delete
*/
export function deleteJob(session: AbstractSession, job: IJob): void {
DeleteJobs.deleteJobForJob(session, job);
}

/**
* Delete a job from the mainframe using Zowe SDKs - jobid, jobname
* @param {AbstractSession} session - z/OSMF connection info
* @param {params} ICommonJobParms - constains jobname and jobid for job to delete
*/
export function deleteJobCommon(session: AbstractSession, params: ICommonJobParms): void {
DeleteJobs.deleteJobCommon(session, params as IDeleteJobParms);
}


/**
* Execute a CLI script
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,16 +11,14 @@

import * as fs from "fs";
import * as nodePath from "path";

import * as yaml from "js-yaml";
import { v4 as uuidv4 } from "uuid";
import { ImperativeError, ImperativeExpect, IO, Logger, LoggingConfigurer, TextUtils } from "@zowe/imperative";

import { AbstractSession, ImperativeError, ImperativeExpect, IO, Logger, LoggingConfigurer, ProfileInfo, TextUtils } from "@zowe/imperative";
import { ISetupEnvironmentParms } from "./doc/parms/ISetupEnvironmentParms";
import { ITestEnvironment } from "./doc/response/ITestEnvironment";
import { TempTestProfiles } from "./TempTestProfiles";
import { PROJECT_ROOT_DIR, TEST_RESOURCE_DIR, TEST_RESULT_DATA_DIR, TEST_USING_WORKSPACE } from "../TestConstants";
import { runCliScript } from "../TestUtils";
import { runCliScript, deleteFiles, deleteJob, deleteDataset, deleteLocalFile, deleteJobCommon } from "../TestUtils";

/**
* Use the utility methods here to setup the test environment for running APIs
Expand All @@ -43,7 +41,7 @@ export class TestEnvironment {
* @returns {Promise<ITestEnvironment>}
* @memberof TestEnvironment
*/
public static async setUp(params: ISetupEnvironmentParms): Promise<ITestEnvironment<any>> {
public static async setUp(params: ISetupEnvironmentParms, session?: AbstractSession): Promise<ITestEnvironment<any>> {
// Validate the input parameters
ImperativeExpect.toNotBeNullOrUndefined(params,
`${TestEnvironment.ERROR_TAG} createTestEnv(): No parameters supplied.`);
Expand All @@ -66,7 +64,15 @@ export class TestEnvironment {
const result: ITestEnvironment<any> = {
workingDir: testDirectory,
systemTestProperties: systemProps,
env
env,
resources: {
localFiles: [],
files: [],
jobs: [], // Array of IJob objects
jobData: [], // Array of objects with jobid and jobname
datasets: [],
...(session && { session }) // Only include session if it is passed in
}
};

if (params.installPlugin) {
Expand All @@ -85,6 +91,24 @@ export class TestEnvironment {
return result;
}

/**
* Create a session using the default z/OSMF profile (if a session has not been added to test)
* @returns {Promise<ISession>} - A promise that resolves to the created session object
* @throws Will throw an error if reading profiles or creating the session fails
* @memberof TestEnvironment
*/
public static async createSession(): Promise<AbstractSession> {
// Load connection info from default z/OSMF profile
const profInfo = new ProfileInfo("zowe");
await profInfo.readProfilesFromDisk();
const zosmfProfAttrs = profInfo.getDefaultProfile("zosmf");
ATorrise marked this conversation as resolved.
Show resolved Hide resolved
if (!zosmfProfAttrs) {
throw new Error("Default z/OSMF profile not found.");
}
const zosmfMergedArgs = profInfo.mergeArgsForProfile(zosmfProfAttrs, { getSecureVals: true });
return ProfileInfo.createSession(zosmfMergedArgs.knownArgs);
}

/**
* Clean up your test environment.
* Deletes any temporary profiles that have been created
Expand All @@ -102,6 +126,29 @@ export class TestEnvironment {
const pluginDir = testEnvironment.workingDir + "/plugins";
require("rimraf").sync(pluginDir);
}

// Clean up resources
for (const file of testEnvironment.resources.localFiles) {
deleteLocalFile(file);
}
// Check if session exists; if not, create one
const session = testEnvironment.resources.session || await TestEnvironment.createSession();
for (const file of testEnvironment.resources.files) {
deleteFiles(session, file);
}
for (const job of testEnvironment.resources.jobs) {
deleteJob(session, job);
}
for (const jobData of testEnvironment.resources.jobData) {
if (jobData.jobname && jobData.jobid) {
deleteJobCommon(session, jobData);
} else {
console.error('Error: Missing jobname or jobid for jobData:', jobData);
}
}
for (const dataset of testEnvironment.resources.datasets) {
deleteDataset(session, dataset);
}
ATorrise marked this conversation as resolved.
Show resolved Hide resolved
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,23 @@
*
*/

import { AbstractSession } from "@zowe/imperative";
import { ICommonJobParms, IJob } from "@zowe/zos-jobs-for-zowe-sdk";

/**
* The test environment for your test.
* @export
* @interface ITestEnvironment
*/
export interface ITestEnvironment<TestPropertiesSchema> {
resources: {
ATorrise marked this conversation as resolved.
Show resolved Hide resolved
localFiles: string[];
files: string[];
jobs: IJob[];
jobData: ICommonJobParms[]; // Contains jobname and jobid
datasets: string[];
session?: AbstractSession;
};
/**
* The working directory for your test environment. It is a unique (uuid) area where your tests can create
* their home folders (for imperative, etc.) and you can use the area as scratch for any files, etc. that
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,39 +9,43 @@
*
*/

import { ITestEnvironment, runCliScript } from "@zowe/cli-test-utils";
import { TestEnvironment } from "../../../../../../__tests__/__src__/environment/TestEnvironment";
import { ITestEnvironment, runCliScript, TestEnvironment } from "@zowe/cli-test-utils";
import { ITestPropertiesSchema } from "../../../../../../__tests__/__src__/properties/ITestPropertiesSchema";
import { JobTestsUtils } from "../../../../../zosjobs/__tests__/__system__/JobTestsUtils";
import { IO } from "@zowe/imperative";
import { AbstractSession, IO } from "@zowe/imperative";

// Test Environment populated in the beforeAll();
let REAL_SESSION: AbstractSession;
let TEST_ENVIRONMENT: ITestEnvironment<ITestPropertiesSchema>;
const LOCAL_JCL_FILE: string = __dirname + "/" + "testFileOfLocalJCL.txt";
const jobDataRegexV1 = /Successfully submitted request to cancel job (\w+) \((JOB\d+)\)/;
const jobDataRegex = /Successfully canceled job (\w+) \((JOB\d+)\)/;

describe("zos-jobs cancel job command", () => {
// Create the unique test environment
beforeAll(async () => {
TEST_ENVIRONMENT = await TestEnvironment.setUp({
testName: "zos_jobs_cancel_job_command",
tempProfileTypes: ["zosmf"]
});
const systemProps = TEST_ENVIRONMENT.systemTestProperties;
}, REAL_SESSION = await TestEnvironment.createSession());
ATorrise marked this conversation as resolved.
Show resolved Hide resolved

const systemProps = TEST_ENVIRONMENT.systemTestProperties;
const jcl = JobTestsUtils.getSleepJCL(systemProps.zosmf.user, systemProps.tso.account, systemProps.zosjobs.jobclass);
const bufferJCL: Buffer = Buffer.from(jcl);
IO.createFileSync(LOCAL_JCL_FILE);
IO.writeFile(LOCAL_JCL_FILE, bufferJCL);
TEST_ENVIRONMENT.resources.localFiles.push(LOCAL_JCL_FILE);
});

afterAll(async () => {
IO.deleteFile(LOCAL_JCL_FILE);
await TestEnvironment.cleanUp(TEST_ENVIRONMENT);
});

describe("error handling", () => {

it("should surface an error from z/OSMF if the jobid doesn't exist", () => {
const response = runCliScript(__dirname + "/__scripts__/job/not_found.sh", TEST_ENVIRONMENT);

expect(response.status).toBe(1);
expect(response.stdout.toString()).toBe("");
expect(response.stderr.toString()).toContain("Cannot obtain job info for job id = JOB00000");
Expand All @@ -51,6 +55,14 @@ describe("zos-jobs cancel job command", () => {
it("should surface an error from z/OSMF if the jobid was already canceled", () => {
runCliScript(__dirname + "/__scripts__/job/submit_job.sh", TEST_ENVIRONMENT, [LOCAL_JCL_FILE]);
const response = runCliScript(__dirname + "/__scripts__/job/cancel_job_v2_bad.sh", TEST_ENVIRONMENT, [LOCAL_JCL_FILE]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegex) || [];
Fixed Show fixed Hide fixed
TEST_ENVIRONMENT.resources.jobData.push({ jobname, jobid });

// Calculate the previous job ID (one less) - jobid created by test before, inaccessible until now
t1m0thyj marked this conversation as resolved.
Show resolved Hide resolved
const jobidNumber = parseInt(jobid.replace('JOB', ''), 10); //Extract the numeric part of the job ID
const previousJobid = 'JOB' + String(jobidNumber - 1).padStart(5, '0'); // then decrement by 1, assuming 5 digits in the job ID
TEST_ENVIRONMENT.resources.jobData.push({ jobname, jobid: previousJobid } );

expect(response.status).toBe(1);
expect(response.stderr.toString()).toContain("Failed to cancel job");
expect(response.stderr.toString()).toContain("Job not cancellable or purgeable");
Expand All @@ -62,25 +74,38 @@ describe("zos-jobs cancel job command", () => {
describe("successful scenario", () => {
it("should cancel a job v1", () => {
const response = runCliScript(__dirname + "/__scripts__/job/cancel_job_v1.sh", TEST_ENVIRONMENT, [LOCAL_JCL_FILE]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegexV1) || [];
Fixed Show fixed Hide fixed

expect(response.stderr.toString()).toBe("");
expect(response.status).toBe(0);
expect(response.stdout.toString()).toContain("Successfully submitted request to cancel job");

TEST_ENVIRONMENT.resources.jobData.push({ jobname, jobid });
});

it("should cancel a job v2", () => {
const response = runCliScript(__dirname + "/__scripts__/job/cancel_job_v2.sh", TEST_ENVIRONMENT, [LOCAL_JCL_FILE]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegex) || [];
Fixed Show fixed Hide fixed

expect(response.stderr.toString()).toBe("");
expect(response.status).toBe(0);
expect(response.stdout.toString()).toContain("Successfully canceled job");
expect(response.stdout.toString()).not.toContain("Failed to cancel job");

TEST_ENVIRONMENT.resources.jobData.push({ jobname, jobid });
});

it("should cancel a job default", () => {
const response = runCliScript(__dirname + "/__scripts__/job/cancel_job.sh", TEST_ENVIRONMENT, [LOCAL_JCL_FILE]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegex) || [];
Fixed Show fixed Hide fixed

expect(response.stderr.toString()).toBe("");
expect(response.status).toBe(0);
expect(response.stdout.toString()).toContain("Successfully canceled job");
expect(response.stdout.toString()).not.toContain("Failed to cancel job");
expect(response.stdout.toString()).not.toContain("Failed to cancel job");

TEST_ENVIRONMENT.resources.jobData.push({ jobname, jobid });
});

describe("without profiles", () => {
Expand All @@ -92,7 +117,7 @@ describe("zos-jobs cancel job command", () => {
beforeAll(async () => {
TEST_ENVIRONMENT_NO_PROF = await TestEnvironment.setUp({
testName: "zos_jobs_cancel_job_without_profiles"
});
}, REAL_SESSION = await TestEnvironment.createSession());

DEFAULT_SYSTEM_PROPS = TEST_ENVIRONMENT_NO_PROF.systemTestProperties;
});
Expand All @@ -111,9 +136,13 @@ describe("zos-jobs cancel job command", () => {
DEFAULT_SYSTEM_PROPS.zosmf.user,
DEFAULT_SYSTEM_PROPS.zosmf.password,
]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegexV1) || [];
Fixed Show fixed Hide fixed

expect(response.stderr.toString()).toBe("");
expect(response.status).toBe(0);
expect(response.stdout.toString()).toContain("Successfully submitted request to cancel job");

TEST_ENVIRONMENT_NO_PROF.resources.jobData.push({ jobname, jobid });
});

it("cancel a job without a profile 2.0", async () => {
Expand All @@ -126,9 +155,13 @@ describe("zos-jobs cancel job command", () => {
DEFAULT_SYSTEM_PROPS.zosmf.user,
DEFAULT_SYSTEM_PROPS.zosmf.password,
]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegex) || [];
Fixed Show fixed Hide fixed

expect(response.stderr.toString()).toBe("");
expect(response.status).toBe(0);
expect(response.stdout.toString()).toContain("Successfully canceled job");

TEST_ENVIRONMENT_NO_PROF.resources.jobData.push({ jobname, jobid });
});

it("cancel a job without a profile default", async () => {
Expand All @@ -141,9 +174,13 @@ describe("zos-jobs cancel job command", () => {
DEFAULT_SYSTEM_PROPS.zosmf.user,
DEFAULT_SYSTEM_PROPS.zosmf.password,
]);
const [, jobname, jobid] = response.stdout.toString().match(jobDataRegex) || [];
Fixed Show fixed Hide fixed

expect(response.stderr.toString()).toBe("");
expect(response.status).toBe(0);
expect(response.stdout.toString()).toContain("Successfully canceled job");

TEST_ENVIRONMENT_NO_PROF.resources.jobData.push({ jobname, jobid });
});
});
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,9 @@ then
exit $RC
fi

# Echo the job ID for further use
echo "Submitted job ID: $JOBID"

# Loop until the job goes to the output queue
until [ $ATTEMPTS -gt 0 ]
ATorrise marked this conversation as resolved.
Show resolved Hide resolved
do
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,11 @@ then
exit $RC
fi

# Echo the job ID for further use
echo "Submitted job ID: $JOBID"

# Loop until the job goes to the output queue
until [ $ATTEMPTS -gt 0 ]
until [ $ATTEMPTS -le 0 ]
ATorrise marked this conversation as resolved.
Show resolved Hide resolved
do
STATUS=`zowe jobs view job-status-by-jobid $JOBID --host $HOST --port $PORT --user $USER --password $PASS --ru=false --rff status --rft string`
RC=$?
Expand Down
Loading
Loading