@fgv/ts-utils
Advanced tools
Comparing version 0.3.15 to 0.3.16
@@ -54,2 +54,3 @@ import { Result } from './result'; | ||
} | ||
declare type InnerConvertedToType<TCONV> = TCONV extends Converter<infer TTO> ? (TTO extends Array<infer TTOELEM> ? InnerConvertedToType<TTOELEM>[] : TTO) : (TCONV extends Array<infer TELEM> ? InnerConvertedToType<TELEM>[] : TCONV); | ||
/** | ||
@@ -60,3 +61,3 @@ * Infers the type that will be returned by an intstantiated converter. Works | ||
*/ | ||
export declare type ConvertedToType<TCONV> = TCONV extends Converter<infer TTO> ? (TTO extends Array<infer TTOELEM> ? ConvertedToType<TTOELEM>[] : TTO) : (TCONV extends Array<infer TELEM> ? ConvertedToType<TELEM>[] : TCONV); | ||
export declare type ConvertedToType<TCONV> = TCONV extends Converter<infer TTO> ? InnerConvertedToType<TTO> : never; | ||
/** | ||
@@ -63,0 +64,0 @@ * Simple templated converter wrapper to simplify typed conversion from unknown. |
@@ -128,2 +128,2 @@ "use strict"; | ||
exports.BaseConverter = BaseConverter; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"converter.js","sourceRoot":"","sources":["../src/converter.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 */\nimport { Result, fail, succeed } from './result';\n\ntype OnError = 'failOnError' | 'ignoreErrors';\n\nexport interface Converter<T, TC=undefined> {\n    /**\n     * Converts from unknown to <T>\n     * @param from The unknown to be converted\n     * @param context An optional context applied to the conversion\n     * @returns An @see Result with a value or an error message\n     */\n    convert(from: unknown, context?: TC): Result<T>;\n\n    /**\n     * Converts from unknown to <T> or undefined, as appropriate.\n     * If 'onError' is 'failOnError', the converter succeeds for\n     * 'undefined' or any convertible value, but reports an error\n     * if it encounters a value that cannot be converted.  If 'onError'\n     * is 'ignoreErrors' (default) then values that cannot be converted\n     * result in a successful return of 'undefined'.\n     * @param from The unknown to be converted\n     * @param context Optional context for use by the converter\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     */\n    convertOptional(from: unknown, context?: TC, onError?: OnError): Result<T|undefined>;\n\n    /**\n     * Creates a converter for an optional value. If 'onError'\n     * is 'failOnError', the converter accepts 'undefined' or a\n     * convertible value, but reports an error if it encounters\n     * a value that cannot be converted.  If 'onError' is 'ignoreErrors'\n     * (default) then values that cannot be converted result in a\n     * successful return of 'undefined'.\n     *\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     * */\n    optional(onError?: OnError): Converter<T|undefined, TC>;\n\n    /**\n     * Applies a (possibly) mapping conversion to the converted value.\n     * @param mapper A function which maps from the converted type to some other type.\n     */\n    map<T2>(mapper: (from: T) => Result<T2>): Converter<T2, TC>;\n\n    /**\n     * Applies an additional converter to the converted value.\n     * @param mapConverter The converter to be applied to the converted value\n     */\n    mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2, TC>;\n\n    /**\n     * Creates a converter with an optional constraint.  If the base converter\n     * succeeds, calls a supplied constraint evaluation function with the\n     * value and fails the conversion if the function returns either false\n     * or Failure<T>.\n     *\n     * @param constraint Constraint evaluation function\n     */\n    withConstraint(constraint: (val: T) => boolean|Result<T>): Converter<T, TC>;\n}\n\n/**\n * Infers the type that will be returned by an intstantiated converter.  Works\n * for complex as well as simple types.\n * @example ConvertedToType<typeof Converters.mapOf(Converters.stringArray)> is Map<string, string[]>\n */\nexport type ConvertedToType<TCONV> =\n    TCONV extends Converter<infer TTO>\n        ? (TTO extends Array<infer TTOELEM> ? ConvertedToType<TTOELEM>[] : TTO)\n        : (TCONV extends Array<infer TELEM> ? ConvertedToType<TELEM>[] : TCONV);\n\n/**\n * Simple templated converter wrapper to simplify typed conversion from unknown.\n */\nexport class BaseConverter<T, TC=undefined> implements Converter<T, TC> {\n    protected readonly _defaultContext?: TC;\n    private readonly _converter: (from: unknown, self: Converter<T, TC>, context?: TC) => Result<T>;\n\n    public constructor(converter: (from: unknown, self: Converter<T, TC>, context?: TC) => Result<T>, defaultContext?: TC) {\n        this._converter = converter;\n        this._defaultContext = defaultContext;\n    }\n\n    /**\n     * Converts from unknown to <T>\n     * @param from The unknown to be converted\n     * @returns An @see Result with a value or an error message\n     */\n    public convert(from: unknown, context?: TC): Result<T> {\n        return this._converter(from, this, context ?? this._defaultContext);\n    }\n\n    /**\n     * Converts from unknown to <T> or undefined, as appropriate.\n     * If 'onError' is 'failOnError', the converter succeeds for\n     * 'undefined' or any convertible value, but reports an error\n     * if it encounters a value that cannot be converted.  If 'onError'\n     * is 'ignoreErrors' (default) then values that cannot be converted\n     * result in a successful return of 'undefined'.\n     * @param from The unknown to be converted\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     */\n    public convertOptional(from: unknown, context?: TC, onError?: OnError): Result<T|undefined> {\n        const result = this._converter(from, this, this._context(context));\n        if (result.isFailure()) {\n            onError = onError ?? 'ignoreErrors';\n            return ((from === undefined) || onError === 'ignoreErrors') ? succeed(undefined) : result;\n        }\n        return result;\n    }\n\n    /**\n     * Creates a converter for an optional value. If 'onError'\n     * is 'failOnError', the converter accepts 'undefined' or a\n     * convertible value, but reports an error if it encounters\n     * a value that cannot be converted.  If 'onError' is 'ignoreErrors'\n     * (default) then values that cannot be converted result in a\n     * successful return of 'undefined'.\n     *\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     * */\n    public optional(onError?: OnError): Converter<T|undefined, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T|undefined, TC>, context?: TC) => {\n            onError = onError ?? 'ignoreErrors';\n            return this.convertOptional(from, this._context(context), onError);\n        });\n    }\n\n    /**\n     * Applies a (possibly) mapping conversion to the converted value.\n     * @param mapper A function which maps from the converted type to some other type.\n     */\n    public map<T2>(mapper: (from: T) => Result<T2>): Converter<T2, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T2, TC>, context?: TC) => {\n            const innerResult = this._converter(from, this, this._context(context));\n            if (innerResult.isSuccess()) {\n                return mapper(innerResult.value);\n            }\n            return fail(innerResult.message);\n        });\n    }\n\n    /**\n     * Applies an additional converter to the converted value.\n     * @param mapConverter The converter to be applied to the converted value\n     */\n    public mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T2, TC>, context?: TC) => {\n            const innerResult = this._converter(from, this, this._context(context));\n            if (innerResult.isSuccess()) {\n                return mapConverter.convert(innerResult.value);\n            }\n            return fail(innerResult.message);\n        });\n    }\n\n    /**\n     * Creates a converter with an optional constraint.  If the base converter\n     * succeeds, calls a supplied constraint evaluation function with the\n     * value and fails the conversion if the function returns either false\n     * or Failure<T>.\n     *\n     * @param constraint Constraint evaluation function\n     */\n    public withConstraint(constraint: (val: T) => boolean|Result<T>): Converter<T, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T, TC>, context?: TC) => {\n            const result = this._converter(from, this, this._context(context));\n            if (result.isSuccess()) {\n                const constraintResult = constraint(result.value);\n                if (typeof constraintResult === 'boolean') {\n                    return constraintResult ? result : fail(`Value ${JSON.stringify(result.value)} does not meet constraint.`);\n                }\n                return constraintResult;\n            }\n            return result;\n        });\n    }\n\n    protected _context(supplied?: TC): TC|undefined {\n        return supplied ?? this._defaultContext;\n    }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"converter.js","sourceRoot":"","sources":["../src/converter.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 */\nimport { Result, fail, succeed } from './result';\n\ntype OnError = 'failOnError' | 'ignoreErrors';\n\nexport interface Converter<T, TC=undefined> {\n    /**\n     * Converts from unknown to <T>\n     * @param from The unknown to be converted\n     * @param context An optional context applied to the conversion\n     * @returns An @see Result with a value or an error message\n     */\n    convert(from: unknown, context?: TC): Result<T>;\n\n    /**\n     * Converts from unknown to <T> or undefined, as appropriate.\n     * If 'onError' is 'failOnError', the converter succeeds for\n     * 'undefined' or any convertible value, but reports an error\n     * if it encounters a value that cannot be converted.  If 'onError'\n     * is 'ignoreErrors' (default) then values that cannot be converted\n     * result in a successful return of 'undefined'.\n     * @param from The unknown to be converted\n     * @param context Optional context for use by the converter\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     */\n    convertOptional(from: unknown, context?: TC, onError?: OnError): Result<T|undefined>;\n\n    /**\n     * Creates a converter for an optional value. If 'onError'\n     * is 'failOnError', the converter accepts 'undefined' or a\n     * convertible value, but reports an error if it encounters\n     * a value that cannot be converted.  If 'onError' is 'ignoreErrors'\n     * (default) then values that cannot be converted result in a\n     * successful return of 'undefined'.\n     *\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     * */\n    optional(onError?: OnError): Converter<T|undefined, TC>;\n\n    /**\n     * Applies a (possibly) mapping conversion to the converted value.\n     * @param mapper A function which maps from the converted type to some other type.\n     */\n    map<T2>(mapper: (from: T) => Result<T2>): Converter<T2, TC>;\n\n    /**\n     * Applies an additional converter to the converted value.\n     * @param mapConverter The converter to be applied to the converted value\n     */\n    mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2, TC>;\n\n    /**\n     * Creates a converter with an optional constraint.  If the base converter\n     * succeeds, calls a supplied constraint evaluation function with the\n     * value and fails the conversion if the function returns either false\n     * or Failure<T>.\n     *\n     * @param constraint Constraint evaluation function\n     */\n    withConstraint(constraint: (val: T) => boolean|Result<T>): Converter<T, TC>;\n}\n\ntype InnerConvertedToType<TCONV> =\n    TCONV extends Converter<infer TTO>\n        ? (TTO extends Array<infer TTOELEM> ? InnerConvertedToType<TTOELEM>[] : TTO)\n        : (TCONV extends Array<infer TELEM> ? InnerConvertedToType<TELEM>[] : TCONV);\n\n/**\n * Infers the type that will be returned by an intstantiated converter.  Works\n * for complex as well as simple types.\n * @example ConvertedToType<typeof Converters.mapOf(Converters.stringArray)> is Map<string, string[]>\n */\nexport type ConvertedToType<TCONV> = TCONV extends Converter<infer TTO> ? InnerConvertedToType<TTO> : never;\n\n/**\n * Simple templated converter wrapper to simplify typed conversion from unknown.\n */\nexport class BaseConverter<T, TC=undefined> implements Converter<T, TC> {\n    protected readonly _defaultContext?: TC;\n    private readonly _converter: (from: unknown, self: Converter<T, TC>, context?: TC) => Result<T>;\n\n    public constructor(converter: (from: unknown, self: Converter<T, TC>, context?: TC) => Result<T>, defaultContext?: TC) {\n        this._converter = converter;\n        this._defaultContext = defaultContext;\n    }\n\n    /**\n     * Converts from unknown to <T>\n     * @param from The unknown to be converted\n     * @returns An @see Result with a value or an error message\n     */\n    public convert(from: unknown, context?: TC): Result<T> {\n        return this._converter(from, this, context ?? this._defaultContext);\n    }\n\n    /**\n     * Converts from unknown to <T> or undefined, as appropriate.\n     * If 'onError' is 'failOnError', the converter succeeds for\n     * 'undefined' or any convertible value, but reports an error\n     * if it encounters a value that cannot be converted.  If 'onError'\n     * is 'ignoreErrors' (default) then values that cannot be converted\n     * result in a successful return of 'undefined'.\n     * @param from The unknown to be converted\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     */\n    public convertOptional(from: unknown, context?: TC, onError?: OnError): Result<T|undefined> {\n        const result = this._converter(from, this, this._context(context));\n        if (result.isFailure()) {\n            onError = onError ?? 'ignoreErrors';\n            return ((from === undefined) || onError === 'ignoreErrors') ? succeed(undefined) : result;\n        }\n        return result;\n    }\n\n    /**\n     * Creates a converter for an optional value. If 'onError'\n     * is 'failOnError', the converter accepts 'undefined' or a\n     * convertible value, but reports an error if it encounters\n     * a value that cannot be converted.  If 'onError' is 'ignoreErrors'\n     * (default) then values that cannot be converted result in a\n     * successful return of 'undefined'.\n     *\n     * @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     * */\n    public optional(onError?: OnError): Converter<T|undefined, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T|undefined, TC>, context?: TC) => {\n            onError = onError ?? 'ignoreErrors';\n            return this.convertOptional(from, this._context(context), onError);\n        });\n    }\n\n    /**\n     * Applies a (possibly) mapping conversion to the converted value.\n     * @param mapper A function which maps from the converted type to some other type.\n     */\n    public map<T2>(mapper: (from: T) => Result<T2>): Converter<T2, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T2, TC>, context?: TC) => {\n            const innerResult = this._converter(from, this, this._context(context));\n            if (innerResult.isSuccess()) {\n                return mapper(innerResult.value);\n            }\n            return fail(innerResult.message);\n        });\n    }\n\n    /**\n     * Applies an additional converter to the converted value.\n     * @param mapConverter The converter to be applied to the converted value\n     */\n    public mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T2, TC>, context?: TC) => {\n            const innerResult = this._converter(from, this, this._context(context));\n            if (innerResult.isSuccess()) {\n                return mapConverter.convert(innerResult.value);\n            }\n            return fail(innerResult.message);\n        });\n    }\n\n    /**\n     * Creates a converter with an optional constraint.  If the base converter\n     * succeeds, calls a supplied constraint evaluation function with the\n     * value and fails the conversion if the function returns either false\n     * or Failure<T>.\n     *\n     * @param constraint Constraint evaluation function\n     */\n    public withConstraint(constraint: (val: T) => boolean|Result<T>): Converter<T, TC> {\n        return new BaseConverter((from: unknown, _self: Converter<T, TC>, context?: TC) => {\n            const result = this._converter(from, this, this._context(context));\n            if (result.isSuccess()) {\n                const constraintResult = constraint(result.value);\n                if (typeof constraintResult === 'boolean') {\n                    return constraintResult ? result : fail(`Value ${JSON.stringify(result.value)} does not meet constraint.`);\n                }\n                return constraintResult;\n            }\n            return result;\n        });\n    }\n\n    protected _context(supplied?: TC): TC|undefined {\n        return supplied ?? this._defaultContext;\n    }\n}\n"]} |
{ | ||
"name": "@fgv/ts-utils", | ||
"version": "0.3.15", | ||
"version": "0.3.16", | ||
"description": "Assorted Typescript Utilities", | ||
@@ -5,0 +5,0 @@ "main": "index.js", |
242902
2160