2019-12-05 19:38:19 +01:00
|
|
|
/*
|
|
|
|
This file is part of GNU Taler
|
2020-03-12 14:55:38 +01:00
|
|
|
(C) 2019-2020 Taler Systems SA
|
2019-12-05 19:38:19 +01:00
|
|
|
|
|
|
|
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/>
|
|
|
|
*/
|
|
|
|
|
2020-03-12 14:55:38 +01:00
|
|
|
/**
|
|
|
|
* Classes and helpers for error handling specific to wallet operations.
|
|
|
|
*
|
|
|
|
* @author Florian Dold <dold@taler.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2021-03-17 17:56:37 +01:00
|
|
|
import { TalerErrorCode, TalerErrorDetails } from "@gnu-taler/taler-util";
|
2020-03-12 14:55:38 +01:00
|
|
|
|
2019-12-05 19:38:19 +01:00
|
|
|
/**
|
|
|
|
* This exception is there to let the caller know that an error happened,
|
|
|
|
* but the error has already been reported by writing it to the database.
|
|
|
|
*/
|
|
|
|
export class OperationFailedAndReportedError extends Error {
|
2020-11-03 16:46:43 +01:00
|
|
|
static fromCode(
|
|
|
|
ec: TalerErrorCode,
|
|
|
|
message: string,
|
|
|
|
details: Record<string, unknown>,
|
|
|
|
): OperationFailedAndReportedError {
|
2020-12-14 16:45:15 +01:00
|
|
|
return new OperationFailedAndReportedError(
|
|
|
|
makeErrorDetails(ec, message, details),
|
|
|
|
);
|
2020-11-03 16:46:43 +01:00
|
|
|
}
|
|
|
|
|
2020-09-01 14:57:22 +02:00
|
|
|
constructor(public operationError: TalerErrorDetails) {
|
2020-03-12 14:55:38 +01:00
|
|
|
super(operationError.message);
|
2019-12-05 19:38:19 +01:00
|
|
|
|
|
|
|
// Set the prototype explicitly.
|
|
|
|
Object.setPrototypeOf(this, OperationFailedAndReportedError.prototype);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-04-27 23:42:25 +02:00
|
|
|
* This exception is thrown when an error occurred and the caller is
|
2019-12-05 19:38:19 +01:00
|
|
|
* responsible for recording the failure in the database.
|
|
|
|
*/
|
|
|
|
export class OperationFailedError extends Error {
|
2020-07-22 10:52:03 +02:00
|
|
|
static fromCode(
|
|
|
|
ec: TalerErrorCode,
|
|
|
|
message: string,
|
|
|
|
details: Record<string, unknown>,
|
|
|
|
): OperationFailedError {
|
|
|
|
return new OperationFailedError(makeErrorDetails(ec, message, details));
|
|
|
|
}
|
|
|
|
|
2020-09-01 14:57:22 +02:00
|
|
|
constructor(public operationError: TalerErrorDetails) {
|
2020-03-12 14:55:38 +01:00
|
|
|
super(operationError.message);
|
2019-12-05 19:38:19 +01:00
|
|
|
|
|
|
|
// Set the prototype explicitly.
|
|
|
|
Object.setPrototypeOf(this, OperationFailedError.prototype);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-22 10:52:03 +02:00
|
|
|
export function makeErrorDetails(
|
|
|
|
ec: TalerErrorCode,
|
|
|
|
message: string,
|
|
|
|
details: Record<string, unknown>,
|
2020-09-01 14:57:22 +02:00
|
|
|
): TalerErrorDetails {
|
2020-07-22 10:52:03 +02:00
|
|
|
return {
|
2020-09-01 14:30:46 +02:00
|
|
|
code: ec,
|
|
|
|
hint: `Error: ${TalerErrorCode[ec]}`,
|
2020-07-22 10:52:03 +02:00
|
|
|
details: details,
|
|
|
|
message,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-12-05 19:38:19 +01:00
|
|
|
/**
|
|
|
|
* Run an operation and call the onOpError callback
|
|
|
|
* when there was an exception or operation error that must be reported.
|
|
|
|
* The cause will be re-thrown to the caller.
|
|
|
|
*/
|
|
|
|
export async function guardOperationException<T>(
|
|
|
|
op: () => Promise<T>,
|
2020-09-01 14:57:22 +02:00
|
|
|
onOpError: (e: TalerErrorDetails) => Promise<void>,
|
2019-12-05 19:38:19 +01:00
|
|
|
): Promise<T> {
|
|
|
|
try {
|
2019-12-05 22:17:01 +01:00
|
|
|
return await op();
|
2019-12-05 19:38:19 +01:00
|
|
|
} catch (e) {
|
|
|
|
if (e instanceof OperationFailedAndReportedError) {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
if (e instanceof OperationFailedError) {
|
2020-03-12 14:55:38 +01:00
|
|
|
await onOpError(e.operationError);
|
|
|
|
throw new OperationFailedAndReportedError(e.operationError);
|
2019-12-05 19:38:19 +01:00
|
|
|
}
|
|
|
|
if (e instanceof Error) {
|
2020-07-22 10:52:03 +02:00
|
|
|
const opErr = makeErrorDetails(
|
|
|
|
TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
|
|
|
|
`unexpected exception (message: ${e.message})`,
|
2020-09-01 12:22:13 +02:00
|
|
|
{
|
|
|
|
stack: e.stack,
|
|
|
|
},
|
2020-07-22 10:52:03 +02:00
|
|
|
);
|
2020-03-12 14:55:38 +01:00
|
|
|
await onOpError(opErr);
|
|
|
|
throw new OperationFailedAndReportedError(opErr);
|
2019-12-05 19:38:19 +01:00
|
|
|
}
|
2020-07-22 10:52:03 +02:00
|
|
|
// Something was thrown that is not even an exception!
|
|
|
|
// Try to stringify it.
|
|
|
|
let excString: string;
|
|
|
|
try {
|
|
|
|
excString = e.toString();
|
|
|
|
} catch (e) {
|
|
|
|
// Something went horribly wrong.
|
|
|
|
excString = "can't stringify exception";
|
|
|
|
}
|
|
|
|
const opErr = makeErrorDetails(
|
|
|
|
TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
|
|
|
|
`unexpected exception (not an exception, ${excString})`,
|
|
|
|
{},
|
|
|
|
);
|
2020-03-12 14:55:38 +01:00
|
|
|
await onOpError(opErr);
|
|
|
|
throw new OperationFailedAndReportedError(opErr);
|
2019-12-05 19:38:19 +01:00
|
|
|
}
|
2020-03-12 14:55:38 +01:00
|
|
|
}
|