UNPKG

@theia/core

Version:

Theia is a cloud & desktop IDE framework implemented in TypeScript.

77 lines (74 loc) 3.08 kB
// ***************************************************************************** // Copyright (C) 2018 TypeFox and others. // // This program and the accompanying materials are made available under the // terms of the Eclipse Public License v. 2.0 which is available at // http://www.eclipse.org/legal/epl-2.0. // // This Source Code may also be made available under the following Secondary // Licenses when the conditions for such availability set forth in the Eclipse // Public License v. 2.0 are satisfied: GNU General Public License, version 2 // with the GNU Classpath Exception which is available at // https://www.gnu.org/software/classpath/license.html. // // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 // ***************************************************************************** /* eslint-disable @typescript-eslint/no-explicit-any */ export interface ApplicationError<C extends number, D> extends Error { readonly code: C readonly data: D toJson(): ApplicationError.Literal<D> } export namespace ApplicationError { export interface Literal<D> { message: string data: D stack?: string } export interface Constructor<C extends number, D> { (...args: any[]): ApplicationError<C, D>; code: C; is(arg: object | undefined): arg is ApplicationError<C, D> } const codes = new Set<number>(); export function declare<C extends number, D>(code: C, factory: (...args: any[]) => Literal<D>): Constructor<C, D> { if (codes.has(code)) { throw new Error(`An application error for '${code}' code is already declared`); } codes.add(code); const constructorOpt = Object.assign((...args: any[]) => new Impl(code, factory(...args), constructorOpt), { code, is(arg: object | undefined): arg is ApplicationError<C, D> { return arg instanceof Impl && arg.code === code; } }); return constructorOpt; } export function is<C extends number, D>(arg: object | undefined): arg is ApplicationError<C, D> { return arg instanceof Impl; } export function fromJson<C extends number, D>(code: C, raw: Literal<D>): ApplicationError<C, D> { return new Impl(code, raw); } class Impl<C extends number, D> extends Error implements ApplicationError<C, D> { readonly data: D; constructor( readonly code: C, raw: ApplicationError.Literal<D>, constructorOpt?: Function ) { super(raw.message); this.data = raw.data; Object.setPrototypeOf(this, Impl.prototype); if (raw.stack) { this.stack = raw.stack; } else if (Error.captureStackTrace && constructorOpt) { Error.captureStackTrace(this, constructorOpt); } } toJson(): ApplicationError.Literal<D> { const { message, data, stack } = this; return { message, data, stack }; } } }