2023-01-13 12:42:52 +01:00
|
|
|
/*
|
|
|
|
Copyright 2023 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { ViewModel } from "./ViewModel";
|
|
|
|
import type { Options as BaseOptions } from "./ViewModel";
|
|
|
|
import type { Session } from "../matrix/Session";
|
|
|
|
import { ErrorViewModel } from "./ErrorViewModel";
|
|
|
|
import type { LogCallback, LabelOrValues } from "../logging/types";
|
|
|
|
|
2023-01-20 16:17:22 +01:00
|
|
|
export type Options<N extends object> = BaseOptions<N> & {
|
2023-01-13 12:42:52 +01:00
|
|
|
session: Session
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Base class for view models that need to report errors to the UI. */
|
2023-01-20 16:17:22 +01:00
|
|
|
export class ErrorReportViewModel<N extends object, O extends Options<N> = Options<N>> extends ViewModel<N, O> {
|
|
|
|
private _errorViewModel?: ErrorViewModel<N>;
|
2023-01-13 12:42:52 +01:00
|
|
|
|
2023-01-20 16:17:22 +01:00
|
|
|
get errorViewModel(): ErrorViewModel<N> | undefined {
|
2023-01-13 12:42:52 +01:00
|
|
|
return this._errorViewModel;
|
|
|
|
}
|
|
|
|
|
2023-01-17 17:18:07 +01:00
|
|
|
/** Typically you'd want to use `logAndCatch` when implementing a view model method.
|
|
|
|
* Use `reportError` when showing errors on your model that were set by
|
|
|
|
* background processes using `ErrorBoundary` or you have some other
|
|
|
|
* special low-level need to write your try/catch yourself. */
|
2023-01-13 12:42:52 +01:00
|
|
|
protected reportError(error: Error) {
|
|
|
|
if (this._errorViewModel?.error === error) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.disposeTracked(this._errorViewModel);
|
|
|
|
this._errorViewModel = this.track(new ErrorViewModel(this.childOptions({
|
|
|
|
error,
|
|
|
|
onClose: () => {
|
|
|
|
this._errorViewModel = this.disposeTracked(this._errorViewModel);
|
|
|
|
this.emitChange("errorViewModel");
|
|
|
|
}
|
|
|
|
})));
|
|
|
|
this.emitChange("errorViewModel");
|
|
|
|
}
|
|
|
|
|
2023-01-17 17:18:07 +01:00
|
|
|
/** Combines logging and error reporting in one method.
|
|
|
|
* Wrap the implementation of public view model methods
|
|
|
|
* with this to ensure errors are logged and reported.*/
|
2023-01-13 12:42:52 +01:00
|
|
|
protected logAndCatch<T>(labelOrValues: LabelOrValues, callback: LogCallback<T>, errorValue: T = undefined as unknown as T): T {
|
|
|
|
try {
|
2023-01-13 14:20:33 +01:00
|
|
|
let result = this.logger.run(labelOrValues, callback);
|
2023-01-13 12:42:52 +01:00
|
|
|
if (result instanceof Promise) {
|
2023-01-13 14:20:33 +01:00
|
|
|
result = result.catch(err => {
|
|
|
|
this.reportError(err);
|
|
|
|
return errorValue;
|
|
|
|
}) as unknown as T;
|
2023-01-13 12:42:52 +01:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
} catch (err) {
|
|
|
|
this.reportError(err);
|
|
|
|
return errorValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|