Browse Source

Merge branch 'release/backend-rewrite' into refactor/event-classes

Owen Diffey 9 months ago
parent
commit
c0c2675cf6

+ 2 - 2
.github/workflows/automated-tests.yml

@@ -23,7 +23,7 @@ jobs:
                   ./musare.sh build
             - name: Start Musare
               run: ./musare.sh start
-#            - name: Test Backend
-#              run: ./musare.sh test backend
+            - name: Test Backend
+              run: ./musare.sh test backend
             - name: Test Frontend
               run: ./musare.sh test frontend

+ 10 - 2
backend/.eslintrc

@@ -65,7 +65,11 @@
 		"import/no-extraneous-dependencies": [
 			"error",
 			{
-				"devDependencies": ["**/*.test.ts", "**/*.spec.ts"]
+				"devDependencies": [
+					"**/*.test.ts",
+					"**/*.spec.ts",
+					"src/tests/**/*.ts"
+				]
 			}
 		],
 		"no-restricted-syntax": [
@@ -88,7 +92,11 @@
 	},
 	"overrides": [
 		{
-			"files": ["**/*.test.ts", "**/*.spec.ts"],
+			"files": [
+				"**/*.test.ts",
+				"**/*.spec.ts",
+				"src/tests/**/*.ts"
+			],
 			"rules": {
 				"no-unused-expressions": "off",
 				"prefer-arrow-callback": "off",

+ 43 - 1
backend/package-lock.json

@@ -28,8 +28,10 @@
 				"ws": "^8.13.0"
 			},
 			"devDependencies": {
+				"@faker-js/faker": "^8.4.1",
 				"@microsoft/tsdoc": "^0.14.2",
 				"@types/chai": "^4.3.5",
+				"@types/chai-as-promised": "^7.1.8",
 				"@types/config": "^3.3.0",
 				"@types/express": "^4.17.17",
 				"@types/mocha": "^10.0.1",
@@ -126,6 +128,22 @@
 				"node": "^12.22.0 || ^14.17.0 || >=16.0.0"
 			}
 		},
+		"node_modules/@faker-js/faker": {
+			"version": "8.4.1",
+			"resolved": "https://registry.npmjs.org/@faker-js/faker/-/faker-8.4.1.tgz",
+			"integrity": "sha512-XQ3cU+Q8Uqmrbf2e0cIC/QN43sTBSC8KF12u29Mb47tWrt2hAgBXSgpZMj4Ao8Uk0iJcU99QsOCaIL8934obCg==",
+			"dev": true,
+			"funding": [
+				{
+					"type": "opencollective",
+					"url": "https://opencollective.com/fakerjs"
+				}
+			],
+			"engines": {
+				"node": "^14.17.0 || ^16.13.0 || >=18.0.0",
+				"npm": ">=6.14.13"
+			}
+		},
 		"node_modules/@humanwhocodes/config-array": {
 			"version": "0.11.8",
 			"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
@@ -406,6 +424,15 @@
 			"integrity": "sha512-mEo1sAde+UCE6b2hxn332f1g1E8WfYRu6p5SvTKr2ZKC1f7gFJXk4h5PyGP9Dt6gCaG8y8XhwnXWC6Iy2cmBng==",
 			"dev": true
 		},
+		"node_modules/@types/chai-as-promised": {
+			"version": "7.1.8",
+			"resolved": "https://registry.npmjs.org/@types/chai-as-promised/-/chai-as-promised-7.1.8.tgz",
+			"integrity": "sha512-ThlRVIJhr69FLlh6IctTXFkmhtP3NpMZ2QGq69StYLyKZFp/HOp1VdKZj7RvfNWYYcJ1xlbLGLLWj1UvP5u/Gw==",
+			"dev": true,
+			"dependencies": {
+				"@types/chai": "*"
+			}
+		},
 		"node_modules/@types/config": {
 			"version": "3.3.0",
 			"resolved": "https://registry.npmjs.org/@types/config/-/config-3.3.0.tgz",
@@ -5634,6 +5661,12 @@
 			"integrity": "sha512-ElyB54bJIhXQYVKjDSvCkPO1iU1tSAeVQJbllWJq1XQSmmA4dgFk8CbiBGpiOPxleE48vDogxCtmMYku4HSVLA==",
 			"dev": true
 		},
+		"@faker-js/faker": {
+			"version": "8.4.1",
+			"resolved": "https://registry.npmjs.org/@faker-js/faker/-/faker-8.4.1.tgz",
+			"integrity": "sha512-XQ3cU+Q8Uqmrbf2e0cIC/QN43sTBSC8KF12u29Mb47tWrt2hAgBXSgpZMj4Ao8Uk0iJcU99QsOCaIL8934obCg==",
+			"dev": true
+		},
 		"@humanwhocodes/config-array": {
 			"version": "0.11.8",
 			"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
@@ -5877,6 +5910,15 @@
 			"integrity": "sha512-mEo1sAde+UCE6b2hxn332f1g1E8WfYRu6p5SvTKr2ZKC1f7gFJXk4h5PyGP9Dt6gCaG8y8XhwnXWC6Iy2cmBng==",
 			"dev": true
 		},
+		"@types/chai-as-promised": {
+			"version": "7.1.8",
+			"resolved": "https://registry.npmjs.org/@types/chai-as-promised/-/chai-as-promised-7.1.8.tgz",
+			"integrity": "sha512-ThlRVIJhr69FLlh6IctTXFkmhtP3NpMZ2QGq69StYLyKZFp/HOp1VdKZj7RvfNWYYcJ1xlbLGLLWj1UvP5u/Gw==",
+			"dev": true,
+			"requires": {
+				"@types/chai": "*"
+			}
+		},
 		"@types/config": {
 			"version": "3.3.0",
 			"resolved": "https://registry.npmjs.org/@types/config/-/config-3.3.0.tgz",
@@ -9696,4 +9738,4 @@
 			"dev": true
 		}
 	}
-}
+}

+ 3 - 1
backend/package.json

@@ -36,8 +36,10 @@
 		"ws": "^8.13.0"
 	},
 	"devDependencies": {
+		"@faker-js/faker": "^8.4.1",
 		"@microsoft/tsdoc": "^0.14.2",
 		"@types/chai": "^4.3.5",
+		"@types/chai-as-promised": "^7.1.8",
 		"@types/config": "^3.3.0",
 		"@types/express": "^4.17.17",
 		"@types/mocha": "^10.0.1",
@@ -65,4 +67,4 @@
 		"tsconfig-paths": "^4.2.0",
 		"typescript": "^5.0.4"
 	}
-}
+}

+ 191 - 0
backend/src/Job.spec.ts

@@ -0,0 +1,191 @@
+import "@/tests/support/setup";
+import { faker } from "@faker-js/faker";
+import sinon from "sinon";
+import Job, { JobOptions, JobStatus } from "@/Job";
+import { TestModule } from "@/tests/support/TestModule";
+
+describe("Job", function () {
+	class TestJob extends Job {
+		public constructor(options?: JobOptions) {
+			super(new TestModule(), null, options);
+		}
+
+		protected async _execute() {}
+	}
+
+	describe("getName", function () {
+		it("should return camelcase name of class", function () {
+			TestJob.getName().should.be.equal("testJob");
+			new TestJob().getName().should.be.equal("testJob");
+		});
+	});
+
+	describe("getPath", function () {
+		it("should return joined module and job name", function () {
+			new TestJob().getPath().should.be.equal("test.testJob");
+		});
+	});
+
+	describe("getPriority", function () {
+		it("should return configured priority", function () {
+			const priority = faker.number.int();
+			new TestJob({ priority }).getPriority().should.be.equal(priority);
+		});
+	});
+
+	describe("getUuid", function () {
+		it("should return generated uuid", function () {
+			const job = new TestJob();
+			const uuid = faker.string.uuid();
+			Reflect.set(job, "_uuid", uuid);
+			job.getUuid().should.be.equal(uuid);
+		});
+	});
+
+	describe("getStatus", function () {
+		it("should return current status", function () {
+			new TestJob().getStatus().should.be.equal(JobStatus.QUEUED);
+		});
+	});
+
+	describe("getModule", function () {
+		it("should return configured module", function () {
+			const module = new TestModule();
+			const job = new TestJob();
+			Reflect.set(job, "_module", module);
+
+			job.getModule().should.be.equal(module);
+		});
+	});
+
+	describe("isApiEnabled", function () {
+		it("should return configured value", function () {
+			class EnabledJob extends TestJob {
+				protected static _apiEnabled = true;
+			}
+			EnabledJob.isApiEnabled().should.be.true;
+			new EnabledJob().isApiEnabled().should.be.true;
+
+			class DisabledJob extends TestJob {
+				protected static _apiEnabled = false;
+			}
+			DisabledJob.isApiEnabled().should.be.false;
+			new DisabledJob().isApiEnabled().should.be.false;
+		});
+	});
+
+	describe("execute", function () {
+		it("should prevent multiple executions", async function () {
+			const job = new TestJob();
+			Reflect.set(job, "_authorize", sinon.stub());
+
+			await job.execute();
+
+			await job
+				.execute()
+				.should.eventually.be.rejectedWith(
+					"Job has already been executed."
+				);
+		});
+
+		it("should prevent execution if module can not run jobs", async function () {
+			const module = new TestModule();
+			Reflect.set(
+				module,
+				"canRunJobs",
+				sinon.fake(() => false)
+			);
+
+			const job = new TestJob();
+			Reflect.set(job, "_module", module);
+			Reflect.set(job, "_authorize", sinon.stub());
+
+			await job
+				.execute()
+				.should.eventually.be.rejectedWith(
+					"Module can not currently run jobs."
+				);
+		});
+
+		it("should update status to active", async function () {
+			class ActiveJob extends TestJob {
+				public callback?: (value?: unknown) => void;
+
+				protected async _authorize() {}
+
+				protected async _execute() {
+					this.getStatus().should.be.equal(JobStatus.ACTIVE);
+				}
+			}
+
+			await new ActiveJob().execute();
+		});
+
+		it("should call validation method", async function () {
+			const job = new TestJob();
+			const stub = sinon.stub();
+			Reflect.set(job, "_validate", stub);
+			Reflect.set(job, "_authorize", sinon.stub());
+
+			await job.execute();
+
+			stub.calledOnce.should.be.true;
+		});
+
+		it("should call authorize method", async function () {
+			const job = new TestJob();
+			const stub = sinon.stub();
+			Reflect.set(job, "_authorize", stub);
+
+			await job.execute();
+
+			stub.calledOnce.should.be.true;
+		});
+
+		it("should publish callback event if ref configured on success");
+
+		it("should add log on success");
+
+		it("should update stats on success");
+
+		it("should return data from private execute method on success", async function () {
+			const job = new TestJob();
+			const data = faker.word.words();
+			Reflect.set(job, "_authorize", sinon.stub());
+			Reflect.set(job, "_execute", sinon.fake(async () => data));
+
+			await job
+				.execute()
+				.should.eventually.be.equal(data);
+		});
+
+		it("should publish callback event if ref configured on error");
+
+		it("should add log on error");
+
+		it("should update stats on error");
+
+		it("should rethrow error");
+
+		it("should update stats on completion", async function () {
+			const job = new TestJob();
+			Reflect.set(job, "_authorize", sinon.stub());
+
+			await job.execute();
+
+			job.getStatus().should.be.equal(JobStatus.COMPLETED);
+		});
+	});
+
+	describe("log", function () {
+		it("should adds log to logbook");
+
+		it("should add path as category");
+
+		it("should add job json to log data");
+	});
+
+	describe("toJSON", function () {
+		it("should return job data as json object");
+	});
+});

+ 162 - 0
backend/src/JobStatistics.spec.ts

@@ -0,0 +1,162 @@
+import { faker } from "@faker-js/faker";
+import {
+	JobStatistic,
+	JobStatistics,
+	JobStatisticsType
+} from "./JobStatistics";
+
+describe("JobStatistics", function () {
+	describe("getStats", function () {
+		it("should include jobs statistics", function () {
+			const statistics = new JobStatistics();
+
+			const jobName = faker.lorem.text();
+
+			statistics.updateStats(jobName, JobStatisticsType.TOTAL);
+
+			statistics.getStats()[jobName].total.should.be.equal(1);
+		});
+
+		[
+			JobStatisticsType.CONSTRUCTED,
+			JobStatisticsType.FAILED,
+			JobStatisticsType.QUEUED,
+			JobStatisticsType.SUCCESSFUL,
+			JobStatisticsType.TOTAL
+		].forEach(function (type) {
+			it(`should sum ${type} count for total`, function () {
+				const statistics = new JobStatistics();
+
+				statistics.updateStats(faker.lorem.text(), type);
+				statistics.updateStats(faker.lorem.text(), type);
+
+				statistics
+					.getStats()
+					.total[type as keyof JobStatistic].should.be.equal(2);
+			});
+		});
+
+		it(`should sum total duration for total`, function () {
+			const statistics = new JobStatistics();
+
+			const firstDuration = faker.number.int();
+			const secondDuration = faker.number.int();
+			const totalDuration = firstDuration + secondDuration;
+
+			statistics.updateStats(
+				faker.lorem.text(),
+				JobStatisticsType.DURATION,
+				firstDuration
+			);
+			statistics.updateStats(
+				faker.lorem.text(),
+				JobStatisticsType.DURATION,
+				secondDuration
+			);
+
+			statistics
+				.getStats()
+				.total.totalTime.should.be.equal(totalDuration);
+		});
+
+		it("should calculate average time for total", function () {
+			const statistics = new JobStatistics();
+
+			const firstJobName = faker.lorem.text();
+			const secondJobName = faker.lorem.text();
+			const firstDuration = faker.number.int();
+			const secondDuration = faker.number.int();
+			const averageDuration = (firstDuration + secondDuration) / 2;
+
+			statistics.updateStats(firstJobName, JobStatisticsType.TOTAL);
+			statistics.updateStats(
+				firstJobName,
+				JobStatisticsType.DURATION,
+				firstDuration
+			);
+
+			statistics.updateStats(secondJobName, JobStatisticsType.TOTAL);
+			statistics.updateStats(
+				secondJobName,
+				JobStatisticsType.DURATION,
+				secondDuration
+			);
+
+			statistics
+				.getStats()
+				.total.averageTime.should.be.equal(averageDuration);
+		});
+	});
+
+	describe("updateStats", function () {
+		const jobName = faker.lorem.text();
+
+		[
+			JobStatisticsType.CONSTRUCTED,
+			JobStatisticsType.FAILED,
+			JobStatisticsType.QUEUED,
+			JobStatisticsType.SUCCESSFUL,
+			JobStatisticsType.TOTAL
+		].forEach(function (type) {
+			it(`should increment ${type} count`, function () {
+				const statistics = new JobStatistics();
+
+				statistics.updateStats(jobName, type);
+				statistics.updateStats(jobName, type);
+
+				statistics
+					.getStats()
+					[jobName][type as keyof JobStatistic].should.be.equal(2);
+			});
+		});
+
+		it(`should add to total duration`, function () {
+			const statistics = new JobStatistics();
+
+			const firstDuration = faker.number.int();
+			const secondDuration = faker.number.int();
+			const totalDuration = firstDuration + secondDuration;
+
+			statistics.updateStats(
+				jobName,
+				JobStatisticsType.DURATION,
+				firstDuration
+			);
+			statistics.updateStats(
+				jobName,
+				JobStatisticsType.DURATION,
+				secondDuration
+			);
+
+			statistics
+				.getStats()
+				[jobName].totalTime.should.be.equal(totalDuration);
+		});
+
+		it("should calculate average time", function () {
+			const statistics = new JobStatistics();
+
+			const firstDuration = faker.number.int();
+			const secondDuration = faker.number.int();
+			const averageDuration = (firstDuration + secondDuration) / 2;
+
+			statistics.updateStats(jobName, JobStatisticsType.TOTAL);
+			statistics.updateStats(
+				jobName,
+				JobStatisticsType.DURATION,
+				firstDuration
+			);
+
+			statistics.updateStats(jobName, JobStatisticsType.TOTAL);
+			statistics.updateStats(
+				jobName,
+				JobStatisticsType.DURATION,
+				secondDuration
+			);
+
+			statistics
+				.getStats()
+				[jobName].averageTime.should.be.equal(averageDuration);
+		});
+	});
+});

+ 7 - 10
backend/src/JobStatistics.ts

@@ -7,16 +7,13 @@ export enum JobStatisticsType {
 	DURATION = "duration"
 }
 
+export type JobStatistic = Record<
+	Exclude<JobStatisticsType, "duration"> | "averageTime" | "totalTime",
+	number
+>;
+
 export class JobStatistics {
-	private _stats: Record<
-		string,
-		Record<
-			| Exclude<JobStatisticsType, "duration">
-			| "averageTime"
-			| "totalTime",
-			number
-		>
-	>;
+	private _stats: Record<string, JobStatistic>;
 
 	public constructor() {
 		this._stats = {};
@@ -27,7 +24,7 @@ export class JobStatistics {
 	 *
 	 * @returns Job queue statistics
 	 */
-	public getStats() {
+	public getStats(): Record<string | "total", JobStatistic> {
 		const total = Object.values(this._stats).reduce(
 			(a, b) => ({
 				successful: a.successful + b.successful,

+ 9 - 0
backend/src/tests/support/TestModule.ts

@@ -0,0 +1,9 @@
+import BaseModule, { ModuleStatus } from "@/BaseModule";
+
+export class TestModule extends BaseModule {
+	public constructor() {
+		super("test");
+
+		this._status = ModuleStatus.STARTED;
+	}
+}

+ 12 - 0
backend/src/tests/support/setup.ts

@@ -0,0 +1,12 @@
+import chai from "chai";
+import chaiAsPromised from "chai-as-promised";
+import sinon from "sinon";
+import sinonChai from "sinon-chai";
+
+chai.should();
+chai.use(sinonChai);
+chai.use(chaiAsPromised);
+
+afterEach(async function () {
+	sinon.reset();
+});