@fgv/ts-utils
Advanced tools
Comparing version 0.1.8 to 0.1.9
{ | ||
"name": "@fgv/ts-utils", | ||
"version": "0.1.8", | ||
"version": "0.1.9", | ||
"description": "Assorted Typescript Utilities", | ||
@@ -5,0 +5,0 @@ "main": "index.js", |
@@ -71,2 +71,3 @@ export declare type Result<T> = Success<T> | Failure<T>; | ||
export declare function failWithDetail<T, TD>(message: string, detail: TD): DetailedFailure<T, TD>; | ||
export declare function propagateWithDetail<T, TD>(result: Result<T>, detail: TD): DetailedResult<T, TD>; | ||
/** | ||
@@ -73,0 +74,0 @@ * Wraps a function which returns a value of type <T> or throws |
@@ -24,3 +24,3 @@ "use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.populateObject = exports.allSucceed = exports.mapFailures = exports.mapSuccess = exports.mapResults = exports.captureResult = exports.failWithDetail = exports.succeedWithDetail = exports.DetailedFailure = exports.DetailedSuccess = exports.fail = exports.succeed = exports.Failure = exports.Success = void 0; | ||
exports.populateObject = exports.allSucceed = exports.mapFailures = exports.mapSuccess = exports.mapResults = exports.captureResult = exports.propagateWithDetail = exports.failWithDetail = exports.succeedWithDetail = exports.DetailedFailure = exports.DetailedSuccess = exports.fail = exports.succeed = exports.Failure = exports.Success = void 0; | ||
class Success { | ||
@@ -149,2 +149,6 @@ constructor(value) { | ||
exports.failWithDetail = failWithDetail; | ||
function propagateWithDetail(result, detail) { | ||
return result.isSuccess() ? succeedWithDetail(result.value) : failWithDetail(result.message, detail); | ||
} | ||
exports.propagateWithDetail = propagateWithDetail; | ||
/** | ||
@@ -285,2 +289,2 @@ * Wraps a function which returns a value of type <T> or throws | ||
exports.populateObject = populateObject; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"result.js","sourceRoot":"","sources":["../src/result.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nexport type Result<T> = Success<T> | Failure<T>;\nexport type SuccessContinuation<T, TN> = (value: T) => Result<TN>;\nexport type FailureContinuation<T> = (message: string) => Result<T>;\n\nexport interface IResultLogger {\n    error(message: string): void;\n}\n\nexport interface IResult<T> {\n    isSuccess(): this is Success<T>;\n    isFailure(): this is Failure<T>;\n    getValueOrThrow(logger?: IResultLogger): T;\n    getValueOrDefault(dflt?: T): T|undefined;\n    onSuccess<TN>(cb: SuccessContinuation<T, TN>): Result<TN>;\n    onFailure(cb: FailureContinuation<T>): Result<T>;\n}\n\nexport class Success<T> implements IResult<T> {\n    private readonly _value: T;\n\n    constructor(value: T) {\n        this._value = value;\n    }\n\n    public isSuccess(): this is Success<T> {\n        return true;\n    }\n\n    public isFailure(): this is Failure<T> {\n        return false;\n    }\n\n    public get value(): T {\n        return this._value;\n    }\n\n    public getValueOrThrow(_logger?: IResultLogger): T {\n        return this._value;\n    }\n\n    public getValueOrDefault(dflt?: T): T|undefined {\n        return this._value ?? dflt;\n    }\n\n    public onSuccess<TN>(cb: SuccessContinuation<T, TN>): Result<TN> {\n        return cb(this.value);\n    }\n\n    public onFailure<TN>(_: FailureContinuation<T>): Result<T> {\n        return this;\n    }\n}\n\nexport class Failure<T> implements IResult<T> {\n    private readonly _message: string;\n\n    constructor(message: string) {\n        this._message = message;\n    }\n\n    public isSuccess(): this is Success<T> {\n        return false;\n    }\n\n    public isFailure(): this is Failure<T> {\n        return true;\n    }\n\n    public get message(): string {\n        return this._message;\n    }\n\n    public getValueOrThrow(logger?: IResultLogger): never {\n        if (logger !== undefined) {\n            logger.error(this._message);\n        }\n        throw new Error(this._message);\n    }\n\n    public getValueOrDefault(dflt?: T): T|undefined {\n        return dflt;\n    }\n\n    public onSuccess<TN>(_: SuccessContinuation<T, TN>): Result<TN> {\n        return new Failure(this.message);\n    }\n\n    public onFailure(cb: FailureContinuation<T>): Result<T> {\n        return cb(this.message);\n    }\n\n    public toString(): string {\n        return this.message;\n    }\n}\n\n/**\n * Helper function for successful return\n * @param val The value to be returned\n */\nexport function succeed<T>(val: T): Success<T> {\n    return new Success<T>(val);\n}\n\n/**\n * Helper function for error return\n * @param message Error message to be returned\n */\nexport function fail<T>(message: string): Failure<T> {\n    return new Failure<T>(message);\n}\n\nexport type DetailedSuccessContinuation<T, TD, TN> = (value: T) => DetailedResult<TN, TD>;\nexport type DetailedFailureContinuation<T, TD> = (message: string, detail: TD) => DetailedResult<T, TD>;\n\nexport class DetailedSuccess<T, TD> extends Success<T> {\n    public constructor(value: T) {\n        super(value);\n    }\n\n    public isSuccess(): this is DetailedSuccess<T, TD> {\n        return true;\n    }\n\n    public onSuccess<TN>(cb: DetailedSuccessContinuation<T, TD, TN>): DetailedResult<TN, TD> {\n        return cb(this.value);\n    }\n\n    public onFailure(_cb: DetailedFailureContinuation<T, TD>): DetailedResult<T, TD> {\n        return this;\n    }\n}\n\n/**\n * A DetailedFailure reports optional failure details in addition\n * to the standard failure message.\n */\nexport class DetailedFailure<T, TD> extends Failure<T> {\n    protected _detail: TD;\n\n    public constructor(message: string, detail: TD) {\n        super(message);\n        this._detail = detail;\n    }\n\n    public get detail(): TD {\n        return this._detail;\n    }\n\n    public isFailure(): this is DetailedFailure<T, TD> {\n        return true;\n    }\n\n    public onSuccess<TN>(_cb: DetailedSuccessContinuation<T, TD, TN>): DetailedResult<TN, TD> {\n        return new DetailedFailure<TN, TD>(this.message, this._detail);\n    }\n\n    public onFailure(cb: DetailedFailureContinuation<T, TD>): DetailedResult<T, TD> {\n        return cb(this.message, this._detail);\n    }\n}\n\nexport type DetailedResult<T, TD> = DetailedSuccess<T, TD>|DetailedFailure<T, TD>;\n\nexport function succeedWithDetail<T, TD>(value: T): DetailedSuccess<T, TD> {\n    return new DetailedSuccess<T, TD>(value);\n}\n\nexport function failWithDetail<T, TD>(message: string, detail: TD): DetailedFailure<T, TD> {\n    return new DetailedFailure<T, TD>(message, detail);\n}\n\n/**\n * Wraps a function which returns a value of type <T> or throws\n * to produce Success<T> or Failure<T>\n * @param func The method to be captured\n */\nexport function captureResult<T>(func: () => T): Result<T> {\n    try {\n        return succeed(func());\n    }\n    catch (err) {\n        return fail((err as Error).message);\n    }\n}\n\n/**\n * Maps an array of Result<T> to an array of <T>, if all results are\n * successful.  If any results fail, returns failure with a concatenated\n * summary of all failure messages.\n * @param resultsIn The results to be mapped.\n */\nexport function mapResults<T>(resultsIn: Iterable<Result<T>>): Result<T[]> {\n    const errors: string[] = [];\n    const elements: T[] = [];\n\n    for (const result of resultsIn) {\n        if (result.isSuccess()) {\n            elements.push(result.value);\n        }\n        else {\n            errors.push(result.message);\n        }\n    }\n\n    if (errors.length > 0) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(elements);\n}\n\n/**\n * Maps an array of Result<T> to an array of <T>, omitting any error\n * results.  If no results were successful, returns failure with a\n * concatenated summary of all failure messages.\n */\nexport function mapSuccess<T>(resultsIn: Iterable<Result<T>>): Result<T[]> {\n    const errors: string[] = [];\n    const elements: T[] = [];\n\n    for (const result of resultsIn) {\n        if (result.isSuccess()) {\n            elements.push(result.value);\n        }\n        else {\n            errors.push(result.message);\n        }\n    }\n\n    if ((elements.length === 0) && (errors.length > 0)) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(elements);\n}\n\n/**\n * Maps an array of Result<T> to an array of strings consisting of all\n * error messages returned by results in the source array. Ignores\n * success results and returns an empty array if there were no errors.\n * @param resultsIn results to be reported\n */\nexport function mapFailures<T>(resultsIn: Iterable<Result<T>>): string[] {\n    const errors: string[] = [];\n    for (const result of resultsIn) {\n        if (result.isFailure()) {\n            errors.push(result.message);\n        }\n    }\n    return errors;\n}\n\n/**\n * Returns success with true if all results are successful.  If any are unsuccessful,\n * returns failure with a concatenated summary of all failure messages.\n * @param results The results to be tested.\n */\nexport function allSucceed<T>(results: Iterable<Result<unknown>>, successValue: T): Result<T> {\n    const errors: string[] = [];\n\n    // istanbul ignore else\n    if (results !== undefined) {\n        for (const result of results) {\n            if (result.isFailure()) {\n                errors.push(result.message);\n            }\n        }\n    }\n\n    if (errors.length > 0) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(successValue);\n}\n\nexport type FieldInitializers<T> = { [ key in keyof T ]: (state: Partial<T>) => Result<T[key]> };\n\n/**\n * Populates an an object based on a prototype full of field initializers that return Result<T[key]>.\n * Returns success with the populated object if all initializers succeed, or failure with a\n * concatenated list of all failure messages.\n * @param initializers An object with the shape of the target but with initializer functions for\n * each property.\n */\nexport function populateObject<T>(initializers: FieldInitializers<T>, order?: (keyof T)[]): Result<T> {\n    const state = {} as { [key in keyof T]: T[key] };\n    const errors: string[] = [];\n    const keys: (keyof T)[] = Array.from(order ?? []);\n    const foundKeys = new Set<keyof T>(order);\n\n    // start with the supplied order then append anything else we find\n    for (const key in initializers) {\n        if (!foundKeys.has(key)) {\n            keys.push(key);\n            foundKeys.add(key);\n        }\n    }\n\n    for (const key of keys) {\n        if (initializers[key]) {\n            const result = initializers[key](state);\n            if (result.isSuccess()) {\n                state[key] = result.value;\n            }\n            else {\n                errors.push(result.message);\n            }\n        }\n        else {\n            errors.push(`populateObject: Key ${key} is present but has no initializer`);\n        }\n    }\n\n    if (errors.length > 0) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(state as T);\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"result.js","sourceRoot":"","sources":["../src/result.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nexport type Result<T> = Success<T> | Failure<T>;\nexport type SuccessContinuation<T, TN> = (value: T) => Result<TN>;\nexport type FailureContinuation<T> = (message: string) => Result<T>;\n\nexport interface IResultLogger {\n    error(message: string): void;\n}\n\nexport interface IResult<T> {\n    isSuccess(): this is Success<T>;\n    isFailure(): this is Failure<T>;\n    getValueOrThrow(logger?: IResultLogger): T;\n    getValueOrDefault(dflt?: T): T|undefined;\n    onSuccess<TN>(cb: SuccessContinuation<T, TN>): Result<TN>;\n    onFailure(cb: FailureContinuation<T>): Result<T>;\n}\n\nexport class Success<T> implements IResult<T> {\n    private readonly _value: T;\n\n    constructor(value: T) {\n        this._value = value;\n    }\n\n    public isSuccess(): this is Success<T> {\n        return true;\n    }\n\n    public isFailure(): this is Failure<T> {\n        return false;\n    }\n\n    public get value(): T {\n        return this._value;\n    }\n\n    public getValueOrThrow(_logger?: IResultLogger): T {\n        return this._value;\n    }\n\n    public getValueOrDefault(dflt?: T): T|undefined {\n        return this._value ?? dflt;\n    }\n\n    public onSuccess<TN>(cb: SuccessContinuation<T, TN>): Result<TN> {\n        return cb(this.value);\n    }\n\n    public onFailure<TN>(_: FailureContinuation<T>): Result<T> {\n        return this;\n    }\n}\n\nexport class Failure<T> implements IResult<T> {\n    private readonly _message: string;\n\n    constructor(message: string) {\n        this._message = message;\n    }\n\n    public isSuccess(): this is Success<T> {\n        return false;\n    }\n\n    public isFailure(): this is Failure<T> {\n        return true;\n    }\n\n    public get message(): string {\n        return this._message;\n    }\n\n    public getValueOrThrow(logger?: IResultLogger): never {\n        if (logger !== undefined) {\n            logger.error(this._message);\n        }\n        throw new Error(this._message);\n    }\n\n    public getValueOrDefault(dflt?: T): T|undefined {\n        return dflt;\n    }\n\n    public onSuccess<TN>(_: SuccessContinuation<T, TN>): Result<TN> {\n        return new Failure(this.message);\n    }\n\n    public onFailure(cb: FailureContinuation<T>): Result<T> {\n        return cb(this.message);\n    }\n\n    public toString(): string {\n        return this.message;\n    }\n}\n\n/**\n * Helper function for successful return\n * @param val The value to be returned\n */\nexport function succeed<T>(val: T): Success<T> {\n    return new Success<T>(val);\n}\n\n/**\n * Helper function for error return\n * @param message Error message to be returned\n */\nexport function fail<T>(message: string): Failure<T> {\n    return new Failure<T>(message);\n}\n\nexport type DetailedSuccessContinuation<T, TD, TN> = (value: T) => DetailedResult<TN, TD>;\nexport type DetailedFailureContinuation<T, TD> = (message: string, detail: TD) => DetailedResult<T, TD>;\n\nexport class DetailedSuccess<T, TD> extends Success<T> {\n    public constructor(value: T) {\n        super(value);\n    }\n\n    public isSuccess(): this is DetailedSuccess<T, TD> {\n        return true;\n    }\n\n    public onSuccess<TN>(cb: DetailedSuccessContinuation<T, TD, TN>): DetailedResult<TN, TD> {\n        return cb(this.value);\n    }\n\n    public onFailure(_cb: DetailedFailureContinuation<T, TD>): DetailedResult<T, TD> {\n        return this;\n    }\n}\n\n/**\n * A DetailedFailure reports optional failure details in addition\n * to the standard failure message.\n */\nexport class DetailedFailure<T, TD> extends Failure<T> {\n    protected _detail: TD;\n\n    public constructor(message: string, detail: TD) {\n        super(message);\n        this._detail = detail;\n    }\n\n    public get detail(): TD {\n        return this._detail;\n    }\n\n    public isFailure(): this is DetailedFailure<T, TD> {\n        return true;\n    }\n\n    public onSuccess<TN>(_cb: DetailedSuccessContinuation<T, TD, TN>): DetailedResult<TN, TD> {\n        return new DetailedFailure<TN, TD>(this.message, this._detail);\n    }\n\n    public onFailure(cb: DetailedFailureContinuation<T, TD>): DetailedResult<T, TD> {\n        return cb(this.message, this._detail);\n    }\n}\n\nexport type DetailedResult<T, TD> = DetailedSuccess<T, TD>|DetailedFailure<T, TD>;\n\nexport function succeedWithDetail<T, TD>(value: T): DetailedSuccess<T, TD> {\n    return new DetailedSuccess<T, TD>(value);\n}\n\nexport function failWithDetail<T, TD>(message: string, detail: TD): DetailedFailure<T, TD> {\n    return new DetailedFailure<T, TD>(message, detail);\n}\n\nexport function propagateWithDetail<T, TD>(result: Result<T>, detail: TD): DetailedResult<T, TD> {\n    return result.isSuccess() ? succeedWithDetail(result.value) : failWithDetail(result.message, detail);\n}\n\n/**\n * Wraps a function which returns a value of type <T> or throws\n * to produce Success<T> or Failure<T>\n * @param func The method to be captured\n */\nexport function captureResult<T>(func: () => T): Result<T> {\n    try {\n        return succeed(func());\n    }\n    catch (err) {\n        return fail((err as Error).message);\n    }\n}\n\n/**\n * Maps an array of Result<T> to an array of <T>, if all results are\n * successful.  If any results fail, returns failure with a concatenated\n * summary of all failure messages.\n * @param resultsIn The results to be mapped.\n */\nexport function mapResults<T>(resultsIn: Iterable<Result<T>>): Result<T[]> {\n    const errors: string[] = [];\n    const elements: T[] = [];\n\n    for (const result of resultsIn) {\n        if (result.isSuccess()) {\n            elements.push(result.value);\n        }\n        else {\n            errors.push(result.message);\n        }\n    }\n\n    if (errors.length > 0) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(elements);\n}\n\n/**\n * Maps an array of Result<T> to an array of <T>, omitting any error\n * results.  If no results were successful, returns failure with a\n * concatenated summary of all failure messages.\n */\nexport function mapSuccess<T>(resultsIn: Iterable<Result<T>>): Result<T[]> {\n    const errors: string[] = [];\n    const elements: T[] = [];\n\n    for (const result of resultsIn) {\n        if (result.isSuccess()) {\n            elements.push(result.value);\n        }\n        else {\n            errors.push(result.message);\n        }\n    }\n\n    if ((elements.length === 0) && (errors.length > 0)) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(elements);\n}\n\n/**\n * Maps an array of Result<T> to an array of strings consisting of all\n * error messages returned by results in the source array. Ignores\n * success results and returns an empty array if there were no errors.\n * @param resultsIn results to be reported\n */\nexport function mapFailures<T>(resultsIn: Iterable<Result<T>>): string[] {\n    const errors: string[] = [];\n    for (const result of resultsIn) {\n        if (result.isFailure()) {\n            errors.push(result.message);\n        }\n    }\n    return errors;\n}\n\n/**\n * Returns success with true if all results are successful.  If any are unsuccessful,\n * returns failure with a concatenated summary of all failure messages.\n * @param results The results to be tested.\n */\nexport function allSucceed<T>(results: Iterable<Result<unknown>>, successValue: T): Result<T> {\n    const errors: string[] = [];\n\n    // istanbul ignore else\n    if (results !== undefined) {\n        for (const result of results) {\n            if (result.isFailure()) {\n                errors.push(result.message);\n            }\n        }\n    }\n\n    if (errors.length > 0) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(successValue);\n}\n\nexport type FieldInitializers<T> = { [ key in keyof T ]: (state: Partial<T>) => Result<T[key]> };\n\n/**\n * Populates an an object based on a prototype full of field initializers that return Result<T[key]>.\n * Returns success with the populated object if all initializers succeed, or failure with a\n * concatenated list of all failure messages.\n * @param initializers An object with the shape of the target but with initializer functions for\n * each property.\n */\nexport function populateObject<T>(initializers: FieldInitializers<T>, order?: (keyof T)[]): Result<T> {\n    const state = {} as { [key in keyof T]: T[key] };\n    const errors: string[] = [];\n    const keys: (keyof T)[] = Array.from(order ?? []);\n    const foundKeys = new Set<keyof T>(order);\n\n    // start with the supplied order then append anything else we find\n    for (const key in initializers) {\n        if (!foundKeys.has(key)) {\n            keys.push(key);\n            foundKeys.add(key);\n        }\n    }\n\n    for (const key of keys) {\n        if (initializers[key]) {\n            const result = initializers[key](state);\n            if (result.isSuccess()) {\n                state[key] = result.value;\n            }\n            else {\n                errors.push(result.message);\n            }\n        }\n        else {\n            errors.push(`populateObject: Key ${key} is present but has no initializer`);\n        }\n    }\n\n    if (errors.length > 0) {\n        return fail(errors.join('\\n'));\n    }\n    return succeed(state as T);\n}\n"]} |
188150
1749