Skip to content

Commit

Permalink
test: use rewire to test exhort services
Browse files Browse the repository at this point in the history
Signed-off-by: Ilona Shishov <[email protected]>
  • Loading branch information
IlonaShishov committed Feb 20, 2024
1 parent d07f029 commit 1ce2048
Show file tree
Hide file tree
Showing 8 changed files with 309 additions and 99 deletions.
185 changes: 96 additions & 89 deletions package-lock.json

Large diffs are not rendered by default.

5 changes: 3 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -262,6 +262,7 @@
"@typescript-eslint/eslint-plugin": "^6.8.0",
"@typescript-eslint/parser": "^6.8.0",
"@vscode/test-electron": "^2.3.5",
"babel-plugin-rewire": "^1.2.0",
"chai": "^4.3.10",
"decache": "^4.6.2",
"eslint": "^8.51.0",
Expand All @@ -278,11 +279,11 @@
"webpack-cli": "^5.1.4"
},
"dependencies": {
"@fabric8-analytics/fabric8-analytics-lsp-server": "^0.9.1-ea.0",
"@fabric8-analytics/fabric8-analytics-lsp-server": "^0.9.3",
"@redhat-developer/vscode-redhat-telemetry": "^0.7.0",
"@RHEcosystemAppEng/exhort-javascript-api": "^0.1.1-ea.14",
"fs": "^0.0.1-security",
"path": "^0.12.7",
"vscode-languageclient": "^8.1.0"
}
}
}
2 changes: 1 addition & 1 deletion src/exhortServices.ts
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,7 @@ async function tokenValidationService(options, source) {
vscode.window.showWarningMessage(`Failed to validate token. Status: ${tokenValidationStatus}`);
}
} catch (error) {
vscode.window.showErrorMessage(`Failed to validate token, Error: ${error}`);
vscode.window.showErrorMessage(`Failed to validate token, Error: ${error.message}`);
}
}

Expand Down
1 change: 0 additions & 1 deletion test/caStatusBarProvider.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,6 @@ suite('CAStatusBarProvider module', () => {
const uri = 'file:///mock/path';

caStatusBarProvider.showSummary(text, uri);
console.log(caStatusBarProvider['statusBarItem'].command);

expect(caStatusBarProvider['statusBarItem'].text).to.equal(text);
expect(caStatusBarProvider['statusBarItem'].tooltip).to.equal(PromptText.FULL_STACK_PROMPT_TEXT);
Expand Down
104 changes: 104 additions & 0 deletions test/exhortServices.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
import * as chai from 'chai';
import * as sinon from 'sinon';
import * as sinonChai from 'sinon-chai';
import { rewireModule, cleanupRewireFiles } from './utils';

const expect = chai.expect;
chai.use(sinonChai);

suite('ExhortServices module', async () => {
let sandbox: sinon.SinonSandbox;

const compiledFilePath = 'out/src/exhortServices';
const stackAnalysisReportHtmlMock = '<html>RHDA Report Mock</html>';

let exhortMock = {
default: {
stackAnalysis: async () => stackAnalysisReportHtmlMock,
validateToken: async (statusCode) => statusCode,
}
};

let vscodeMock = {
window: {
showInformationMessage: sinon.spy(),
showWarningMessage: sinon.spy(),
showErrorMessage: sinon.spy(),
}
};

let exhortServicesRewire;

setup(async () => {
sandbox = sinon.createSandbox();
exhortServicesRewire = await rewireModule(compiledFilePath);
exhortServicesRewire.__Rewire__('exhort_javascript_api_1', exhortMock);
exhortServicesRewire.__Rewire__('vscode', vscodeMock);
});

teardown(() => {
sandbox.restore();
cleanupRewireFiles(compiledFilePath);
});

test('should generate RHDA report HTML from Exhort Stack Analysis service', async () => {
await exhortServicesRewire.stackAnalysisService('mock/path/to/manifest', {})
.then((result) => {
expect(result).to.equal(stackAnalysisReportHtmlMock);
})
});

test('should perform token validation with Exhort Validate Token service', async () => {
await exhortServicesRewire.tokenValidationService(200, 'provider');
await exhortServicesRewire.tokenValidationService(400, 'provider');
await exhortServicesRewire.tokenValidationService(401, 'provider');
await exhortServicesRewire.tokenValidationService(403, 'provider');
await exhortServicesRewire.tokenValidationService(429, 'provider');
await exhortServicesRewire.tokenValidationService(500, 'provider');

expect(vscodeMock.window.showInformationMessage).to.have.been.calledWith('provider Token Validated Successfully');
expect(vscodeMock.window.showWarningMessage).to.have.been.calledWith('Missing token. Please provide a valid provider Token in the extension workspace settings. Status: 400');
expect(vscodeMock.window.showWarningMessage).to.have.been.calledWith('Invalid token. Please provide a valid provider Token in the extension workspace settings. Status: 401');
expect(vscodeMock.window.showWarningMessage).to.have.been.calledWith('Forbidden. The token does not have permissions. Please provide a valid provider Token in the extension workspace settings. Status: 403');
expect(vscodeMock.window.showWarningMessage).to.have.been.calledWith('Too many requests. Rate limit exceeded. Please try again in a little while. Status: 429');
expect(vscodeMock.window.showWarningMessage).to.have.been.calledWith('Failed to validate token. Status: 500');
});

test('should fail to generate RHDA report HTML from Exhort Stack Analysis service and reject with error', async () => {

let exhortMock = {
default: {
stackAnalysis: async () => {
throw new Error('Analysis Error');
},
}
};

exhortServicesRewire.__Rewire__('exhort_javascript_api_1', exhortMock);

await exhortServicesRewire.stackAnalysisService('mock/path/to/manifest', {})
.then(() => {
throw new Error('should have thrown Analysis Error')
})
.catch((error) => {
expect(error.message).to.equal('Analysis Error');
})
});

test('should fail to perform token validation with Exhort Validate Token service and display error message', async () => {

let exhortMock = {
default: {
validateToken: async () => {
throw new Error('Validation Error');
},
}
};

exhortServicesRewire.__Rewire__('exhort_javascript_api_1', exhortMock);

await exhortServicesRewire.tokenValidationService(500, 'provider');

expect(vscodeMock.window.showErrorMessage).to.have.been.calledWith('Failed to validate token, Error: Validation Error');
});
});
64 changes: 64 additions & 0 deletions test/redhatTelemetry.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
import * as chai from 'chai';
import * as sinon from 'sinon';
import * as sinonChai from 'sinon-chai';
import { rewireModule, cleanupRewireFiles } from './utils';

const expect = chai.expect;
chai.use(sinonChai);

suite('RedhatTelemetry module', async () => {
let sandbox: sinon.SinonSandbox;

const compiledFilePath = 'out/src/redhatTelemetry';
const redHatUUIDMock = 'Mock UUID';

let getRedHatUUIDMock = {
getRedHatUUID: async () => redHatUUIDMock
}

let sendEventMock = {
sendStartupEvent: sinon.spy(),
send: sinon.spy()
}

let getIdProviderMock = {
getIdProvider: async () => getRedHatUUIDMock,
getTelemetryService: async () => sendEventMock
}

let getRedHatServiceMock = {
getRedHatService: async () => getIdProviderMock
};

let redhatTelemetryRewire;

setup(async () => {
sandbox = sinon.createSandbox();
redhatTelemetryRewire = await rewireModule(compiledFilePath);
redhatTelemetryRewire.__Rewire__('vscode_redhat_telemetry_1', getRedHatServiceMock);
});

teardown(() => {
sandbox.restore();
cleanupRewireFiles(compiledFilePath);
});

test('should get UUID from vscode redhat telemetry service', async () => {
let telemetryId = await redhatTelemetryRewire.getTelemetryId({})
expect(telemetryId).to.equal(redHatUUIDMock);
});

test('should send statup telemetry event', async () => {
await redhatTelemetryRewire.startUp({})
expect(sendEventMock.sendStartupEvent).to.have.been.calledOnce;
});

test('should record telemetry event', async () => {
await redhatTelemetryRewire.record({}, 'telemetry_event_mock', { mockProp: 'mockValue' })
expect(sendEventMock.send).to.have.been.calledWith({
type: 'track',
name: 'telemetry_event_mock',
properties: { mockProp: 'mockValue' }
});
});
});
40 changes: 40 additions & 0 deletions test/utils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import * as babelCore from "@babel/core";
import * as fs from "fs";
import * as path from 'path';

/**
* Asynchronously imports a module.
*
* @param filepath Path to the module to be imported.
* @returns A promise that resolves to the module's exports.
*/
async function dynamicImportProvider(filepath) {
return await import(filepath)
}

/**
* Rewires a module for testing purposes.
* @param filepath path to the compiled JavaScript file of the tested module.
* @return A module instance that exposes private methods/functions/properties to be mocked/stubbed.
*/
export function rewireModule(filepath) {
let providerBuffeer = fs.readFileSync(filepath + ".js")
let providerSource = babelCore.transform(providerBuffeer, { plugins: ["babel-plugin-rewire"] }).code;
providerSource += `\n//# sourceMappingURL=${path.basename(filepath)}.js.map`;

fs.writeFileSync(filepath + "_rewire.js", providerSource)
return dynamicImportProvider("../../" + filepath + "_rewire.js")
}

/**
* Removes rewired modules from file system.
* @param filepath path to the compiled JavaScript file of the tested module.
*/
export function cleanupRewireFiles(filepath) {
const fileToRemove = filepath + "_rewire.js"
try {
fs.unlinkSync(fileToRemove);
} catch (err) {
console.error(`Error deleting rewire module ${fileToRemove}: ${err.message}`);
}
}
7 changes: 1 addition & 6 deletions webpack.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -41,14 +41,9 @@ module.exports = (env, argv) => {
},
{
test: /\.ts$/,
exclude: /node_modules/,
exclude: [/node_modules/, /utils\.ts$/],
use: [{
loader: 'ts-loader',
options: {
compilerOptions: {
"module": "es6" // override `tsconfig.json` so that TypeScript emits native JavaScript modules.
}
}
}]
}]
},
Expand Down

0 comments on commit 1ce2048

Please sign in to comment.