From 0ad4ca6d5e9c5fc442aa1d18e0aa93b9bca46e26 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Sat, 18 May 2024 20:09:27 +0400 Subject: [PATCH 01/19] fix(util): tune yaml validation schema --- src/util/validations.ts | 30 +++++++++++++++++------------- 1 file changed, 17 insertions(+), 13 deletions(-) diff --git a/src/util/validations.ts b/src/util/validations.ts index 5b1aff0cf..15c224e16 100644 --- a/src/util/validations.ts +++ b/src/util/validations.ts @@ -50,19 +50,23 @@ export const manifestSchema = z.object({ ), outputs: z.array(z.string()).optional(), }), - execution: z.object({ - command: z.string(), - environment: z.object({ - 'if-version': z.string(), - os: z.string(), - 'os-version': z.string(), - 'node-version': z.string(), - 'date-time': z.string(), - dependencies: z.array(z.string()), - }), - status: z.string(), - error: z.string().optional(), - }), + execution: z + .object({ + command: z.string().optional(), + environment: z + .object({ + 'if-version': z.string(), + os: z.string(), + 'os-version': z.string(), + 'node-version': z.string(), + 'date-time': z.string(), + dependencies: z.array(z.string()), + }) + .optional(), + status: z.string(), + error: z.string().optional(), + }) + .optional(), tree: z.record(z.string(), z.any()), }); From 5248f24e7dfa28c259d42a4d24ce5f46f5139c49 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Sat, 18 May 2024 20:10:08 +0400 Subject: [PATCH 02/19] fix(util): tune convert to xorable to handle objects --- src/util/helpers.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/util/helpers.ts b/src/util/helpers.ts index 5dcbbeb80..1f71a66a3 100644 --- a/src/util/helpers.ts +++ b/src/util/helpers.ts @@ -64,6 +64,10 @@ const convertToXorable = (value: any) => { return value.length > 0 ? 1 : 0; } + if (typeof value === 'object') { + return 1; + } + return 0; }; From 98c8cd3b877118fe68ef22440440e36570eccb5c Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Sat, 18 May 2024 20:10:43 +0400 Subject: [PATCH 03/19] feat(lib): tune compare to do advanced execution check --- src/lib/compare.ts | 42 ++++++++++++++++++++++++++++++------------ 1 file changed, 30 insertions(+), 12 deletions(-) diff --git a/src/lib/compare.ts b/src/lib/compare.ts index d920b0072..03e62fd20 100644 --- a/src/lib/compare.ts +++ b/src/lib/compare.ts @@ -2,21 +2,31 @@ import {oneIsPrimitive} from '../util/helpers'; import {Difference} from '../types/lib/compare'; +/** + * Returns `status` and `exception` properties from execution context. + */ +const omitExecutionParams = (object: any) => ({ + status: object.status, + ...(object.error && { + error: object.error, + }), +}); + /** * 1. If objects are not of the same type or are primitive types, compares directly. * 2. Gets the keys from both objects. - * 3. Checks for keys present in both objects. - * 4. If both are arrays, checks their elements. - * 5. Checks for differences in values for common keys. - * 6. If all common keys are checked and no differences are found, return empty object. + * 3. If both are arrays, checks their elements. + * 4. Checks for keys present in both objects. + * If key is `execution`, omit unnecessary params. + * 5. If all keys are checked and no differences are found, return empty object. */ export const compare = (source: any, target: any, path = ''): Difference => { if (oneIsPrimitive(source, target)) { return source !== target ? { path, - source: source, - target: target, + source, + target, } : {}; } @@ -28,7 +38,7 @@ export const compare = (source: any, target: any, path = ''): Difference => { if (Array.isArray(source) && Array.isArray(target)) { if (source.length !== target.length) { - return {path, source: source, target: target}; + return {path, source, target}; } for (let i = 0; i < source.length; i++) { @@ -37,11 +47,19 @@ export const compare = (source: any, target: any, path = ''): Difference => { } for (const key of allKeys) { - const result = compare( - source[key], - target[key], - path ? `${path}.${key}` : key - ); + let result: any = {}; + + if (key === 'execution') { + if (source[key] && target[key]) { + result = compare( + omitExecutionParams(source[key]), + omitExecutionParams(target[key]), + path ? `${path}.${key}` : key + ); + } + } else { + result = compare(source[key], target[key], path ? `${path}.${key}` : key); + } if (Object.keys(result).length) { return result; From e6df2cd31d480519d929487b6e6918de02fe8547 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Sat, 18 May 2024 20:11:43 +0400 Subject: [PATCH 04/19] fix(src): execution is present in catch --- src/index.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/index.ts b/src/index.ts index 041086559..692195c4e 100644 --- a/src/index.ts +++ b/src/index.ts @@ -34,8 +34,8 @@ const impactEngine = async () => { await exhaust(aggregatedTree, context, outputOptions); } catch (error) { if (error instanceof Error) { - envManifest.execution.status = 'fail'; - envManifest.execution.error = error.toString(); + envManifest.execution!.status = 'fail'; + envManifest.execution!.error = error.toString(); logger.error(error); const {tree, ...context} = envManifest; From ee57cabbc131e6920c97b90519522f48d5e56130 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:02:38 +0400 Subject: [PATCH 05/19] fix(util): in helpers return empty string if piped source is unvailable --- src/util/helpers.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/util/helpers.ts b/src/util/helpers.ts index 1f71a66a3..60f8e16ce 100644 --- a/src/util/helpers.ts +++ b/src/util/helpers.ts @@ -122,7 +122,7 @@ export const formatNotMatchingLog = (message: Difference) => { */ const collectPipedData = async () => { if (process.stdin.isTTY) { - return; + return ''; } const readline = createInterface({ From e5e7d91b3a1685ed953a3fa1fa8bfaad6b821302 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:06:46 +0400 Subject: [PATCH 06/19] fix(util): move messages to strings, fix strategy --- src/util/args.ts | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/src/util/args.ts b/src/util/args.ts index a78d0a507..56466fd48 100644 --- a/src/util/args.ts +++ b/src/util/args.ts @@ -14,7 +14,14 @@ const {CliInputError} = ERRORS; const {IE, IF_DIFF} = CONFIG; -const {FILE_IS_NOT_YAML, MANIFEST_IS_MISSING, NO_OUTPUT} = STRINGS; +const { + FILE_IS_NOT_YAML, + MANIFEST_IS_MISSING, + NO_OUTPUT, + SOURCE_IS_NOT_YAML, + TARGET_IS_NOT_YAML, + INVALID_TARGET, +} = STRINGS; /** * Validates `ie` process arguments. @@ -106,6 +113,10 @@ export const parseIfDiffArgs = () => { const {source, target} = validateAndParseIfDiffArgs(); if (target) { + if (source && !checkIfFileIsYaml(source)) { + throw new CliInputError(SOURCE_IS_NOT_YAML); + } + if (checkIfFileIsYaml(target)) { const response: LoadDiffParams = { targetPath: prependFullFilePath(target), @@ -118,8 +129,8 @@ export const parseIfDiffArgs = () => { return response; } - throw new CliInputError(FILE_IS_NOT_YAML); // change to one of the source or target parts are not a yaml + throw new CliInputError(TARGET_IS_NOT_YAML); } - throw new CliInputError(MANIFEST_IS_MISSING); // change to one of the source or target are missing + throw new CliInputError(INVALID_TARGET); }; From 810d56aae5731e28f3a8cc8eba14dd4090d5fe5c Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:11:29 +0400 Subject: [PATCH 07/19] fix(types): add message to difference object --- src/types/lib/compare.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/types/lib/compare.ts b/src/types/lib/compare.ts index 756fca9eb..0f0cae8cf 100644 --- a/src/types/lib/compare.ts +++ b/src/types/lib/compare.ts @@ -2,5 +2,6 @@ export type Difference = { path?: string; source?: any; target?: any; + message?: string; [key: string]: any; }; From f2908b75040f3f5d6ee070207b6417fff3191909 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:12:06 +0400 Subject: [PATCH 08/19] feat(lib): introduce classified error to load --- src/lib/load.ts | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/src/lib/load.ts b/src/lib/load.ts index b8ac3841b..447153068 100644 --- a/src/lib/load.ts +++ b/src/lib/load.ts @@ -1,15 +1,21 @@ import * as YAML from 'js-yaml'; +import {ERRORS} from '../util/errors'; import {openYamlFileAsObject} from '../util/yaml'; import {readAndParseJson} from '../util/json'; import {parseManifestFromStdin} from '../util/helpers'; import {PARAMETERS} from '../config'; +import {STRINGS} from '../config'; import {Parameters} from '../types/parameters'; import {LoadDiffParams} from '../types/util/args'; import {Manifest} from '../types/manifest'; +const {CliInputError} = ERRORS; + +const {INVALID_SOURCE} = STRINGS; + /** * Parses manifest file as an object. Checks if parameter file is passed via CLI, then loads it too. * Returns context, tree and parameters (either the default one, or from CLI). @@ -37,7 +43,7 @@ export const loadIfDiffFiles = async (params: LoadDiffParams) => { const pipedSourceManifest = await parseManifestFromStdin(); if (!sourcePath && !pipedSourceManifest) { - throw new Error('Source is invalid.'); + throw new CliInputError(INVALID_SOURCE); } const loadFromSource = From 0ffb6ed50af1355cc484134831d1011c6aa4145c Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:12:33 +0400 Subject: [PATCH 09/19] feat(config): move all error messages from if-diff to strings --- src/config/strings.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/config/strings.ts b/src/config/strings.ts index 5bfb7d134..04551b9b9 100644 --- a/src/config/strings.ts +++ b/src/config/strings.ts @@ -52,4 +52,8 @@ You have not selected an output method. To see your output data, you can choose --stdout: this will print your output data to the console --output : this will save your output data to the given filepath (do not provide file extension) Note that for the '--output' option you also need to define the output type in your manifest file. See https://if.greensoftware.foundation/major-concepts/manifest-file#initialize`, + SOURCE_IS_NOT_YAML: 'Given source file is not in yaml format.', + TARGET_IS_NOT_YAML: 'Given target is not in yaml format.', + INVALID_TARGET: 'Target is invalid.', + INVALID_SOURCE: 'Source is invalid.', }; From f979d67d6d1160a9ab928d8fe873929955a2f114 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:13:05 +0400 Subject: [PATCH 10/19] test(util): implement units for helpers --- src/__tests__/unit/util/helpers.test.ts | 213 +++++++++++++++++++++++- 1 file changed, 212 insertions(+), 1 deletion(-) diff --git a/src/__tests__/unit/util/helpers.test.ts b/src/__tests__/unit/util/helpers.test.ts index 5bda8b752..721c21c36 100644 --- a/src/__tests__/unit/util/helpers.test.ts +++ b/src/__tests__/unit/util/helpers.test.ts @@ -1,14 +1,24 @@ const mockWarn = jest.fn(); const mockError = jest.fn(); +jest.mock('node:readline/promises', () => + require('../../../__mocks__/readline') +); jest.mock('../../../util/logger', () => ({ logger: { warn: mockWarn, error: mockError, }, })); -import {andHandle, mergeObjects} from '../../../util/helpers'; +import { + andHandle, + formatNotMatchingLog, + mergeObjects, + oneIsPrimitive, + parseManifestFromStdin, +} from '../../../util/helpers'; import {ERRORS} from '../../../util/errors'; +import {Difference} from '../../../types/lib/compare'; const {WriteFileError} = ERRORS; @@ -175,4 +185,205 @@ describe('util/helpers: ', () => { expect(result).toEqual(expectedResult); }); }); + + describe('formatNotMatchingLog(): ', () => { + const actualLogger = console.log; + const mockLogger = jest.fn(); + console.log = mockLogger; + + beforeEach(() => { + mockLogger.mockReset(); + }); + + it('logs the message.', () => { + const difference: Difference = { + message: 'mock-message', + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(1); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + }); + + it('logs message and path.', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(2); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + }); + + it('logs message, path and formatted source/target (one is missing).', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + source: 'mock-source', + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(4); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + expect(mockLogger).toHaveBeenCalledWith(`source: ${difference.source}`); + expect(mockLogger).toHaveBeenCalledWith('target: missing'); + }); + + it('logs message, path and formatted source/target.', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + source: 'mock-source', + target: 'mock-target', + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(4); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + expect(mockLogger).toHaveBeenCalledWith(`source: ${difference.source}`); + expect(mockLogger).toHaveBeenCalledWith(`target: ${difference.target}`); + }); + + it('logs message, path and formatted source/target (numbers).', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + source: 10, + target: 0, + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(4); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + expect(mockLogger).toHaveBeenCalledWith(`source: ${difference.source}`); + expect(mockLogger).toHaveBeenCalledWith(`target: ${difference.target}`); + }); + + it('logs message, path and formatted source/target (booleans).', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + source: true, + target: false, + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(4); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + expect(mockLogger).toHaveBeenCalledWith(`source: ${difference.source}`); + expect(mockLogger).toHaveBeenCalledWith(`target: ${difference.target}`); + }); + + it('logs message, path and formatted source/target (objects).', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + source: {}, + target: false, + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(4); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + expect(mockLogger).toHaveBeenCalledWith(`source: ${difference.source}`); + expect(mockLogger).toHaveBeenCalledWith(`target: ${difference.target}`); + }); + + it('logs message, path and formatted source/target (empty string).', () => { + const difference: Difference = { + message: 'mock-message', + path: 'mock.path', + source: '', + target: false, + }; + + formatNotMatchingLog(difference); + expect(mockLogger).toHaveBeenCalledTimes(4); + expect(mockLogger).toHaveBeenCalledWith(difference.message); + expect(mockLogger).toHaveBeenCalledWith(difference.path); + expect(mockLogger).toHaveBeenCalledWith(`source: ${difference.source}`); + expect(mockLogger).toHaveBeenCalledWith(`target: ${difference.target}`); + }); + + afterAll(() => { + console.log = actualLogger; + }); + }); + + describe('oneIsPrimitive(): ', () => { + it('returns true if values are nullish.', () => { + const source = null; + const target = undefined; + + const result = oneIsPrimitive(source, target); + expect(result).toBeTruthy(); + }); + + it('returns true if values are string or number.', () => { + const source = 'string'; + const target = 10; + + const result = oneIsPrimitive(source, target); + expect(result).toBeTruthy(); + }); + + it('returns false if one of values is object.', () => { + const source = 'string'; + const target = {}; + + const result = oneIsPrimitive(source, target); + expect(result).toBeFalsy(); + }); + }); + + describe('parseManifestFromStdin(): ', () => { + it('returns empty string if there is no data in stdin.', async () => { + const response = await parseManifestFromStdin(); + const expectedResult = ''; + + expect(response).toEqual(expectedResult); + }); + + it('returns empty string if nothing is piped.', async () => { + const originalIsTTY = process.stdin.isTTY; + process.stdin.isTTY = true; + const response = await parseManifestFromStdin(); + const expectedResult = ''; + + expect(response).toEqual(expectedResult); + process.stdin.isTTY = originalIsTTY; + }); + + it('throws error if there is no manifest in stdin.', async () => { + process.env.readline = 'no_manifest'; + const expectedMessage = 'Manifest not found in STDIN.'; + expect.assertions(1); + + try { + await parseManifestFromStdin(); + } catch (error) { + if (error instanceof Error) { + expect(error.message).toEqual(expectedMessage); + } + } + }); + + it('returns empty string if there is no data in stdin.', async () => { + process.env.readline = 'manifest'; + const response = await parseManifestFromStdin(); + const expectedMessage = ` +name: mock-name +description: mock-description +`; + + expect(response).toEqual(expectedMessage); + }); + }); }); From 79c1492ed1d177374862477877105627b782f251 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:13:41 +0400 Subject: [PATCH 11/19] test(util): implement units for if diff cli args --- src/__tests__/unit/util/args.test.ts | 119 +++++++++++++++++++++++++-- 1 file changed, 112 insertions(+), 7 deletions(-) diff --git a/src/__tests__/unit/util/args.test.ts b/src/__tests__/unit/util/args.test.ts index cefda328d..87acad810 100644 --- a/src/__tests__/unit/util/args.test.ts +++ b/src/__tests__/unit/util/args.test.ts @@ -28,15 +28,33 @@ jest.mock('ts-command-line-args', () => ({ manifest: 'manifest-mock.yml', 'override-params': 'override-params-mock.yml', }; - case 'help': - return { - manifest: path.normalize(`${processRunningPath}/manifest-mock.yml`), - help: true, - }; case 'not-yaml': return { manifest: 'mock.notyaml', }; + /** If-diff mocks */ + case 'only-target': + return { + target: 'target-mock.yml', + }; + case 'target-is-not-yaml': + return { + target: 'target-mock', + }; + case 'source-is-not-yaml': + return { + target: 'target-mock.yml', + source: 'source-mock', + }; + case 'target-source': + return { + target: 'target-mock.yml', + source: 'source-mock.yml', + }; + case 'diff-throw-error': + throw new Error('mock-error'); + case 'diff-throw': + throw 'mock-error'; default: return { manifest: 'mock-manifest.yaml', @@ -48,14 +66,20 @@ jest.mock('ts-command-line-args', () => ({ import path = require('path'); -import {parseIEProcessArgs} from '../../../util/args'; +import {parseIEProcessArgs, parseIfDiffArgs} from '../../../util/args'; import {ERRORS} from '../../../util/errors'; import {STRINGS} from '../../../config'; const {CliInputError} = ERRORS; -const {MANIFEST_IS_MISSING, FILE_IS_NOT_YAML} = STRINGS; +const { + MANIFEST_IS_MISSING, + FILE_IS_NOT_YAML, + TARGET_IS_NOT_YAML, + INVALID_TARGET, + SOURCE_IS_NOT_YAML, +} = STRINGS; describe('util/args: ', () => { const originalEnv = process.env; @@ -172,5 +196,86 @@ describe('util/args: ', () => { }); }); + describe('parseIfDiffArgs(): ', () => { + it('throws error if `target` is missing.', async () => { + expect.assertions(1); + + try { + await parseIfDiffArgs(); + } catch (error) { + if (error instanceof Error) { + expect(error).toEqual(new CliInputError(INVALID_TARGET)); + } + } + }); + + it('throws error if `target` is not a yaml.', async () => { + process.env.result = 'target-is-not-yaml'; + expect.assertions(1); + + try { + await parseIfDiffArgs(); + } catch (error) { + if (error instanceof Error) { + expect(error).toEqual(new CliInputError(TARGET_IS_NOT_YAML)); + } + } + }); + + it('returns `target`s full path.', async () => { + process.env.result = 'only-target'; + expect.assertions(1); + + const response = await parseIfDiffArgs(); + expect(response).toHaveProperty('targetPath'); + }); + + it('throws error if source is not a yaml.', async () => { + process.env.result = 'source-is-not-yaml'; + expect.assertions(1); + + try { + await parseIfDiffArgs(); + } catch (error) { + if (error instanceof Error) { + expect(error).toEqual(new CliInputError(SOURCE_IS_NOT_YAML)); + } + } + }); + + it('returns target and source full paths.', async () => { + process.env.result = 'target-source'; + expect.assertions(2); + + const response = await parseIfDiffArgs(); + expect(response).toHaveProperty('targetPath'); + expect(response).toHaveProperty('sourcePath'); + }); + + it('throws error if parsing failed.', async () => { + process.env.result = 'diff-throw-error'; + expect.assertions(1); + + try { + await parseIfDiffArgs(); + } catch (error) { + if (error instanceof Error) { + expect(error).toEqual(new CliInputError('mock-error')); + } + } + }); + + it('throws error if parsing failed (not instance of error).', async () => { + process.env.result = 'diff-throw'; + expect.assertions(1); + + try { + await parseIfDiffArgs(); + } catch (error) { + expect(error).toEqual('mock-error'); + } + }); + }); + process.env = originalEnv; }); From f8d6a07a057da1d5b6a831e4afa6eace1a14fc4a Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:14:20 +0400 Subject: [PATCH 12/19] test(lib): refine old tests, add units for if-diff loader --- src/__tests__/unit/lib/load.test.ts | 170 ++++++++++++++-------------- 1 file changed, 82 insertions(+), 88 deletions(-) diff --git a/src/__tests__/unit/lib/load.test.ts b/src/__tests__/unit/lib/load.test.ts index fed158b92..f9ad3e73b 100644 --- a/src/__tests__/unit/lib/load.test.ts +++ b/src/__tests__/unit/lib/load.test.ts @@ -1,4 +1,4 @@ -jest.mock('fs/promises', () => require('../../../__mocks__/fs')); +jest.mock('../../../util/json', () => require('../../../__mocks__/json')); jest.mock( 'mockavizta', () => ({ @@ -12,64 +12,52 @@ jest.mock( }), {virtual: true} ); +jest.mock('../../../util/helpers', () => ({ + parseManifestFromStdin: () => { + if (process.env.readline === 'valid-source') { + return ` +name: 'mock-name' +description: 'mock-description' +`; + } + return ''; + }, +})); +jest.mock('../../../util/yaml', () => ({ + openYamlFileAsObject: (path: string) => { + switch (path) { + case 'load-default.yml': + return 'raw-manifest'; + case 'source-path.yml': + return 'source-manifest'; + case 'target-path.yml': + return 'target-manifest'; + default: + return ''; + } + }, +})); -import {load} from '../../../lib/load'; +import {load, loadIfDiffFiles} from '../../../lib/load'; import {PARAMETERS} from '../../../config'; import {PluginParams} from '../../../types/interface'; +import {STRINGS} from '../../../config'; + +const {INVALID_SOURCE} = STRINGS; + describe('lib/load: ', () => { describe('load(): ', () => { it('loads yaml with default parameters.', async () => { - const inputPath = 'mock.yaml'; + const inputPath = 'load-default.yml'; const paramPath = undefined; const result = await load(inputPath, paramPath); const expectedValue = { - rawManifest: { - name: 'gsf-demo', - description: 'Hello', - tags: { - kind: 'web', - complexity: 'moderate', - category: 'cloud', - }, - initialize: { - plugins: { - mockavizta: { - path: 'mockavizta', - method: 'Mockavizta', - }, - }, - }, - tree: { - children: { - 'front-end': { - pipeline: ['boavizta-cpu'], - config: { - 'boavizta-cpu': { - 'core-units': 24, - processor: 'Intel® Core™ i7-1185G7', - }, - }, - inputs: [ - { - timestamp: '2023-07-06T00:00', - duration: 3600, - 'cpu/utilization': 18.392, - }, - { - timestamp: '2023-08-06T00:00', - duration: 3600, - 'cpu/utilization': 16, - }, - ], - }, - }, - }, - }, + rawManifest: 'raw-manifest', parameters: PARAMETERS, }; @@ -77,54 +65,13 @@ describe('lib/load: ', () => { }); it('loads yaml with custom parameters.', async () => { - const inputPath = 'param-mock.yaml'; + const inputPath = 'load-default.yml'; const paramPath = 'param-mock.json'; const result = await load(inputPath, paramPath); const expectedValue = { - rawManifest: { - name: 'gsf-demo', - description: 'Hello', - tags: { - kind: 'web', - complexity: 'moderate', - category: 'cloud', - }, - initialize: { - plugins: { - mockavizta: { - path: 'mockavizta', - method: 'Mockavizta', - }, - }, - }, - tree: { - children: { - 'front-end': { - pipeline: ['boavizta-cpu'], - config: { - 'boavizta-cpu': { - 'core-units': 24, - processor: 'Intel® Core™ i7-1185G7', - }, - }, - inputs: [ - { - timestamp: '2023-07-06T00:00', - duration: 3600, - 'cpu/utilization': 18.392, - }, - { - timestamp: '2023-08-06T00:00', - duration: 3600, - 'cpu/utilization': 16, - }, - ], - }, - }, - }, - }, + rawManifest: 'raw-manifest', parameters: { 'mock-carbon': { description: 'an amount of carbon emitted into the atmosphere', @@ -142,4 +89,51 @@ describe('lib/load: ', () => { expect(result).toEqual(expectedValue); }); }); + + describe('loadIfDiffFiles(): ', () => { + it('rejects with invalid source error.', async () => { + const params = { + sourcePath: '', + targetPath: '', + }; + + try { + await loadIfDiffFiles(params); + } catch (error) { + if (error instanceof Error) { + expect(error).toBeInstanceOf(Error); + expect(error.message).toBe(INVALID_SOURCE); + } + } + }); + + it('successfully loads target, and source from stdin.', async () => { + process.env.readline = 'valid-source'; + const params = { + targetPath: 'target-path.yml', + }; + + const response = await loadIfDiffFiles(params); + const expectedSource = { + name: 'mock-name', + description: 'mock-description', + }; + const expectedTarget = 'target-manifest'; + expect(response.rawSourceManifest).toEqual(expectedSource); + expect(response.rawTargetManifest).toEqual(expectedTarget); + }); + + it('successfully loads target, and source from stdin.', async () => { + const params = { + targetPath: 'target-path.yml', + sourcePath: 'source-path.yml', + }; + + const response = await loadIfDiffFiles(params); + const expectedSource = 'source-manifest'; + const expectedTarget = 'target-manifest'; + expect(response.rawSourceManifest).toEqual(expectedSource); + expect(response.rawTargetManifest).toEqual(expectedTarget); + }); + }); }); From a7d294f7dd822f905e288bff9cc9d0e6f023c271 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:17:12 +0400 Subject: [PATCH 13/19] test(mocks): refine json --- src/__mocks__/json.ts | 14 ++++++++++++++ 1 file changed, 14 insertions(+) create mode 100644 src/__mocks__/json.ts diff --git a/src/__mocks__/json.ts b/src/__mocks__/json.ts new file mode 100644 index 000000000..d6bb24d2c --- /dev/null +++ b/src/__mocks__/json.ts @@ -0,0 +1,14 @@ +export const readAndParseJson = async () => { + return { + 'mock-carbon': { + description: 'an amount of carbon emitted into the atmosphere', + unit: 'gCO2e', + aggregation: 'sum', + }, + 'mock-cpu': { + description: 'number of cores available', + unit: 'cores', + aggregation: 'none', + }, + }; +}; From 2617ccc753524314e331b4afee8475c84664ecf7 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Thu, 23 May 2024 12:17:32 +0400 Subject: [PATCH 14/19] test(mocks): add readline --- src/__mocks__/readline/index.ts | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) create mode 100644 src/__mocks__/readline/index.ts diff --git a/src/__mocks__/readline/index.ts b/src/__mocks__/readline/index.ts new file mode 100644 index 000000000..c3af2fa16 --- /dev/null +++ b/src/__mocks__/readline/index.ts @@ -0,0 +1,20 @@ +export const createInterface = () => { + if (process.env.readline === 'no_manifest') { + return ` +mock message in console +`; + } + + if (process.env.readline === 'manifest') { + return (async function* (): any { + yield ` +# start +name: mock-name +description: mock-description +# end + `; + })(); + } + + return []; +}; From d0f35acb9ede37c4c885eb952c83f5faf4f27dcb Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Fri, 24 May 2024 18:29:12 +0400 Subject: [PATCH 15/19] test(util): gain unit coverage on args --- src/__tests__/unit/util/args.test.ts | 50 ++++++++++++++++++++-------- 1 file changed, 36 insertions(+), 14 deletions(-) diff --git a/src/__tests__/unit/util/args.test.ts b/src/__tests__/unit/util/args.test.ts index 87acad810..d40d36cae 100644 --- a/src/__tests__/unit/util/args.test.ts +++ b/src/__tests__/unit/util/args.test.ts @@ -32,6 +32,11 @@ jest.mock('ts-command-line-args', () => ({ return { manifest: 'mock.notyaml', }; + case 'stdout': + return { + manifest: 'manifest-mock.yaml', + stdout: true, + }; /** If-diff mocks */ case 'only-target': return { @@ -194,14 +199,31 @@ describe('util/args: ', () => { expect(error).toEqual(new CliInputError(FILE_IS_NOT_YAML)); } }); + + it('returns stdout and manifest.', () => { + expect.assertions(1); + + process.env.result = 'stdout'; + const manifestPath = 'manifest-mock.yaml'; + + const response = parseIEProcessArgs(); + const expectedResult = { + inputPath: path.normalize(`${processRunningPath}/${manifestPath}`), + outputOptions: { + stdout: true, + }, + }; + + expect(response).toEqual(expectedResult); + }); }); describe('parseIfDiffArgs(): ', () => { - it('throws error if `target` is missing.', async () => { + it('throws error if `target` is missing.', () => { expect.assertions(1); try { - await parseIfDiffArgs(); + parseIfDiffArgs(); } catch (error) { if (error instanceof Error) { expect(error).toEqual(new CliInputError(INVALID_TARGET)); @@ -209,12 +231,12 @@ describe('util/args: ', () => { } }); - it('throws error if `target` is not a yaml.', async () => { + it('throws error if `target` is not a yaml.', () => { process.env.result = 'target-is-not-yaml'; expect.assertions(1); try { - await parseIfDiffArgs(); + parseIfDiffArgs(); } catch (error) { if (error instanceof Error) { expect(error).toEqual(new CliInputError(TARGET_IS_NOT_YAML)); @@ -222,20 +244,20 @@ describe('util/args: ', () => { } }); - it('returns `target`s full path.', async () => { + it('returns `target`s full path.', () => { process.env.result = 'only-target'; expect.assertions(1); - const response = await parseIfDiffArgs(); + const response = parseIfDiffArgs(); expect(response).toHaveProperty('targetPath'); }); - it('throws error if source is not a yaml.', async () => { + it('throws error if source is not a yaml.', () => { process.env.result = 'source-is-not-yaml'; expect.assertions(1); try { - await parseIfDiffArgs(); + parseIfDiffArgs(); } catch (error) { if (error instanceof Error) { expect(error).toEqual(new CliInputError(SOURCE_IS_NOT_YAML)); @@ -243,21 +265,21 @@ describe('util/args: ', () => { } }); - it('returns target and source full paths.', async () => { + it('returns target and source full paths.', () => { process.env.result = 'target-source'; expect.assertions(2); - const response = await parseIfDiffArgs(); + const response = parseIfDiffArgs(); expect(response).toHaveProperty('targetPath'); expect(response).toHaveProperty('sourcePath'); }); - it('throws error if parsing failed.', async () => { + it('throws error if parsing failed.', () => { process.env.result = 'diff-throw-error'; expect.assertions(1); try { - await parseIfDiffArgs(); + parseIfDiffArgs(); } catch (error) { if (error instanceof Error) { expect(error).toEqual(new CliInputError('mock-error')); @@ -265,12 +287,12 @@ describe('util/args: ', () => { } }); - it('throws error if parsing failed (not instance of error).', async () => { + it('throws error if parsing failed (not instance of error).', () => { process.env.result = 'diff-throw'; expect.assertions(1); try { - await parseIfDiffArgs(); + parseIfDiffArgs(); } catch (error) { expect(error).toEqual('mock-error'); } From 32ed6dc54cb355a95e69a306b8908632ef1526d9 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Fri, 24 May 2024 18:29:38 +0400 Subject: [PATCH 16/19] test(util): add units for check if equal --- src/__tests__/unit/util/helpers.test.ts | 27 +++++++++++++++++++++++++ 1 file changed, 27 insertions(+) diff --git a/src/__tests__/unit/util/helpers.test.ts b/src/__tests__/unit/util/helpers.test.ts index 721c21c36..d627790e4 100644 --- a/src/__tests__/unit/util/helpers.test.ts +++ b/src/__tests__/unit/util/helpers.test.ts @@ -12,6 +12,7 @@ jest.mock('../../../util/logger', () => ({ })); import { andHandle, + checkIfEqual, formatNotMatchingLog, mergeObjects, oneIsPrimitive, @@ -386,4 +387,30 @@ description: mock-description expect(response).toEqual(expectedMessage); }); }); + + describe('checkIfEqual(): ', () => { + it('checks if values are equal.', () => { + const a = 'mock'; + const b = 'mock'; + + const response = checkIfEqual(a, b); + expect(response).toBeTruthy(); + }); + + it('returns true if one of the values is wildcard.', () => { + const a = 'mock'; + const b = '*'; + + const response = checkIfEqual(a, b); + expect(response).toBeTruthy(); + }); + + it('returns false for number and string with the same value.', () => { + const a = 5; + const b = '5'; + + const response = checkIfEqual(a, b); + expect(response).toBeFalsy(); + }); + }); }); From dacd8092fbfb210af148cd61d1ac7b4abdefbc53 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Fri, 24 May 2024 19:17:23 +0400 Subject: [PATCH 17/19] test(lib): implement units for compare --- src/__tests__/unit/lib/compare.test.ts | 114 +++++++++++++++++++++++++ 1 file changed, 114 insertions(+) create mode 100644 src/__tests__/unit/lib/compare.test.ts diff --git a/src/__tests__/unit/lib/compare.test.ts b/src/__tests__/unit/lib/compare.test.ts new file mode 100644 index 000000000..dbac5a490 --- /dev/null +++ b/src/__tests__/unit/lib/compare.test.ts @@ -0,0 +1,114 @@ +import {compare} from '../../../lib/compare'; + +describe('lib/compare: ', () => { + describe('compare(): ', () => { + it('test if empty objects are equal.', () => { + const a = {}; + const b = {}; + + const response = compare(a, b); + expect(Object.keys(response).length).toEqual(0); + }); + + it('tests for nested objects with arrays.', () => { + const a = { + tree: { + inputs: [1, 2], + }, + }; + const b = { + tree: { + inputs: [1, 2], + }, + }; + + const response = compare(a, b); + expect(Object.keys(response).length).toEqual(0); + }); + + it('tests for nested objects with arrays (negative case).', () => { + const a = { + tree: { + inputs: [1, 2], + }, + }; + const b = { + tree: { + inputs: [1], + }, + }; + + const response = compare(a, b); + expect(response.path).toEqual('tree.inputs.1'); + expect(response.source).toEqual(2); + expect(response.target).toBeUndefined(); + }); + + it('checks if execution params are ignored.', () => { + const a = { + tree: { + inputs: [1, 2], + }, + execution: { + a: 'mock-a', + b: 'mock-b', + status: 'success', + }, + }; + const b = { + tree: { + inputs: [1, 2], + }, + execution: { + status: 'success', + }, + }; + + const response = compare(a, b); + expect(Object.keys(response).length).toEqual(0); + }); + + it('checks if error and status are in place, and others are ignored.', () => { + const a = { + tree: { + inputs: [1, 2], + }, + execution: { + a: 'a', + b: 'b', + error: 'mock-error-message', + status: 'fail', + }, + }; + const b = { + tree: { + inputs: [1, 2], + }, + execution: { + error: 'mock-error-message', + status: 'fail', + }, + }; + + const response = compare(a, b); + expect(Object.keys(response).length).toEqual(0); + }); + + it('checks if arrays are equal.', () => { + const a = [1, 2]; + const b = [1, 2]; + + const response = compare(a, b); + expect(Object.keys(response).length).toEqual(0); + }); + + it('checks if arrays are equal (first one is missing some items).', () => { + const a = [1]; + const b = [1, 2]; + + const response = compare(a, b); + const expectedResponse = {path: '1', source: undefined, target: 2}; + expect(response).toEqual(expectedResponse); + }); + }); +}); From e10754a7391819587bd49e03fccdd07c8dfe6ebb Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Fri, 24 May 2024 19:24:56 +0400 Subject: [PATCH 18/19] test(lib): fix ts error in env --- src/__tests__/unit/lib/environment.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/__tests__/unit/lib/environment.test.ts b/src/__tests__/unit/lib/environment.test.ts index e4d1c579b..c528eaee1 100644 --- a/src/__tests__/unit/lib/environment.test.ts +++ b/src/__tests__/unit/lib/environment.test.ts @@ -23,7 +23,7 @@ describe('lib/envirnoment: ', () => { it('checks environment response type.', async () => { // @ts-ignore const response = await injectEnvironment(context); - const {environment} = response.execution; + const environment = response.execution!.environment!; expect(typeof environment['date-time']).toEqual('string'); expect(Array.isArray(environment.dependencies)).toBeTruthy(); From 096a4602ff73b1dba5f71bb435059b0fec44e789 Mon Sep 17 00:00:00 2001 From: Narek Hovhannisyan Date: Tue, 28 May 2024 12:52:49 +0400 Subject: [PATCH 19/19] chore(package): update lock file --- package-lock.json | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/package-lock.json b/package-lock.json index 591705438..0c4cd07c8 100644 --- a/package-lock.json +++ b/package-lock.json @@ -21,7 +21,8 @@ "zod": "^3.22.4" }, "bin": { - "ie": "build/index.js" + "ie": "build/index.js", + "if-diff": "build/diff.js" }, "devDependencies": { "@babel/core": "^7.22.10",