UNPKG

@sasjs/cli

Version:

Command line interface for SASjs

736 lines 52.2 kB
"use strict"; var __assign = (this && this.__assign) || function () { __assign = Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; var __generator = (this && this.__generator) || function (thisArg, body) { var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; function verb(n) { return function (v) { return step([n, v]); }; } function step(op) { if (f) throw new TypeError("Generator is already executing."); while (g && (g = 0, op[0] && (_ = 0)), _) try { if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; if (y = 0, t) op = [op[0] & 2, t.value]; switch (op[0]) { case 0: case 1: t = op; break; case 4: _.label++; return { value: op[1], done: false }; case 5: _.label++; y = op[1]; op = [0]; continue; case 7: op = _.ops.pop(); _.trys.pop(); continue; default: if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } if (t[2]) _.ops.pop(); _.trys.pop(); continue; } op = body.call(thisArg, _); } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; } }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); var path_1 = __importDefault(require("path")); var test_1 = require("../test"); var types_1 = require("../../../types"); var utils_1 = require("@sasjs/utils"); var test_2 = require("../../../utils/test"); var setConstants_1 = require("../../../utils/setConstants"); var dotenv_1 = __importDefault(require("dotenv")); var __1 = require("../.."); var sasJsModules = __importStar(require("../../../utils/createSASjsInstance")); var testResponses_1 = require("./mockedAdapter/testResponses"); var fileModule = __importStar(require("@sasjs/utils/file")); var utilsModule = __importStar(require("@sasjs/utils/utils")); var readAndValidateInputModule = __importStar(require("@sasjs/utils/input/readAndValidateInput")); var configUtils = __importStar(require("../../../utils/config")); var chalk_1 = __importDefault(require("chalk")); var mocks_1 = require("../../context/spec/mocks"); describe('sasjs test', function () { var expectedCoverageLcov = "TN:testsetup.sas\nSF:standalone\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:exampleprogram.test.sas\nSF:jobs/jobs/exampleprogram.sas\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:standalone.test.sas\nSF:standalone\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:examplemacro.test.sas\nSF:standalone\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:shouldFail.test.sas\nSF:standalone\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:dostuff.test.0.sas\nSF:services/admin/dostuff.sas\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:dostuff.test.1.sas\nSF:services/admin/dostuff.sas\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record\nTN:testteardown.sas\nSF:standalone\nFNF:1\nFNH:1\nLF:1\nLH:1\nBRF:1\nBRH:1\nend_of_record"; describe('SASVIYA', function () { var target = generateTarget(); var testUrl = function (test) { return "".concat(target.serverUrl, "/SASJobExecution/?_program=/Public/app/cli-tests/").concat(target.name, "/tests/").concat(test, "&_debug=2477&_contextName=").concat(encodeURIComponent(target.contextName)); }; var testUrlLink = function (test) { return "\"=HYPERLINK(\"\"".concat(testUrl(test), "\"\")\""); }; beforeAll(function () { return __awaiter(void 0, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, test_2.createTestApp)(__dirname, target.name)]; case 1: _a.sent(); return [4 /*yield*/, copyTestFiles(target.name)]; case 2: _a.sent(); return [4 /*yield*/, (0, __1.build)(target)]; case 3: _a.sent(); process.logger = new utils_1.Logger(utils_1.LogLevel.Off); return [2 /*return*/]; } }); }); }); beforeEach(function () { setupMocksForSASVIYA(); }); afterAll(function () { return __awaiter(void 0, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, test_2.removeTestApp)(__dirname, target.name)]; case 1: _a.sent(); return [2 /*return*/]; } }); }); }); it('should execute tests and create result CSV, XML and JSON files using default source and output locations', function () { return __awaiter(void 0, void 0, void 0, function () { var resultsFolderPath, resultsJsonPath, resultsCsvPath, resultsXmlPath, coverageReportPath, expectedResultsJson, expectedResultsCsv, resultsJson, parsedResults, resultsCsv, resultsXml, expectedResultsXml, coverageLcovPath, coverageLcov; return __generator(this, function (_a) { switch (_a.label) { case 0: resultsFolderPath = path_1.default.join(__dirname, target.name, 'sasjsresults'); resultsJsonPath = path_1.default.join(resultsFolderPath, 'testResults.json'); resultsCsvPath = path_1.default.join(resultsFolderPath, 'testResults.csv'); resultsXmlPath = path_1.default.join(resultsFolderPath, 'testResults.xml'); coverageReportPath = path_1.default.join(resultsFolderPath, 'coverage.lcov'); expectedResultsJson = { csv_result_path: resultsCsvPath, xml_result_path: resultsXmlPath, coverage_report_path: coverageReportPath, tests_that_pass: '2/7 (29%)', tests_with_results: '7/8 (88%)', target_name: target.name, target_server_url: target.serverUrl, target_server_type: target.serverType, local_date_time: expect.anything(), local_user_id: expect.anything(), local_machine_name: expect.anything(), sasjs_test_meta: [ { test_target: 'testsetup', results: [ { test_loc: 'tests/testsetup.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('testsetup') } ] }, { test_target: 'exampleprogram', results: [ { test_loc: 'tests/jobs/jobs/exampleprogram.test.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('jobs/jobs/exampleprogram.test') } ] }, { test_target: 'standalone', results: [ { test_loc: 'tests/jobs/jobs/standalone.test.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('jobs/jobs/standalone.test') } ] }, { test_target: 'examplemacro', results: [ { test_loc: 'tests/macros/examplemacro.test.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'examplemacro test.1 description', TEST_RESULT: 'PASS' } ], test_url: testUrl('macros/examplemacro.test') } ] }, { test_target: 'dostuff', results: [ { test_loc: 'tests/services/admin/dostuff.test.0.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'dostuff 0 test description', TEST_RESULT: 'FAIL' } ], test_url: testUrl('services/admin/dostuff.test.0') }, { test_loc: 'tests/services/admin/dostuff.test.1.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'dostuff 1 test description', TEST_RESULT: 'PASS' } ], test_url: testUrl('services/admin/dostuff.test.1') } ] }, { test_target: 'testteardown', results: [ { test_loc: 'tests/testteardown.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('testteardown') } ] } ] }; expectedResultsCsv = "test_target,test_loc,sasjs_test_id,test_suite_result,test_description,test_url\ntestsetup,tests/testsetup.sas,sasjs_test_id,not provided,,".concat(testUrlLink('testsetup'), "\nexampleprogram,tests/jobs/jobs/exampleprogram.test.sas,sasjs_test_id,not provided,,").concat(testUrlLink('jobs/jobs/exampleprogram.test'), "\nstandalone,tests/jobs/jobs/standalone.test.sas,sasjs_test_id,not provided,,").concat(testUrlLink('jobs/jobs/standalone.test'), "\nexamplemacro,tests/macros/examplemacro.test.sas,sasjs_test_id,PASS,examplemacro test.1 description,").concat(testUrlLink('macros/examplemacro.test'), "\ndostuff,tests/services/admin/dostuff.test.0.sas,sasjs_test_id,FAIL,dostuff 0 test description,").concat(testUrlLink('services/admin/dostuff.test.0'), "\ndostuff,tests/services/admin/dostuff.test.1.sas,sasjs_test_id,PASS,dostuff 1 test description,").concat(testUrlLink('services/admin/dostuff.test.1'), "\ntestteardown,tests/testteardown.sas,sasjs_test_id,not provided,,").concat(testUrlLink('testteardown')); return [4 /*yield*/, (0, test_1.runTest)(target, undefined, undefined, undefined, true)]; case 1: _a.sent(); return [4 /*yield*/, expect((0, utils_1.folderExists)(resultsFolderPath)).resolves.toEqual(true)]; case 2: _a.sent(); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsJsonPath)).resolves.toEqual(true)]; case 3: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(resultsJsonPath)]; case 4: resultsJson = _a.sent(); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsCsvPath)).resolves.toEqual(true)]; case 5: _a.sent(); parsedResults = JSON.parse(resultsJson); parsedResults.sasjs_test_meta = parsedResults.sasjs_test_meta.flatMap(function (res) { return (__assign(__assign({}, res), { results: res.results.map(function (r) { return (__assign(__assign({}, r), { sasjs_test_id: '' })); }) })); }); expect(parsedResults).toEqual(expectedResultsJson); return [4 /*yield*/, (0, utils_1.readFile)(resultsCsvPath)]; case 6: resultsCsv = _a.sent(); resultsCsv = resultsCsv.replace(/\b[0-9a-f]{8}\b-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-\b[0-9a-f]{12}\b/g, 'sasjs_test_id'); expect(resultsCsv).toEqual(expectedResultsCsv); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsXmlPath)).resolves.toEqual(true)]; case 7: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(resultsXmlPath)]; case 8: resultsXml = _a.sent(); expectedResultsXml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<testsuites id=\"\" name=\"SASjs Test Meta\" tests=\"7\" failures=\"5\">\n <testsuite id=\"testsetup\" name=\"testsetup\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"exampleprogram\" name=\"exampleprogram\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"standalone\" name=\"standalone\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"examplemacro\" name=\"examplemacro\" tests=\"1\" failures=\"0\">\n <testcase id=\"\">\n </testcase>\n </testsuite>\n <testsuite id=\"dostuff\" name=\"dostuff\" tests=\"2\" failures=\"1\">\n <testcase id=\"\">\n <failure>Description: dostuff 0 test description</failure>\n </testcase>\n <testcase id=\"\">\n </testcase>\n </testsuite>\n <testsuite id=\"testteardown\" name=\"testteardown\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n</testsuites>"; resultsXml = resultsXml.replace(/testsuites id="[^ ]*"/gm, "testsuites id=\"\""); resultsXml = resultsXml.replace(/testcase id="[^ ]*"/gm, "testcase id=\"\""); expect(resultsXml).toEqual(expectedResultsXml); coverageLcovPath = path_1.default.join(resultsFolderPath, 'coverage.lcov'); return [4 /*yield*/, expect((0, utils_1.fileExists)(coverageLcovPath)).resolves.toEqual(true)]; case 9: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(coverageLcovPath)]; case 10: coverageLcov = _a.sent(); expect(coverageLcov).toEqual(expectedCoverageLcov); return [2 /*return*/]; } }); }); }); it('should execute filtered tests and create result CSV, XML and JSON files using custom source and output locations', function () { return __awaiter(void 0, void 0, void 0, function () { var outputFolder, movedTestFlow, resultsFolderPath, resultsJsonPath, resultsCsvPath, resultsXmlPath, coverageReportPath, expectedResultsJson, expectedResultsCsv, resultsJson, parsedResults, resultsCsv, logFolder, logPath, log, resultsXml, expectedResultsXml; return __generator(this, function (_a) { switch (_a.label) { case 0: outputFolder = 'customOutPut'; movedTestFlow = 'movedTestFlow.json'; return [4 /*yield*/, (0, utils_1.copy)(path_1.default.join(__dirname, target.name, 'sasjsbuild', 'testFlow.json'), path_1.default.join(__dirname, target.name, movedTestFlow))]; case 1: _a.sent(); resultsFolderPath = path_1.default.join(__dirname, target.name, outputFolder); resultsJsonPath = path_1.default.join(resultsFolderPath, 'testResults.json'); resultsCsvPath = path_1.default.join(resultsFolderPath, 'testResults.csv'); resultsXmlPath = path_1.default.join(resultsFolderPath, 'testResults.xml'); coverageReportPath = path_1.default.join(resultsFolderPath, 'coverage.lcov'); expectedResultsJson = { csv_result_path: resultsCsvPath, xml_result_path: resultsXmlPath, coverage_report_path: coverageReportPath, tests_that_pass: '0/4 (0%)', tests_with_results: '4/5 (80%)', target_name: target.name, target_server_url: target.serverUrl, target_server_type: target.serverType, local_date_time: expect.anything(), local_user_id: expect.anything(), local_machine_name: expect.anything(), sasjs_test_meta: [ { test_target: 'testsetup', results: [ { test_loc: 'tests/testsetup.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('testsetup') } ] }, { test_target: 'standalone', results: [ { test_loc: 'tests/jobs/jobs/standalone.test.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('jobs/jobs/standalone.test') } ] }, { test_target: 'dostuff', results: [ { test_loc: 'tests/services/admin/dostuff.test.0.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'dostuff 0 test description', TEST_RESULT: 'FAIL' } ], test_url: testUrl('services/admin/dostuff.test.0') } ] }, { test_target: 'testteardown', results: [ { test_loc: 'tests/testteardown.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('testteardown') } ] } ] }; expectedResultsCsv = "test_target,test_loc,sasjs_test_id,test_suite_result,test_description,test_url\ntestsetup,tests/testsetup.sas,sasjs_test_id,not provided,,".concat(testUrlLink('testsetup'), "\nstandalone,tests/jobs/jobs/standalone.test.sas,sasjs_test_id,not provided,,").concat(testUrlLink('jobs/jobs/standalone.test'), "\ndostuff,tests/services/admin/dostuff.test.0.sas,sasjs_test_id,FAIL,dostuff 0 test description,").concat(testUrlLink('services/admin/dostuff.test.0'), "\ntestteardown,tests/testteardown.sas,sasjs_test_id,not provided,,").concat(testUrlLink('testteardown')); return [4 /*yield*/, (0, test_1.runTest)(target, ['jobs/standalone', 'shouldFail', 'services/admin/dostuff.test.0'], outputFolder, movedTestFlow, true)]; case 2: _a.sent(); return [4 /*yield*/, expect((0, utils_1.folderExists)(resultsFolderPath)).resolves.toEqual(true)]; case 3: _a.sent(); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsJsonPath)).resolves.toEqual(true)]; case 4: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(resultsJsonPath)]; case 5: resultsJson = _a.sent(); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsCsvPath)).resolves.toEqual(true)]; case 6: _a.sent(); parsedResults = JSON.parse(resultsJson); parsedResults.sasjs_test_meta = parsedResults.sasjs_test_meta.flatMap(function (res) { return (__assign(__assign({}, res), { results: res.results.map(function (r) { return (__assign(__assign({}, r), { sasjs_test_id: '' })); }) })); }); expect(parsedResults).toEqual(expectedResultsJson); return [4 /*yield*/, (0, utils_1.readFile)(resultsCsvPath)]; case 7: resultsCsv = _a.sent(); resultsCsv = resultsCsv.replace(/\b[0-9a-f]{8}\b-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-\b[0-9a-f]{12}\b/g, 'sasjs_test_id'); expect(resultsCsv).toEqual(expectedResultsCsv); logFolder = path_1.default.join(resultsFolderPath, 'logs'); logPath = path_1.default.join(logFolder, 'macros_shouldFail.test.log'); return [4 /*yield*/, expect((0, utils_1.listFilesInFolder)(logFolder)).resolves.toEqual([ 'jobs_jobs_standalone.test.log', 'macros_shouldFail.test.log', 'services_admin_dostuff.test.0.log', 'testsetup.log', 'testteardown.log' ])]; case 8: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(logPath)]; case 9: log = _a.sent(); expect(log.length).toBeGreaterThan(0); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsXmlPath)).resolves.toEqual(true)]; case 10: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(resultsXmlPath)]; case 11: resultsXml = _a.sent(); expectedResultsXml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<testsuites id=\"\" name=\"SASjs Test Meta\" tests=\"4\" failures=\"4\">\n <testsuite id=\"testsetup\" name=\"testsetup\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"standalone\" name=\"standalone\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"dostuff\" name=\"dostuff\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Description: dostuff 0 test description</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"testteardown\" name=\"testteardown\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n</testsuites>"; resultsXml = resultsXml.replace(/testsuites id="[^ ]*"/gm, "testsuites id=\"\""); resultsXml = resultsXml.replace(/testcase id="[^ ]*"/gm, "testcase id=\"\""); expect(resultsXml).toEqual(expectedResultsXml); return [2 /*return*/]; } }); }); }); }); describe('SASJS', function () { var target = generateTarget(utils_1.ServerType.Sasjs); var testUrl = function (test) { return "".concat((0, test_1.getTestUrl)(target, "/Public/app/cli-tests/".concat(target.name, "/tests/").concat(test)), "&_contextName=").concat(encodeURIComponent(target.contextName)); }; var testUrlLink = function (test) { return "\"=HYPERLINK(\"\"".concat(testUrl(test), "\"\")\""); }; beforeAll(function () { return __awaiter(void 0, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, test_2.createTestApp)(__dirname, target.name)]; case 1: _a.sent(); return [4 /*yield*/, copyTestFiles(target.name)]; case 2: _a.sent(); return [4 /*yield*/, (0, __1.build)(target)]; case 3: _a.sent(); process.logger = new utils_1.Logger(utils_1.LogLevel.Off); return [2 /*return*/]; } }); }); }); beforeEach(function () { setupMocksForSASJS(); }); afterAll(function () { return __awaiter(void 0, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, test_2.removeTestApp)(__dirname, target.name)]; case 1: _a.sent(); return [2 /*return*/]; } }); }); }); it('should execute tests and create result CSV, XML and JSON files using default source and output locations', function () { return __awaiter(void 0, void 0, void 0, function () { var resultsFolderPath, resultsJsonPath, resultsCsvPath, resultsXmlPath, coverageReportPath, expectedResultsJson, expectedResultsCsv, resultsJson, parsedResults, resultsCsv, resultsXml, expectedResultsXml, coverageLcovPath, coverageLcov; return __generator(this, function (_a) { switch (_a.label) { case 0: resultsFolderPath = path_1.default.join(__dirname, target.name, 'sasjsresults'); resultsJsonPath = path_1.default.join(resultsFolderPath, 'testResults.json'); resultsCsvPath = path_1.default.join(resultsFolderPath, 'testResults.csv'); resultsXmlPath = path_1.default.join(resultsFolderPath, 'testResults.xml'); coverageReportPath = path_1.default.join(resultsFolderPath, 'coverage.lcov'); expectedResultsJson = { csv_result_path: resultsCsvPath, xml_result_path: resultsXmlPath, coverage_report_path: coverageReportPath, tests_that_pass: '2/8 (25%)', tests_with_results: '8/8 (100%)', target_name: target.name, target_server_url: target.serverUrl, target_server_type: target.serverType, local_date_time: expect.anything(), local_user_id: expect.anything(), local_machine_name: expect.anything(), sasjs_test_meta: [ { test_target: 'testsetup', results: [ { test_loc: 'tests/testsetup.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('testsetup') } ] }, { test_target: 'exampleprogram', results: [ { test_loc: 'tests/jobs/jobs/exampleprogram.test.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('jobs/jobs/exampleprogram.test') } ] }, { test_target: 'standalone', results: [ { test_loc: 'tests/jobs/jobs/standalone.test.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('jobs/jobs/standalone.test') } ] }, { test_target: 'examplemacro', results: [ { test_loc: 'tests/macros/examplemacro.test.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'examplemacro test.1 description', TEST_RESULT: 'PASS' } ], test_url: testUrl('macros/examplemacro.test') } ] }, { test_target: 'shouldFail', results: [ { test_loc: 'tests/macros/shouldFail.test.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('macros/shouldFail.test') } ] }, { test_target: 'dostuff', results: [ { test_loc: 'tests/services/admin/dostuff.test.0.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'dostuff 0 test description', TEST_RESULT: 'FAIL' } ], test_url: testUrl('services/admin/dostuff.test.0') }, { test_loc: 'tests/services/admin/dostuff.test.1.sas', sasjs_test_id: '', result: [ { TEST_DESCRIPTION: 'dostuff 1 test description', TEST_RESULT: 'PASS' } ], test_url: testUrl('services/admin/dostuff.test.1') } ] }, { test_target: 'testteardown', results: [ { test_loc: 'tests/testteardown.sas', sasjs_test_id: '', result: 'not provided', test_url: testUrl('testteardown') } ] } ] }; expectedResultsCsv = "test_target,test_loc,sasjs_test_id,test_suite_result,test_description,test_url\ntestsetup,tests/testsetup.sas,sasjs_test_id,not provided,,".concat(testUrlLink('testsetup'), "\nexampleprogram,tests/jobs/jobs/exampleprogram.test.sas,sasjs_test_id,not provided,,").concat(testUrlLink('jobs/jobs/exampleprogram.test'), "\nstandalone,tests/jobs/jobs/standalone.test.sas,sasjs_test_id,not provided,,").concat(testUrlLink('jobs/jobs/standalone.test'), "\nexamplemacro,tests/macros/examplemacro.test.sas,sasjs_test_id,PASS,examplemacro test.1 description,").concat(testUrlLink('macros/examplemacro.test'), "\nshouldFail,tests/macros/shouldFail.test.sas,sasjs_test_id,not provided,,").concat(testUrlLink('macros/shouldFail.test'), "\ndostuff,tests/services/admin/dostuff.test.0.sas,sasjs_test_id,FAIL,dostuff 0 test description,").concat(testUrlLink('services/admin/dostuff.test.0'), "\ndostuff,tests/services/admin/dostuff.test.1.sas,sasjs_test_id,PASS,dostuff 1 test description,").concat(testUrlLink('services/admin/dostuff.test.1'), "\ntestteardown,tests/testteardown.sas,sasjs_test_id,not provided,,").concat(testUrlLink('testteardown')); return [4 /*yield*/, (0, test_1.runTest)(target, undefined, undefined, undefined, true)]; case 1: _a.sent(); return [4 /*yield*/, expect((0, utils_1.folderExists)(resultsFolderPath)).resolves.toEqual(true)]; case 2: _a.sent(); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsJsonPath)).resolves.toEqual(true)]; case 3: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(resultsJsonPath)]; case 4: resultsJson = _a.sent(); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsCsvPath)).resolves.toEqual(true)]; case 5: _a.sent(); parsedResults = JSON.parse(resultsJson); parsedResults.sasjs_test_meta = parsedResults.sasjs_test_meta.flatMap(function (res) { return (__assign(__assign({}, res), { results: res.results.map(function (r) { return (__assign(__assign({}, r), { sasjs_test_id: '' })); }) })); }); expect(parsedResults).toEqual(expectedResultsJson); return [4 /*yield*/, (0, utils_1.readFile)(resultsCsvPath)]; case 6: resultsCsv = _a.sent(); resultsCsv = resultsCsv.replace(/\b[0-9a-f]{8}\b-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-\b[0-9a-f]{12}\b/g, 'sasjs_test_id'); expect(resultsCsv).toEqual(expectedResultsCsv); return [4 /*yield*/, expect((0, utils_1.fileExists)(resultsXmlPath)).resolves.toEqual(true)]; case 7: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(resultsXmlPath)]; case 8: resultsXml = _a.sent(); expectedResultsXml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<testsuites id=\"\" name=\"SASjs Test Meta\" tests=\"8\" failures=\"6\">\n <testsuite id=\"testsetup\" name=\"testsetup\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"exampleprogram\" name=\"exampleprogram\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"standalone\" name=\"standalone\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"examplemacro\" name=\"examplemacro\" tests=\"1\" failures=\"0\">\n <testcase id=\"\">\n </testcase>\n </testsuite>\n <testsuite id=\"shouldFail\" name=\"shouldFail\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n <testsuite id=\"dostuff\" name=\"dostuff\" tests=\"2\" failures=\"1\">\n <testcase id=\"\">\n <failure>Description: dostuff 0 test description</failure>\n </testcase>\n <testcase id=\"\">\n </testcase>\n </testsuite>\n <testsuite id=\"testteardown\" name=\"testteardown\" tests=\"1\" failures=\"1\">\n <testcase id=\"\">\n <failure>Status was not provided</failure>\n </testcase>\n </testsuite>\n</testsuites>"; resultsXml = resultsXml.replace(/testsuites id="[^ ]*"/gm, "testsuites id=\"\""); resultsXml = resultsXml.replace(/testcase id="[^ ]*"/gm, "testcase id=\"\""); expect(resultsXml).toEqual(expectedResultsXml); coverageLcovPath = path_1.default.join(resultsFolderPath, 'coverage.lcov'); return [4 /*yield*/, expect((0, utils_1.fileExists)(coverageLcovPath)).resolves.toEqual(true)]; case 9: _a.sent(); return [4 /*yield*/, (0, utils_1.readFile)(coverageLcovPath)]; case 10: coverageLcov = _a.sent(); expect(coverageLcov).toEqual(expectedCoverageLcov); return [2 /*return*/]; } }); }); }); it('should FAIL test suit if any test has status FAIL', function () { return __awaiter(void 0, void 0, void 0, function () { var testService; return __generator(this, function (_a) { switch (_a.label) { case 0: testService = 'notexisting'; jest.spyOn(process.logger, 'table'); jest.spyOn(utilsModule, 'uuidv4').mockImplementation(function () { return 'uuidv4'; }); jest.spyOn(fileModule, 'readFile').mockImplementation(function () { return Promise.resolve("{\n \"tests\": [\n \"tests/macros/".concat(testService, ".test.sas\"\n ],\n \"testSetUp\": \"\",\n \"testTearDown\": \"\"\n}")); }); return [4 /*yield*/, (0, test_1.runTest)(target, undefined, undefined, undefined, true)]; case 1: _a.sent(); expect(process.logger.table).toHaveBeenCalledWith([['uuidv4', testService, chalk_1.default.redBright(types_1.TestResultStatus.fail)]], { head: ['SASjs Test ID', 'Test Target', 'Test Suite Result'] }); return [2 /*return*/]; } }); }); }); it('should fail on tests failing', function () { return __awaiter(void 0, void 0, void 0, function () { var error; return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, test_1.runTest)(target).catch(function (err) { return err; })]; case 1: error = _a.sent(); expect(error).toEqual('6 tests completed with failures!'); return [2 /*return*/]; } }); }); }); it('should not throw error on tests failing', function () { return __awaiter(void 0, void 0, void 0, function () { var error; return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, test_1.runTest)(target, undefined, undefined, undefined, true).catch(function (err) { return err; })]; case 1: error = _a.sent(); expect(error).toBeUndefined(); return [2 /*return*/]; } }); }); }); }); }); function generateTarget(serverType) { if (serverType === void 0) { serverType = utils_1.ServerType.SasViya; } dotenv_1.default.config(); var timestamp = (0, utils_1.generateTimestamp)(); var targetName = "cli-tests-test-command-".concat(serverType, "-").concat(timestamp); return new utils_1.Target({ name: targetName, serverType: serverType, serverUrl: (serverType === utils_1.ServerType.SasViya ? process.env.VIYA_SERVER_URL : serverType === utils_1.ServerType.Sas9 ? process.env.SAS9_SERVER_URL : process.env.SASJS_SERVER_URL), appLoc: "/Public/app/cli-tests/".concat(targetName), contextName: setConstants_1.contextName, macroFolders: ['sasjs/macros'], serviceConfig: { serviceFolders: [], initProgram: 'sasjs/build/serviceinit.sas', termProgram: 'sasjs/build/serviceterm.sas', macroVars: { name: 'viyavalue', extravar: 'this too' } }, jobConfig: { jobFolders: ['sasjs/jobs'], initProgram: '', termProgram: '', macroVars: {} }, authConfig: { client: process.env.CLIENT, secret: process.env.SECRET, access_token: process.env.ACCESS_TOKEN, refresh_token: process.env.REFRESH_TOKEN }, deployConfig: { deployServicePack: true, deployScripts: [] }, testConfig: { initProgram: path_1.default.join('sasjs', 'tests', 'testinit.sas'), termProgram: path_1.default.join('sasjs', 'tests', 'testterm.sas'), macroVars: { testVar: 'testValue' }, testSetUp: path_1.default.join('sasjs', 'tests', 'testsetup.sas'), testTearDown: path_1.default.join('sasjs', 'tests', 'testteardown.sas') } }); } var copyTestFiles = function (appName) { return __awaiter(void 0, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, (0, utils_1.copy)(path_1.default.join(__dirname, 'testServicesFiles'), path_1.default.join(__dirname, appName, 'sasjs', 'services', 'admin'))]; case 1: _a.sent(); return [4 /*yield*/, (0, utils_1.copy)(path_1.default.join(__dirname, 'testJobsFiles'), path_1.default.join(__dirname, appName, 'sasjs', 'jobs'))]; case 2: _a.sent(); return [4 /*yield*/, (0, utils_1.copy)(path_1.default.join(__dirname, 'testFiles'), path_1.default.join(__dirname, appName, 'sasjs', 'tes