2021-01-12 20:04:16 +01:00
|
|
|
/*
|
|
|
|
This file is part of GNU Taler
|
|
|
|
(C) 2021 Taler Systems S.A.
|
|
|
|
|
|
|
|
GNU Taler is free software; you can redistribute it and/or modify it under the
|
|
|
|
terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation; either version 3, or (at your option) any later version.
|
|
|
|
|
|
|
|
GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
|
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
2021-02-03 14:35:29 +01:00
|
|
|
import {
|
2021-08-20 12:31:35 +02:00
|
|
|
minimatch
|
|
|
|
} from "@gnu-taler/taler-util";
|
|
|
|
import {
|
2021-02-03 14:35:29 +01:00
|
|
|
GlobalTestState,
|
|
|
|
runTestWithState,
|
|
|
|
shouldLingerInTest,
|
|
|
|
TestRunResult,
|
2021-10-20 13:06:31 +02:00
|
|
|
} from "../harness/harness.js";
|
2021-01-12 20:04:16 +01:00
|
|
|
import { runPaymentTest } from "./test-payment";
|
2021-07-08 09:33:41 +02:00
|
|
|
import { runPaymentDemoTest } from "./test-payment-on-demo";
|
2021-01-12 20:04:16 +01:00
|
|
|
import * as fs from "fs";
|
|
|
|
import * as path from "path";
|
|
|
|
import * as os from "os";
|
2021-01-13 13:17:38 +01:00
|
|
|
import * as child_process from "child_process";
|
2021-01-12 20:04:16 +01:00
|
|
|
import { runBankApiTest } from "./test-bank-api";
|
|
|
|
import { runClaimLoopTest } from "./test-claim-loop";
|
|
|
|
import { runExchangeManagementTest } from "./test-exchange-management";
|
|
|
|
import { runFeeRegressionTest } from "./test-fee-regression";
|
|
|
|
import { runMerchantLongpollingTest } from "./test-merchant-longpolling";
|
|
|
|
import { runMerchantRefundApiTest } from "./test-merchant-refund-api";
|
|
|
|
import { runPayAbortTest } from "./test-pay-abort";
|
|
|
|
import { runPayPaidTest } from "./test-pay-paid";
|
|
|
|
import { runPaymentClaimTest } from "./test-payment-claim";
|
|
|
|
import { runPaymentFaultTest } from "./test-payment-fault";
|
|
|
|
import { runPaymentIdempotencyTest } from "./test-payment-idempotency";
|
|
|
|
import { runPaymentMultipleTest } from "./test-payment-multiple";
|
|
|
|
import { runPaymentTransientTest } from "./test-payment-transient";
|
|
|
|
import { runPaywallFlowTest } from "./test-paywall-flow";
|
|
|
|
import { runRefundAutoTest } from "./test-refund-auto";
|
|
|
|
import { runRefundGoneTest } from "./test-refund-gone";
|
|
|
|
import { runRefundIncrementalTest } from "./test-refund-incremental";
|
|
|
|
import { runRefundTest } from "./test-refund";
|
|
|
|
import { runRevocationTest } from "./test-revocation";
|
|
|
|
import { runTimetravelAutorefreshTest } from "./test-timetravel-autorefresh";
|
|
|
|
import { runTimetravelWithdrawTest } from "./test-timetravel-withdraw";
|
|
|
|
import { runTippingTest } from "./test-tipping";
|
|
|
|
import { runWallettestingTest } from "./test-wallettesting";
|
|
|
|
import { runTestWithdrawalManualTest } from "./test-withdrawal-manual";
|
|
|
|
import { runWithdrawalAbortBankTest } from "./test-withdrawal-abort-bank";
|
|
|
|
import { runWithdrawalBankIntegratedTest } from "./test-withdrawal-bank-integrated";
|
2021-01-14 01:47:56 +01:00
|
|
|
import { runMerchantExchangeConfusionTest } from "./test-merchant-exchange-confusion";
|
2021-01-17 01:18:37 +01:00
|
|
|
import { runLibeufinBasicTest } from "./test-libeufin-basic";
|
2021-09-02 11:29:13 +02:00
|
|
|
import { runLibeufinC5xTest } from "./test-libeufin-c5x";
|
2021-09-02 23:48:41 +02:00
|
|
|
import { runLibeufinNexusBalanceTest } from "./test-libeufin-nexus-balance";
|
2021-10-02 10:20:20 +02:00
|
|
|
import { runLibeufinBadGatewayTest } from "./test-libeufin-bad-gateway";
|
2021-07-27 10:08:00 +02:00
|
|
|
import { runLibeufinKeyrotationTest } from "./test-libeufin-keyrotation";
|
2021-02-15 10:43:05 +01:00
|
|
|
import { runLibeufinRefundTest } from "./test-libeufin-refund";
|
2021-05-07 08:49:26 +02:00
|
|
|
import { runLibeufinRefundMultipleUsersTest } from "./test-libeufin-refund-multiple-users";
|
2021-02-03 13:05:10 +01:00
|
|
|
import { runLibeufinTutorialTest } from "./test-libeufin-tutorial";
|
2021-05-07 11:39:01 +02:00
|
|
|
import { runLibeufinApiPermissionsTest } from "./test-libeufin-api-permissions";
|
2021-05-18 18:01:41 +02:00
|
|
|
import { runLibeufinApiFacadeTest } from "./test-libeufin-api-facade";
|
2021-08-15 11:35:39 +02:00
|
|
|
import { runLibeufinApiFacadeBadRequestTest } from "./test-libeufin-api-facade-bad-request";
|
2021-08-17 16:53:16 +02:00
|
|
|
import { runLibeufinAnastasisFacadeTest } from "./test-libeufin-facade-anastasis";
|
2021-05-14 12:50:44 +02:00
|
|
|
import { runLibeufinApiSchedulingTest } from "./test-libeufin-api-scheduling";
|
2021-05-12 11:52:38 +02:00
|
|
|
import { runLibeufinApiBankconnectionTest } from "./test-libeufin-api-bankconnection";
|
2021-05-10 11:54:59 +02:00
|
|
|
import { runLibeufinApiUsersTest } from "./test-libeufin-api-users";
|
2021-05-12 09:58:17 +02:00
|
|
|
import { runLibeufinApiBankaccountTest } from "./test-libeufin-api-bankaccount";
|
2021-08-07 17:06:39 +02:00
|
|
|
import { runLibeufinApiSandboxTransactionsTest } from "./test-libeufin-api-sandbox-transactions";
|
2021-08-24 12:46:21 +02:00
|
|
|
import { runLibeufinApiSandboxCamtTest } from "./test-libeufin-api-sandbox-camt";
|
2021-08-23 16:31:48 +02:00
|
|
|
import { runLibeufinSandboxWireTransferCliTest } from "./test-libeufin-sandbox-wire-transfer-cli";
|
2021-01-18 23:35:41 +01:00
|
|
|
import { runDepositTest } from "./test-deposit";
|
2021-02-04 15:07:47 +01:00
|
|
|
import CancellationToken from "cancellationtoken";
|
2021-03-02 20:03:39 +01:00
|
|
|
import { runMerchantInstancesTest } from "./test-merchant-instances";
|
2021-03-02 21:47:57 +01:00
|
|
|
import { runMerchantInstancesUrlsTest } from "./test-merchant-instances-urls";
|
2021-03-03 21:20:05 +01:00
|
|
|
import { runWalletBackupBasicTest } from "./test-wallet-backup-basic";
|
2021-03-04 15:59:31 +01:00
|
|
|
import { runMerchantInstancesDeleteTest } from "./test-merchant-instances-delete";
|
2021-03-11 13:08:41 +01:00
|
|
|
import { runWalletBackupDoublespendTest } from "./test-wallet-backup-doublespend";
|
2021-06-16 12:02:08 +02:00
|
|
|
import { runPaymentForgettableTest } from "./test-payment-forgettable.js";
|
2021-07-12 15:55:19 +02:00
|
|
|
import { runPaymentZeroTest } from "./test-payment-zero.js";
|
2021-08-12 21:01:40 +02:00
|
|
|
import { runMerchantSpecPublicOrdersTest } from "./test-merchant-spec-public-orders.js";
|
2021-08-19 17:59:52 +02:00
|
|
|
import { runExchangeTimetravelTest } from "./test-exchange-timetravel.js";
|
2021-08-23 22:28:36 +02:00
|
|
|
import { runDenomUnofferedTest } from "./test-denom-unoffered.js";
|
2022-01-24 20:34:29 +01:00
|
|
|
import { runWithdrawalFakebankTest } from "./test-withdrawal-fakebank.js";
|
2021-01-12 20:04:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Test runner.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Spec for one test.
|
|
|
|
*/
|
|
|
|
interface TestMainFunction {
|
|
|
|
(t: GlobalTestState): Promise<void>;
|
2021-02-04 16:22:05 +01:00
|
|
|
timeoutMs?: number;
|
2021-08-19 13:48:36 +02:00
|
|
|
excludeByDefault?: boolean;
|
2021-03-02 14:19:01 +01:00
|
|
|
suites?: string[];
|
2021-01-12 20:04:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const allTests: TestMainFunction[] = [
|
|
|
|
runBankApiTest,
|
|
|
|
runClaimLoopTest,
|
2021-01-18 23:35:41 +01:00
|
|
|
runDepositTest,
|
2021-08-23 22:28:36 +02:00
|
|
|
runDenomUnofferedTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runExchangeManagementTest,
|
2021-08-19 17:59:52 +02:00
|
|
|
runExchangeTimetravelTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runFeeRegressionTest,
|
2021-01-17 01:18:37 +01:00
|
|
|
runLibeufinBasicTest,
|
2021-07-27 10:08:00 +02:00
|
|
|
runLibeufinKeyrotationTest,
|
2021-02-03 13:05:10 +01:00
|
|
|
runLibeufinTutorialTest,
|
2021-02-15 10:43:05 +01:00
|
|
|
runLibeufinRefundTest,
|
2021-09-02 11:29:13 +02:00
|
|
|
runLibeufinC5xTest,
|
2021-09-02 23:48:41 +02:00
|
|
|
runLibeufinNexusBalanceTest,
|
2021-10-02 10:20:20 +02:00
|
|
|
runLibeufinBadGatewayTest,
|
2021-05-07 08:49:26 +02:00
|
|
|
runLibeufinRefundMultipleUsersTest,
|
2021-05-07 11:39:01 +02:00
|
|
|
runLibeufinApiPermissionsTest,
|
2021-05-18 18:01:41 +02:00
|
|
|
runLibeufinApiFacadeTest,
|
2021-08-15 11:35:39 +02:00
|
|
|
runLibeufinApiFacadeBadRequestTest,
|
2021-08-17 16:53:16 +02:00
|
|
|
runLibeufinAnastasisFacadeTest,
|
2021-05-14 12:50:44 +02:00
|
|
|
runLibeufinApiSchedulingTest,
|
2021-05-10 11:54:59 +02:00
|
|
|
runLibeufinApiUsersTest,
|
2021-05-12 09:58:17 +02:00
|
|
|
runLibeufinApiBankaccountTest,
|
2021-05-12 11:52:38 +02:00
|
|
|
runLibeufinApiBankconnectionTest,
|
2021-08-07 17:06:39 +02:00
|
|
|
runLibeufinApiSandboxTransactionsTest,
|
2021-08-24 12:46:21 +02:00
|
|
|
runLibeufinApiSandboxCamtTest,
|
2021-08-23 16:31:48 +02:00
|
|
|
runLibeufinSandboxWireTransferCliTest,
|
2021-01-17 01:18:37 +01:00
|
|
|
runMerchantExchangeConfusionTest,
|
2021-03-02 20:03:39 +01:00
|
|
|
runMerchantInstancesTest,
|
2021-03-04 15:59:31 +01:00
|
|
|
runMerchantInstancesDeleteTest,
|
2021-03-02 21:47:57 +01:00
|
|
|
runMerchantInstancesUrlsTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runMerchantLongpollingTest,
|
2021-08-12 21:01:40 +02:00
|
|
|
runMerchantSpecPublicOrdersTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runMerchantRefundApiTest,
|
|
|
|
runPayAbortTest,
|
|
|
|
runPaymentClaimTest,
|
|
|
|
runPaymentFaultTest,
|
2021-06-16 12:02:08 +02:00
|
|
|
runPaymentForgettableTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runPaymentIdempotencyTest,
|
|
|
|
runPaymentMultipleTest,
|
2021-01-13 00:50:56 +01:00
|
|
|
runPaymentTest,
|
2021-07-08 09:33:41 +02:00
|
|
|
runPaymentDemoTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runPaymentTransientTest,
|
2021-07-12 15:55:19 +02:00
|
|
|
runPaymentZeroTest,
|
2021-01-13 00:50:56 +01:00
|
|
|
runPayPaidTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runPaywallFlowTest,
|
|
|
|
runRefundAutoTest,
|
|
|
|
runRefundGoneTest,
|
|
|
|
runRefundIncrementalTest,
|
|
|
|
runRefundTest,
|
|
|
|
runRevocationTest,
|
2021-01-17 01:18:37 +01:00
|
|
|
runTestWithdrawalManualTest,
|
2022-01-24 20:34:29 +01:00
|
|
|
runWithdrawalFakebankTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runTimetravelAutorefreshTest,
|
|
|
|
runTimetravelWithdrawTest,
|
|
|
|
runTippingTest,
|
2021-03-03 21:20:05 +01:00
|
|
|
runWalletBackupBasicTest,
|
2021-03-11 13:08:41 +01:00
|
|
|
runWalletBackupDoublespendTest,
|
2021-01-12 20:04:16 +01:00
|
|
|
runWallettestingTest,
|
|
|
|
runWithdrawalAbortBankTest,
|
|
|
|
runWithdrawalBankIntegratedTest,
|
|
|
|
];
|
|
|
|
|
|
|
|
export interface TestRunSpec {
|
2021-03-02 14:19:01 +01:00
|
|
|
includePattern?: string;
|
|
|
|
suiteSpec?: string;
|
2021-06-09 15:26:18 +02:00
|
|
|
dryRun?: boolean;
|
2021-06-17 14:18:05 +02:00
|
|
|
verbosity: number;
|
2021-01-12 20:04:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface TestInfo {
|
|
|
|
name: string;
|
2021-08-19 13:48:36 +02:00
|
|
|
suites: string[];
|
|
|
|
excludeByDefault: boolean;
|
2021-01-12 20:04:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function updateCurrentSymlink(testDir: string): void {
|
2021-07-09 13:15:50 +02:00
|
|
|
const currLink = path.join(
|
|
|
|
os.tmpdir(),
|
2021-07-12 15:55:19 +02:00
|
|
|
`taler-integrationtests-${os.userInfo().username}-current`,
|
2021-07-09 13:15:50 +02:00
|
|
|
);
|
2021-01-12 20:04:16 +01:00
|
|
|
try {
|
|
|
|
fs.unlinkSync(currLink);
|
|
|
|
} catch (e) {
|
|
|
|
// Ignore
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
fs.symlinkSync(testDir, currLink);
|
|
|
|
} catch (e) {
|
|
|
|
console.log(e);
|
|
|
|
// Ignore
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTestName(tf: TestMainFunction): string {
|
|
|
|
const res = tf.name.match(/run([a-zA-Z0-9]*)Test/);
|
|
|
|
if (!res) {
|
|
|
|
throw Error("invalid test name, must be 'run${NAME}Test'");
|
|
|
|
}
|
|
|
|
return res[1]
|
2021-01-14 01:47:56 +01:00
|
|
|
.replace(/[a-z0-9][A-Z]/g, (x) => {
|
2021-01-12 20:04:16 +01:00
|
|
|
return x[0] + "-" + x[1];
|
|
|
|
})
|
|
|
|
.toLowerCase();
|
|
|
|
}
|
|
|
|
|
2021-01-13 13:17:38 +01:00
|
|
|
interface RunTestChildInstruction {
|
|
|
|
testName: string;
|
|
|
|
testRootDir: string;
|
|
|
|
}
|
|
|
|
|
2021-01-12 20:04:16 +01:00
|
|
|
export async function runTests(spec: TestRunSpec) {
|
|
|
|
const testRootDir = fs.mkdtempSync(
|
|
|
|
path.join(os.tmpdir(), "taler-integrationtests-"),
|
|
|
|
);
|
|
|
|
updateCurrentSymlink(testRootDir);
|
|
|
|
console.log("testsuite root directory: ", testRootDir);
|
|
|
|
|
|
|
|
const testResults: TestRunResult[] = [];
|
|
|
|
|
2021-01-13 13:17:38 +01:00
|
|
|
let currentChild: child_process.ChildProcess | undefined;
|
|
|
|
|
2021-02-04 15:32:26 +01:00
|
|
|
const handleSignal = (s: NodeJS.Signals) => {
|
|
|
|
console.log(`received signal ${s} in test parent`);
|
2021-01-13 13:17:38 +01:00
|
|
|
if (currentChild) {
|
|
|
|
currentChild.kill("SIGTERM");
|
|
|
|
}
|
2021-02-04 15:32:26 +01:00
|
|
|
reportAndQuit(testRootDir, testResults, true);
|
2021-01-13 13:17:38 +01:00
|
|
|
};
|
|
|
|
|
2021-02-04 15:32:26 +01:00
|
|
|
process.on("SIGINT", (s) => handleSignal(s));
|
|
|
|
process.on("SIGTERM", (s) => handleSignal(s));
|
2021-01-13 13:33:25 +01:00
|
|
|
//process.on("unhandledRejection", handleSignal);
|
|
|
|
//process.on("uncaughtException", handleSignal);
|
2021-01-13 13:17:38 +01:00
|
|
|
|
2021-03-02 14:19:01 +01:00
|
|
|
let suites: Set<string> | undefined;
|
|
|
|
|
|
|
|
if (spec.suiteSpec) {
|
|
|
|
suites = new Set(spec.suiteSpec.split(",").map((x) => x.trim()));
|
|
|
|
}
|
|
|
|
|
2021-01-12 20:04:16 +01:00
|
|
|
for (const [n, testCase] of allTests.entries()) {
|
|
|
|
const testName = getTestName(testCase);
|
2021-08-20 12:31:35 +02:00
|
|
|
if (spec.includePattern && !minimatch(testName, spec.includePattern)) {
|
2021-03-02 14:19:01 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (suites) {
|
|
|
|
const ts = new Set(testCase.suites ?? []);
|
|
|
|
const intersection = new Set([...suites].filter((x) => ts.has(x)));
|
|
|
|
if (intersection.size === 0) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-08-19 13:48:36 +02:00
|
|
|
} else {
|
|
|
|
if (testCase.excludeByDefault) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-02 14:19:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (spec.dryRun) {
|
|
|
|
console.log(`dry run: would run test ${testName}`);
|
2021-01-12 20:04:16 +01:00
|
|
|
continue;
|
|
|
|
}
|
2021-01-13 13:17:38 +01:00
|
|
|
|
|
|
|
const testInstr: RunTestChildInstruction = {
|
|
|
|
testName,
|
|
|
|
testRootDir,
|
|
|
|
};
|
|
|
|
|
2021-01-14 17:24:44 +01:00
|
|
|
currentChild = child_process.fork(__filename, ["__TWCLI_TESTWORKER"], {
|
2021-01-13 13:17:38 +01:00
|
|
|
env: {
|
|
|
|
TWCLI_RUN_TEST_INSTRUCTION: JSON.stringify(testInstr),
|
|
|
|
...process.env,
|
|
|
|
},
|
|
|
|
stdio: ["pipe", "pipe", "pipe", "ipc"],
|
2021-01-12 20:04:16 +01:00
|
|
|
});
|
2021-01-13 13:17:38 +01:00
|
|
|
|
2021-01-13 13:33:25 +01:00
|
|
|
const testDir = path.join(testRootDir, testName);
|
|
|
|
fs.mkdirSync(testDir, { recursive: true });
|
|
|
|
|
|
|
|
const harnessLogFilename = path.join(testRootDir, testName, "harness.log");
|
|
|
|
const harnessLogStream = fs.createWriteStream(harnessLogFilename);
|
|
|
|
|
2021-06-17 14:18:05 +02:00
|
|
|
if (spec.verbosity > 0) {
|
|
|
|
currentChild.stderr?.pipe(process.stderr);
|
|
|
|
currentChild.stdout?.pipe(process.stdout);
|
|
|
|
}
|
2021-01-13 13:17:38 +01:00
|
|
|
|
2021-01-13 13:33:25 +01:00
|
|
|
currentChild.stdout?.pipe(harnessLogStream);
|
|
|
|
currentChild.stderr?.pipe(harnessLogStream);
|
|
|
|
|
2021-02-04 16:22:05 +01:00
|
|
|
const defaultTimeout = 60000;
|
|
|
|
const testTimeoutMs = testCase.timeoutMs ?? defaultTimeout;
|
2021-02-04 15:07:47 +01:00
|
|
|
|
2021-02-04 19:27:10 +01:00
|
|
|
console.log(`running ${testName} with timeout ${testTimeoutMs}ms`);
|
|
|
|
|
2021-02-04 16:22:05 +01:00
|
|
|
const { token } = CancellationToken.timeout(testTimeoutMs);
|
2021-02-04 15:07:47 +01:00
|
|
|
|
|
|
|
const resultPromise: Promise<TestRunResult> = new Promise(
|
|
|
|
(resolve, reject) => {
|
|
|
|
let msg: TestRunResult | undefined;
|
|
|
|
currentChild!.on("message", (m) => {
|
|
|
|
if (token.isCancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
msg = m as TestRunResult;
|
|
|
|
});
|
|
|
|
currentChild!.on("exit", (code, signal) => {
|
|
|
|
if (token.isCancelled) {
|
|
|
|
return;
|
|
|
|
}
|
2021-02-04 15:32:26 +01:00
|
|
|
console.log(`process exited code=${code} signal=${signal}`);
|
2021-02-04 15:07:47 +01:00
|
|
|
if (signal) {
|
|
|
|
reject(new Error(`test worker exited with signal ${signal}`));
|
|
|
|
} else if (code != 0) {
|
|
|
|
reject(new Error(`test worker exited with code ${code}`));
|
|
|
|
} else if (!msg) {
|
|
|
|
reject(
|
|
|
|
new Error(
|
|
|
|
`test worker exited without giving back the test results`,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
resolve(msg);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
currentChild!.on("error", (err) => {
|
|
|
|
if (token.isCancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
reject(err);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
let result: TestRunResult;
|
|
|
|
|
|
|
|
try {
|
|
|
|
result = await token.racePromise(resultPromise);
|
2021-10-14 11:36:43 +02:00
|
|
|
} catch (e: any) {
|
2021-02-04 15:07:47 +01:00
|
|
|
console.error(`test ${testName} timed out`);
|
|
|
|
if (token.isCancelled) {
|
|
|
|
result = {
|
|
|
|
status: "fail",
|
|
|
|
reason: "timeout",
|
|
|
|
timeSec: testTimeoutMs / 1000,
|
|
|
|
name: testName,
|
|
|
|
};
|
|
|
|
currentChild.kill("SIGTERM");
|
|
|
|
} else {
|
|
|
|
throw Error(e);
|
|
|
|
}
|
|
|
|
}
|
2021-01-13 13:17:38 +01:00
|
|
|
|
2021-01-13 13:33:25 +01:00
|
|
|
harnessLogStream.close();
|
|
|
|
|
2021-01-13 13:17:38 +01:00
|
|
|
console.log(`parent: got result ${JSON.stringify(result)}`);
|
|
|
|
|
2021-01-12 20:04:16 +01:00
|
|
|
testResults.push(result);
|
2021-02-04 15:32:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
reportAndQuit(testRootDir, testResults);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function reportAndQuit(
|
|
|
|
testRootDir: string,
|
|
|
|
testResults: TestRunResult[],
|
|
|
|
interrupted: boolean = false,
|
|
|
|
): never {
|
|
|
|
let numTotal = 0;
|
|
|
|
let numFail = 0;
|
|
|
|
let numSkip = 0;
|
|
|
|
let numPass = 0;
|
|
|
|
|
|
|
|
for (const result of testResults) {
|
2021-01-12 20:04:16 +01:00
|
|
|
numTotal++;
|
|
|
|
if (result.status === "fail") {
|
|
|
|
numFail++;
|
|
|
|
} else if (result.status === "skip") {
|
|
|
|
numSkip++;
|
|
|
|
} else if (result.status === "pass") {
|
|
|
|
numPass++;
|
|
|
|
}
|
|
|
|
}
|
2021-01-13 13:17:38 +01:00
|
|
|
|
2021-01-12 20:04:16 +01:00
|
|
|
const resultsFile = path.join(testRootDir, "results.json");
|
|
|
|
fs.writeFileSync(
|
|
|
|
path.join(testRootDir, "results.json"),
|
2021-02-04 15:32:26 +01:00
|
|
|
JSON.stringify({ testResults, interrupted }, undefined, 2),
|
2021-01-12 20:04:16 +01:00
|
|
|
);
|
2021-02-04 15:32:26 +01:00
|
|
|
if (interrupted) {
|
|
|
|
console.log("test suite was interrupted");
|
|
|
|
}
|
2021-01-12 20:04:16 +01:00
|
|
|
console.log(`See ${resultsFile} for details`);
|
2021-01-13 00:50:56 +01:00
|
|
|
console.log(`Skipped: ${numSkip}/${numTotal}`);
|
|
|
|
console.log(`Failed: ${numFail}/${numTotal}`);
|
2021-01-12 20:04:16 +01:00
|
|
|
console.log(`Passed: ${numPass}/${numTotal}`);
|
2021-02-04 15:32:26 +01:00
|
|
|
|
|
|
|
if (interrupted) {
|
|
|
|
process.exit(3);
|
|
|
|
} else if (numPass < numTotal - numSkip) {
|
2021-01-13 00:50:56 +01:00
|
|
|
process.exit(1);
|
2021-02-04 15:32:26 +01:00
|
|
|
} else {
|
|
|
|
process.exit(0);
|
2021-01-13 00:50:56 +01:00
|
|
|
}
|
2021-01-12 20:04:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export function getTestInfo(): TestInfo[] {
|
|
|
|
return allTests.map((x) => ({
|
|
|
|
name: getTestName(x),
|
2021-08-19 13:48:36 +02:00
|
|
|
suites: x.suites ?? [],
|
|
|
|
excludeByDefault: x.excludeByDefault ?? false,
|
2021-01-12 20:04:16 +01:00
|
|
|
}));
|
|
|
|
}
|
2021-01-13 13:17:38 +01:00
|
|
|
|
|
|
|
const runTestInstrStr = process.env["TWCLI_RUN_TEST_INSTRUCTION"];
|
2021-01-14 17:24:44 +01:00
|
|
|
if (runTestInstrStr && process.argv.includes("__TWCLI_TESTWORKER")) {
|
2021-01-13 13:17:38 +01:00
|
|
|
// Test will call taler-wallet-cli, so we must not propagate this variable.
|
2021-01-14 17:24:44 +01:00
|
|
|
delete process.env["TWCLI_RUN_TEST_INSTRUCTION"];
|
2021-01-13 13:17:38 +01:00
|
|
|
const { testRootDir, testName } = JSON.parse(
|
|
|
|
runTestInstrStr,
|
|
|
|
) as RunTestChildInstruction;
|
|
|
|
console.log(`running test ${testName} in worker process`);
|
|
|
|
|
|
|
|
process.on("disconnect", () => {
|
2021-01-13 13:33:25 +01:00
|
|
|
console.log("got disconnect from parent");
|
2021-01-13 13:17:38 +01:00
|
|
|
process.exit(3);
|
|
|
|
});
|
|
|
|
|
2021-01-13 13:33:25 +01:00
|
|
|
try {
|
|
|
|
require("source-map-support").install();
|
|
|
|
} catch (e) {
|
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
|
2021-01-13 13:17:38 +01:00
|
|
|
const runTest = async () => {
|
|
|
|
let testMain: TestMainFunction | undefined;
|
|
|
|
for (const t of allTests) {
|
|
|
|
if (getTestName(t) === testName) {
|
|
|
|
testMain = t;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!process.send) {
|
|
|
|
console.error("can't communicate with parent");
|
|
|
|
process.exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!testMain) {
|
|
|
|
console.log(`test ${testName} not found`);
|
|
|
|
process.exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
const testDir = path.join(testRootDir, testName);
|
|
|
|
console.log(`running test ${testName}`);
|
|
|
|
const gc = new GlobalTestState({
|
|
|
|
testDir,
|
|
|
|
});
|
|
|
|
const testResult = await runTestWithState(gc, testMain, testName);
|
|
|
|
process.send(testResult);
|
|
|
|
};
|
|
|
|
|
2021-01-13 13:48:28 +01:00
|
|
|
runTest()
|
|
|
|
.then(() => {
|
|
|
|
console.log(`test ${testName} finished in worker`);
|
2021-01-17 01:18:37 +01:00
|
|
|
if (shouldLingerInTest()) {
|
|
|
|
console.log("lingering ...");
|
|
|
|
return;
|
|
|
|
}
|
2021-01-13 13:48:28 +01:00
|
|
|
process.exit(0);
|
|
|
|
})
|
|
|
|
.catch((e) => {
|
|
|
|
console.log(e);
|
|
|
|
process.exit(1);
|
|
|
|
});
|
2021-01-13 13:17:38 +01:00
|
|
|
}
|