New Case Study:See how Anthropic automated 95% of dependency reviews with Socket.Learn More
Socket
Sign inDemoInstall
Socket

@fgv/ts-utils

Package Overview
Dependencies
Maintainers
1
Versions
95
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@fgv/ts-utils - npm Package Compare versions

Comparing version 0.3.4 to 0.3.5

36

converter.d.ts
import { Result } from './result';
declare type OnError = 'failOnError' | 'ignoreErrors';
export interface Converter<T> {
export interface Converter<T, TC = undefined> {
/**
* Converts from unknown to <T>
* @param from The unknown to be converted
* @param context An optional context applied to the conversion
* @returns An @see Result with a value or an error message
*/
convert(from: unknown): Result<T>;
convert(from: unknown, context?: TC): Result<T>;
/**

@@ -18,5 +19,6 @@ * Converts from unknown to <T> or undefined, as appropriate.

* @param from The unknown to be converted
* @param context Optional context for use by the converter
* @param onError Specifies handling of values that cannot be converted, default 'ignoreErrors'
*/
convertOptional(from: unknown, onError?: OnError): Result<T | undefined>;
convertOptional(from: unknown, context?: TC, onError?: OnError): Result<T | undefined>;
/**

@@ -32,3 +34,3 @@ * Creates a converter for an optional value. If 'onError'

* */
optional(onError?: OnError): Converter<T | undefined>;
optional(onError?: OnError): Converter<T | undefined, TC>;
/**

@@ -38,3 +40,3 @@ * Applies a (possibly) mapping conversion to the converted value.

*/
map<T2>(mapper: (from: T) => Result<T2>): Converter<T2>;
map<T2>(mapper: (from: T) => Result<T2>): Converter<T2, TC>;
/**

@@ -44,3 +46,3 @@ * Applies an additional converter to the converted value.

*/
mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2>;
mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2, TC>;
/**

@@ -54,3 +56,3 @@ * Creates a converter with an optional constraint. If the base converter

*/
withConstraint(constraint: (val: T) => boolean | Result<T>): Converter<T>;
withConstraint(constraint: (val: T) => boolean | Result<T>): Converter<T, TC>;
}

@@ -60,5 +62,6 @@ /**

*/
export declare class BaseConverter<T> implements Converter<T> {
private _converter;
constructor(converter: (from: unknown, self: Converter<T>) => Result<T>);
export declare class BaseConverter<T, TC = undefined> implements Converter<T, TC> {
protected readonly _defaultContext?: TC;
private readonly _converter;
constructor(converter: (from: unknown, self: Converter<T, TC>, context?: TC) => Result<T>, defaultContext?: TC);
/**

@@ -69,3 +72,3 @@ * Converts from unknown to <T>

*/
convert(from: unknown): Result<T>;
convert(from: unknown, context?: TC): Result<T>;
/**

@@ -81,3 +84,3 @@ * Converts from unknown to <T> or undefined, as appropriate.

*/
convertOptional(from: unknown, onError?: OnError): Result<T | undefined>;
convertOptional(from: unknown, context?: TC, onError?: OnError): Result<T | undefined>;
/**

@@ -93,3 +96,3 @@ * Creates a converter for an optional value. If 'onError'

* */
optional(onError?: OnError): Converter<T | undefined>;
optional(onError?: OnError): Converter<T | undefined, TC>;
/**

@@ -99,3 +102,3 @@ * Applies a (possibly) mapping conversion to the converted value.

*/
map<T2>(mapper: (from: T) => Result<T2>): Converter<T2>;
map<T2>(mapper: (from: T) => Result<T2>): Converter<T2, TC>;
/**

@@ -105,3 +108,3 @@ * Applies an additional converter to the converted value.

*/
mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2>;
mapConvert<T2>(mapConverter: Converter<T2>): Converter<T2, TC>;
/**

@@ -115,4 +118,5 @@ * Creates a converter with an optional constraint. If the base converter

*/
withConstraint(constraint: (val: T) => boolean | Result<T>): Converter<T>;
withConstraint(constraint: (val: T) => boolean | Result<T>): Converter<T, TC>;
protected _context(supplied?: TC): TC | undefined;
}
export {};

@@ -30,4 +30,5 @@ "use strict";

class BaseConverter {
constructor(converter) {
constructor(converter, defaultContext) {
this._converter = converter;
this._defaultContext = defaultContext;
}

@@ -39,4 +40,4 @@ /**

*/
convert(from) {
return this._converter(from, this);
convert(from, context) {
return this._converter(from, this, context !== null && context !== void 0 ? context : this._defaultContext);
}

@@ -53,4 +54,4 @@ /**

*/
convertOptional(from, onError) {
const result = this._converter(from, this);
convertOptional(from, context, onError) {
const result = this._converter(from, this, this._context(context));
if (result.isFailure()) {

@@ -73,5 +74,5 @@ onError = onError !== null && onError !== void 0 ? onError : 'ignoreErrors';

optional(onError) {
return new BaseConverter((from) => {
return new BaseConverter((from, _self, context) => {
onError = onError !== null && onError !== void 0 ? onError : 'ignoreErrors';
return this.convertOptional(from, onError);
return this.convertOptional(from, this._context(context), onError);
});

@@ -84,4 +85,4 @@ }

map(mapper) {
return new BaseConverter((from) => {
const innerResult = this._converter(from, this);
return new BaseConverter((from, _self, context) => {
const innerResult = this._converter(from, this, this._context(context));
if (innerResult.isSuccess()) {

@@ -98,4 +99,4 @@ return mapper(innerResult.value);

mapConvert(mapConverter) {
return new BaseConverter((from) => {
const innerResult = this._converter(from, this);
return new BaseConverter((from, _self, context) => {
const innerResult = this._converter(from, this, this._context(context));
if (innerResult.isSuccess()) {

@@ -116,4 +117,4 @@ return mapConverter.convert(innerResult.value);

withConstraint(constraint) {
return new BaseConverter((from) => {
const result = this._converter(from, this);
return new BaseConverter((from, _self, context) => {
const result = this._converter(from, this, this._context(context));
if (result.isSuccess()) {

@@ -129,4 +130,7 @@ const constraintResult = constraint(result.value);

}
_context(supplied) {
return supplied !== null && supplied !== void 0 ? supplied : this._defaultContext;
}
}
exports.BaseConverter = BaseConverter;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"converter.js","sourceRoot":"","sources":["../src/converter.ts"],"names":[],"mappings":";;;AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,qCAAiD;AA2DjD;;GAEG;AACH,MAAa,aAAa;IAGtB,YAAmB,SAA2D;QAC1E,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;IAChC,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAC,IAAa;QACxB,OAAO,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IACvC,CAAC;IAED;;;;;;;;;OASG;IACI,eAAe,CAAC,IAAa,EAAE,OAAiB;QACnD,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QAC3C,IAAI,MAAM,CAAC,SAAS,EAAE,EAAE;YACpB,OAAO,GAAG,OAAO,aAAP,OAAO,cAAP,OAAO,GAAI,cAAc,CAAC;YACpC,OAAO,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,IAAI,OAAO,KAAK,cAAc,CAAC,CAAC,CAAC,CAAC,gBAAO,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;SAC7F;QACD,OAAO,MAAM,CAAC;IAClB,CAAC;IAED;;;;;;;;;SASK;IACE,QAAQ,CAAC,OAAiB;QAC7B,OAAO,IAAI,aAAa,CAAC,CAAC,IAAa,EAAE,EAAE;YACvC,OAAO,GAAG,OAAO,aAAP,OAAO,cAAP,OAAO,GAAI,cAAc,CAAC;YACpC,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;OAGG;IACI,GAAG,CAAK,MAA+B;QAC1C,OAAO,IAAI,aAAa,CAAC,CAAC,IAAa,EAAE,EAAE;YACvC,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAChD,IAAI,WAAW,CAAC,SAAS,EAAE,EAAE;gBACzB,OAAO,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;aACpC;YACD,OAAO,aAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACrC,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;OAGG;IACI,UAAU,CAAK,YAA2B;QAC7C,OAAO,IAAI,aAAa,CAAC,CAAC,IAAa,EAAE,EAAE;YACvC,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAChD,IAAI,WAAW,CAAC,SAAS,EAAE,EAAE;gBACzB,OAAO,YAAY,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;aAClD;YACD,OAAO,aAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACrC,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;;;;OAOG;IACI,cAAc,CAAC,UAAyC;QAC3D,OAAO,IAAI,aAAa,CAAC,CAAC,IAAa,EAAE,EAAE;YACvC,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAC3C,IAAI,MAAM,CAAC,SAAS,EAAE,EAAE;gBACpB,MAAM,gBAAgB,GAAG,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAClD,IAAI,OAAO,gBAAgB,KAAK,SAAS,EAAE;oBACvC,OAAO,gBAAgB,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,aAAI,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,4BAA4B,CAAC,CAAC;iBAC9G;gBACD,OAAO,gBAAgB,CAAC;aAC3B;YACD,OAAO,MAAM,CAAC;QAClB,CAAC,CAAC,CAAC;IACP,CAAC;CACJ;AArGD,sCAqGC","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> {\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    convert(from: unknown): 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 onError Specifies handling of values that cannot be converted, default 'ignoreErrors'\n     */\n    convertOptional(from: unknown, 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>;\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>;\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>;\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>;\n}\n\n/**\n * Simple templated converter wrapper to simplify typed conversion from unknown.\n */\nexport class BaseConverter<T> implements Converter<T> {\n    private _converter: (from: unknown, self: Converter<T>) => Result<T>;\n\n    public constructor(converter: (from: unknown, self: Converter<T>) => Result<T>) {\n        this._converter = converter;\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): Result<T> {\n        return this._converter(from, this);\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, onError?: OnError): Result<T|undefined> {\n        const result = this._converter(from, this);\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> {\n        return new BaseConverter((from: unknown) => {\n            onError = onError ?? 'ignoreErrors';\n            return this.convertOptional(from, 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> {\n        return new BaseConverter((from: unknown) => {\n            const innerResult = this._converter(from, this);\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> {\n        return new BaseConverter((from: unknown) => {\n            const innerResult = this._converter(from, this);\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> {\n        return new BaseConverter((from: unknown) => {\n            const result = this._converter(from, this);\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"]}
//# 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 * 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"]}

@@ -10,14 +10,20 @@ import { BaseConverter, Converter } from './converter';

*/
export declare const string: BaseConverter<string>;
export declare const string: BaseConverter<string, undefined>;
/**
*
* @param values A converter to convert unknown to one of a set of
* supplied enumerated values. Anything else fails.
* Helper function to create a converter which converts unknown to string, applying
* template conversions supplied at construction time or at runtime as context.
* @param defaultContext optional default context to use for template values
*/
export declare function enumeratedValue<T>(values: T[]): Converter<T>;
export declare function templateString(defaultContext?: unknown): Converter<string, unknown>;
/**
* A converter to convert unknown to one of a set of supplied enumerated values. Anything else fails.
* Allowed enumerated values can also be supplied as context at conversion time.
* @param values Array of allowed values
*/
export declare function enumeratedValue<T>(values: T[]): Converter<T, T[]>;
/**
* A converter to convert unknown to a number. Numbers and strings
* with a numeric format succeed. Anything else fails.
*/
export declare const number: BaseConverter<number>;
export declare const number: BaseConverter<number, undefined>;
/**

@@ -28,3 +34,3 @@ * A converter to convert unknown to boolean. Boolean values or the

*/
export declare const boolean: BaseConverter<boolean>;
export declare const boolean: BaseConverter<boolean, undefined>;
/**

@@ -34,13 +40,14 @@ * A converter to convert an optional string value. Values of type string

*/
export declare const optionalString: Converter<string | undefined>;
export declare const optionalString: Converter<string | undefined, undefined>;
/**
* Creates a converter which converts any string into an array of strings
* by separating at a supplied delimiter.
* by separating at a supplied delimiter. Delimeter may also be supplied
* as context at conversion time.
* @param delimiter The delimiter at which to split.
*/
export declare function delimitedString(delimiter: string, options?: 'filtered' | 'all'): Converter<string[]>;
export declare function delimitedString(delimiter: string, options?: 'filtered' | 'all'): Converter<string[], string>;
/**
* A converter to convert an iso formatted string, a number or a Date object to a Date object
*/
export declare const isoDate: BaseConverter<Date>;
export declare const isoDate: BaseConverter<Date, undefined>;
/**

@@ -51,3 +58,3 @@ * A converter to convert an optional number value. Values of type number

*/
export declare const optionalNumber: Converter<number | undefined>;
export declare const optionalNumber: Converter<number | undefined, undefined>;
/**

@@ -58,3 +65,3 @@ * A converter to convert an optional boolean value. Values of type boolean

*/
export declare const optionalBoolean: Converter<boolean | undefined>;
export declare const optionalBoolean: Converter<boolean | undefined, undefined>;
/**

@@ -81,3 +88,3 @@ * A helper wrapper for polymorphic fields. Invokes the wrapped converters

*/
export declare function arrayOf<T>(converter: Converter<T>, onError?: OnError): Converter<T[]>;
export declare function arrayOf<T, TC = undefined>(converter: Converter<T, TC>, onError?: OnError): Converter<T[], TC>;
/**

@@ -90,3 +97,3 @@ * A helper wrapper for converting to Itemrray<T>. If onError is 'failOnError' (default),

*/
export declare function extendedArrayOf<T>(label: string, converter: Converter<T>, onError?: OnError): Converter<ExtendedArray<T>>;
export declare function extendedArrayOf<T, TC = undefined>(label: string, converter: Converter<T, TC>, onError?: OnError): Converter<ExtendedArray<T>, TC>;
/**

@@ -99,3 +106,3 @@ * A helper wrapper to convert the string-keyed properties of an object to a Record of T.

*/
export declare function recordOf<T>(converter: Converter<T>, onError?: 'fail' | 'ignore'): Converter<Record<string, T>>;
export declare function recordOf<T, TC = undefined>(converter: Converter<T, TC>, onError?: 'fail' | 'ignore'): Converter<Record<string, T>, TC>;
/**

@@ -108,3 +115,3 @@ * A helper wrapper to convert the string-keyed properties of an object to a Map of T.

*/
export declare function mapOf<T>(converter: Converter<T>, onError?: 'fail' | 'ignore'): Converter<Map<string, T>>;
export declare function mapOf<T, TC = undefined>(converter: Converter<T, TC>, onError?: 'fail' | 'ignore'): Converter<Map<string, T>, TC>;
/**

@@ -117,3 +124,3 @@ * A helper function to extract and convert a field from an object. Succeeds and returns

*/
export declare function field<T>(name: string, converter: Converter<T>): Converter<T>;
export declare function field<T, TC = undefined>(name: string, converter: Converter<T, TC>): Converter<T, TC>;
/**

@@ -127,3 +134,3 @@ * A helper function to extract and convert an optional field from an object. Succeeds

*/
export declare function optionalField<T>(name: string, converter: Converter<T>): Converter<T | undefined>;
export declare function optionalField<T, TC = undefined>(name: string, converter: Converter<T, TC>): Converter<T | undefined, TC>;
export declare type FieldConverters<T> = {

@@ -130,0 +137,0 @@ [key in keyof T]: Converter<T[key]>;

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.rangeOf = exports.rangeTypeOf = exports.transform = exports.object = exports.ObjectConverter = exports.optionalField = exports.field = exports.mapOf = exports.recordOf = exports.extendedArrayOf = exports.arrayOf = exports.oneOf = exports.optionalBoolean = exports.optionalNumber = exports.isoDate = exports.delimitedString = exports.optionalString = exports.boolean = exports.number = exports.enumeratedValue = exports.string = void 0;
exports.rangeOf = exports.rangeTypeOf = exports.transform = exports.object = exports.ObjectConverter = exports.optionalField = exports.field = exports.mapOf = exports.recordOf = exports.extendedArrayOf = exports.arrayOf = exports.oneOf = exports.optionalBoolean = exports.optionalNumber = exports.isoDate = exports.delimitedString = exports.optionalString = exports.boolean = exports.number = exports.enumeratedValue = exports.templateString = exports.string = void 0;
/*

@@ -30,2 +33,3 @@ * Copyright (c) 2020 Erik Fortune

const extendedArray_1 = require("./extendedArray");
const mustache_1 = __importDefault(require("mustache"));
const utils_1 = require("./utils");

@@ -42,10 +46,25 @@ /**

/**
*
* @param values A converter to convert unknown to one of a set of
* supplied enumerated values. Anything else fails.
* Helper function to create a converter which converts unknown to string, applying
* template conversions supplied at construction time or at runtime as context.
* @param defaultContext optional default context to use for template values
*/
function templateString(defaultContext) {
return new converter_1.BaseConverter((from, _self, context) => {
if (typeof from !== 'string') {
return result_1.fail(`Not a string: ${JSON.stringify(from)}`);
}
return result_1.captureResult(() => mustache_1.default.render(from, context));
}, defaultContext);
}
exports.templateString = templateString;
/**
* A converter to convert unknown to one of a set of supplied enumerated values. Anything else fails.
* Allowed enumerated values can also be supplied as context at conversion time.
* @param values Array of allowed values
*/
function enumeratedValue(values) {
return new converter_1.BaseConverter((from) => {
const index = values.indexOf(from);
return (index >= 0 ? result_1.succeed(values[index]) : result_1.fail(`Invalid enumerated value ${JSON.stringify(from)}`));
return new converter_1.BaseConverter((from, _self, context) => {
const v = context !== null && context !== void 0 ? context : values;
const index = v.indexOf(from);
return (index >= 0 ? result_1.succeed(v[index]) : result_1.fail(`Invalid enumerated value ${JSON.stringify(from)}`));
});

@@ -91,10 +110,11 @@ }

* Creates a converter which converts any string into an array of strings
* by separating at a supplied delimiter.
* by separating at a supplied delimiter. Delimeter may also be supplied
* as context at conversion time.
* @param delimiter The delimiter at which to split.
*/
function delimitedString(delimiter, options = 'filtered') {
return new converter_1.BaseConverter((from) => {
return new converter_1.BaseConverter((from, _self, context) => {
const result = exports.string.convert(from);
if (result.isSuccess()) {
let strings = result.value.split(delimiter);
let strings = result.value.split(context !== null && context !== void 0 ? context : delimiter);
if (options !== 'all') {

@@ -180,3 +200,3 @@ strings = strings.filter((s) => (s.trim().length > 0));

function arrayOf(converter, onError = 'failOnError') {
return new converter_1.BaseConverter((from) => {
return new converter_1.BaseConverter((from, _self, context) => {
if (!Array.isArray(from)) {

@@ -188,3 +208,3 @@ return result_1.fail(`Not an array: ${JSON.stringify(from)}`);

for (const item of from) {
const result = converter.convert(item);
const result = converter.convert(item, context);
if (result.isSuccess() && result.value !== undefined) {

@@ -224,3 +244,3 @@ successes.push(result.value);

function recordOf(converter, onError = 'fail') {
return new converter_1.BaseConverter((from) => {
return new converter_1.BaseConverter((from, _self, context) => {
if ((typeof from !== 'object') || Array.isArray(from)) {

@@ -233,3 +253,3 @@ return result_1.fail(`Not a string-keyed object: ${JSON.stringify(from)}`);

if (utils_1.isKeyOf(key, from)) {
const result = converter.convert(from[key]);
const result = converter.convert(from[key], context);
if (result.isSuccess()) {

@@ -257,3 +277,3 @@ record[key] = result.value;

function mapOf(converter, onError = 'fail') {
return new converter_1.BaseConverter((from) => {
return new converter_1.BaseConverter((from, _self, context) => {
if ((typeof from !== 'object') || Array.isArray(from)) {

@@ -266,3 +286,3 @@ return result_1.fail(`Not a string-keyed object: ${JSON.stringify(from)}`);

if (utils_1.isKeyOf(key, from)) {
const result = converter.convert(from[key]);
const result = converter.convert(from[key], context);
if (result.isSuccess()) {

@@ -290,6 +310,6 @@ map.set(key, result.value);

function field(name, converter) {
return new converter_1.BaseConverter((from) => {
return new converter_1.BaseConverter((from, _self, context) => {
if (typeof from === 'object' && from !== null) {
if (utils_1.isKeyOf(name, from)) {
return converter.convert(from[name]).onFailure((message) => {
return converter.convert(from[name], context).onFailure((message) => {
return result_1.fail(`Field ${name}: ${message}`);

@@ -313,6 +333,6 @@ });

function optionalField(name, converter) {
return new converter_1.BaseConverter((from) => {
return new converter_1.BaseConverter((from, _self, context) => {
if (typeof from === 'object' && from !== null) {
if (utils_1.isKeyOf(name, from)) {
const result = converter.convert(from[name]).onFailure((message) => {
const result = converter.convert(from[name], context).onFailure((message) => {
return result_1.fail(`${name}: ${message}`);

@@ -441,2 +461,2 @@ });

exports.rangeOf = rangeOf;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"converters.js","sourceRoot":"","sources":["../src/converters.ts"],"names":[],"mappings":"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","sourcesContent":["\n/*\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 { BaseConverter, Converter } from './converter';\nimport { RangeOf, RangeOfProperties } from './rangeOf';\nimport { Result, captureResult, fail, succeed } from './result';\n\nimport { DateTime } from 'luxon';\nimport { ExtendedArray } from './extendedArray';\nimport { isKeyOf } from './utils';\n\ntype OnError = 'failOnError' | 'ignoreErrors';\n\n/**\n * A converter to convert unknown to string. Values of type\n * string succeed.  Anything else fails.\n */\nexport const string = new BaseConverter<string>((from: unknown) => {\n    return typeof from === 'string'\n        ? succeed(from as string)\n        : fail(`Not a string: ${JSON.stringify(from)}`);\n});\n\n/**\n *\n * @param values A converter to convert unknown to one of a set of\n * supplied enumerated values.  Anything else fails.\n */\nexport function enumeratedValue<T>(values: T[]): Converter<T> {\n    return new BaseConverter<T>((from: unknown): Result<T> => {\n        const index = values.indexOf(from as T);\n        return (index >= 0 ? succeed(values[index]) : fail(`Invalid enumerated value ${JSON.stringify(from)}`));\n    });\n}\n\n/**\n * A converter to convert unknown to a number.  Numbers and strings\n * with a numeric format succeed.  Anything else fails.\n */\nexport const number = new BaseConverter<number>((from: unknown) => {\n    if (typeof from !== 'number') {\n        const num: number = (typeof from === 'string' ? Number(from) : NaN);\n        return isNaN(num)\n            ? fail(`Not a number: ${JSON.stringify(from)}`)\n            : succeed(num);\n    }\n    return succeed(from);\n});\n\n/**\n * A converter to convert unknown to boolean. Boolean values or the\n * case-insensitive strings 'true' and 'false' succeed.  Anything\n * else fails.\n */\nexport const boolean = new BaseConverter<boolean>((from: unknown) => {\n    if (typeof from === 'boolean') {\n        return succeed(from as boolean);\n    }\n    else if (typeof from === 'string') {\n        switch (from.toLowerCase()) {\n            case 'true': return succeed(true);\n            case 'false': return succeed(false);\n        }\n    }\n    return fail(`Not a boolean: ${JSON.stringify(from)}`);\n});\n\n/**\n * A converter to convert an optional string value. Values of type string\n * are returned.  Anything else returns success with an undefined value.\n */\nexport const optionalString = string.optional();\n\n/**\n * Creates a converter which converts any string into an array of strings\n * by separating at a supplied delimiter.\n * @param delimiter The delimiter at which to split.\n */\nexport function delimitedString(delimiter: string, options: 'filtered'|'all' = 'filtered'): Converter<string[]> {\n    return new BaseConverter<string[]>((from: unknown) => {\n        const result = string.convert(from);\n        if (result.isSuccess()) {\n            let strings = result.value.split(delimiter);\n            if (options !== 'all') {\n                strings = strings.filter((s) => (s.trim().length > 0));\n            }\n            return succeed(strings);\n        }\n        return fail(result.message);\n    });\n}\n\n/**\n * A converter to convert an iso formatted string, a number or a Date object to a Date object\n */\nexport const isoDate = new BaseConverter<Date>((from: unknown) => {\n    if (typeof from === 'string') {\n        const dt = DateTime.fromISO(from);\n        if (dt.isValid) {\n            return succeed(dt.toJSDate());\n        }\n        return fail(`Invalid date: ${dt.invalidExplanation}`);\n    }\n    else if (typeof from === 'number') {\n        return succeed(new Date(from));\n    }\n    else if (from instanceof Date) {\n        return succeed(from);\n    }\n    return fail(`Cannot convert ${JSON.stringify(from)} to Date`);\n});\n\n/**\n * A converter to convert an optional number value. Values of type number\n * or numeric strings are converted and returned. Anything else returns\n * success with an undefined value.\n */\nexport const optionalNumber = number.optional();\n\n/**\n * A converter to convert an optional boolean value. Values of type boolean\n * or strings that match (case-insensitive) 'true' or 'false' are converted\n * and returned.  Anything else returns success with an undefined value.\n */\nexport const optionalBoolean = boolean.optional();\n\n/**\n * A helper wrapper for polymorphic fields. Invokes the wrapped converters\n * in sequence, returning the first successful result.  Returns an error\n * if none of the supplied converters can convert the value.\n *\n * If onError is 'ignoreErrors' (default), then errors from any of the\n * converters are ignored provided that some converter succeeds.  If\n * onError is 'failOnError', then an error from any converter fails the entire\n * conversion.\n *\n * @param converters An ordered list of converters to be considered\n * @param onError Specifies treatment of unconvertable elements\n */\nexport function oneOf<T>(converters: Array<Converter<T>>, onError: OnError = 'ignoreErrors'): Converter<T> {\n    return new BaseConverter((from: unknown) => {\n        const errors: string[] = [];\n        for (const converter of converters) {\n            const result = converter.convert(from);\n            if (result.isSuccess() && (result.value !== undefined)) {\n                return result;\n            }\n\n            if (result.isFailure()) {\n                if (onError === 'failOnError') {\n                    return result;\n                }\n                errors.push(result.message);\n            }\n        }\n        return fail(`No matching converter for ${JSON.stringify(from)}: ${errors.join('\\n')}`);\n    });\n}\n\n/**\n * A helper wrapper for converting an array of <T>.  If onError is 'failOnError' (default),\n * then the entire conversion fails if any element cannot be converted.  If onError\n * is 'ignoreErrors', failing elements are silently ignored.\n * @param converter Converter used to convert each item in the array\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function arrayOf<T>(converter: Converter<T>, onError: OnError = 'failOnError'): Converter<T[]> {\n    return new BaseConverter((from: unknown) => {\n        if (!Array.isArray(from)) {\n            return fail(`Not an array: ${JSON.stringify(from)}`);\n        }\n\n        const successes: T[] = [];\n        const errors: string[] = [];\n        for (const item of from) {\n            const result = converter.convert(item);\n            if (result.isSuccess() && result.value !== undefined) {\n                successes.push(result.value);\n            }\n            else if (result.isFailure()) {\n                errors.push(result.message);\n            }\n        }\n\n        return (errors.length === 0) || (onError === 'ignoreErrors')\n            ? succeed(successes)\n            : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper wrapper for converting to Itemrray<T>.  If onError is 'failOnError' (default),\n * then the entire conversion fails if any element cannot be converted.  If onError\n * is 'ignoreErrors', failing elements are silently ignored.\n * @param converter Converter used to convert each item in the array\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function extendedArrayOf<T>(label: string, converter: Converter<T>, onError: OnError = 'failOnError'): Converter<ExtendedArray<T>> {\n    return arrayOf(converter, onError).map((items: T[]) => {\n        return captureResult(() => new ExtendedArray(label, ...items));\n    });\n}\n\n/**\n * A helper wrapper to convert the string-keyed properties of an object to a Record of T.\n * If onError is 'fail' (default),  then the entire conversion fails if any element\n * cannot be converted.  If onError is 'ignore' failing elements are silently ignored.\n * @param converter Converter used to convert each item in the record\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function recordOf<T>(converter: Converter<T>, onError: 'fail'|'ignore' = 'fail'): Converter<Record<string, T>> {\n    return new BaseConverter((from: unknown) => {\n        if ((typeof from !== 'object') || Array.isArray(from)) {\n            return fail(`Not a string-keyed object: ${JSON.stringify(from)}`);\n        }\n\n        const record: Record<string, T> = {};\n        const errors: string[] = [];\n\n        for (const key in from) {\n            if (isKeyOf(key, from)) {\n                const result = converter.convert(from[key] as unknown);\n                if (result.isSuccess()) {\n                    record[key] = result.value;\n                }\n                else {\n                    errors.push(result.message);\n                }\n            }\n        }\n\n        return (errors.length === 0) || (onError === 'ignore')\n            ? succeed(record)\n            : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper wrapper to convert the string-keyed properties of an object to a Map of T.\n * If onError is 'fail' (default),  then the entire conversion fails if any element\n * cannot be converted.  If onError is 'ignore' failing elements are silently ignored.\n * @param converter Converter used to convert each item in the record\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function mapOf<T>(converter: Converter<T>, onError: 'fail'|'ignore' = 'fail'): Converter<Map<string, T>> {\n    return new BaseConverter((from: unknown) => {\n        if ((typeof from !== 'object') || Array.isArray(from)) {\n            return fail(`Not a string-keyed object: ${JSON.stringify(from)}`);\n        }\n\n        const map = new Map<string, T>();\n        const errors: string[] = [];\n\n        for (const key in from) {\n            if (isKeyOf(key, from)) {\n                const result = converter.convert(from[key] as unknown);\n                if (result.isSuccess()) {\n                    map.set(key, result.value);\n                }\n                else {\n                    errors.push(result.message);\n                }\n            }\n        }\n\n        return (errors.length === 0) || (onError === 'ignore')\n            ? succeed(map)\n            : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper function to extract and convert a field from an object. Succeeds and returns\n * the converted value if the field exists in the supplied parameter and can be converted.\n * Fails otherwise.\n * @param name The name of the field to be extracted.\n * @param converter Converter used to convert the extracted field.\n */\nexport function field<T>(name: string, converter: Converter<T>): Converter<T> {\n    return new BaseConverter((from: unknown) => {\n        if (typeof from === 'object' && from !== null) {\n            if (isKeyOf(name, from)) {\n                return converter.convert(from[name]).onFailure((message) => {\n                    return fail(`Field ${name}: ${message}`);\n                });\n            }\n            return fail(`Field ${name} not found in: ${JSON.stringify(from)}`);\n        }\n        return fail(`Cannot convert field \"${name}\" from non-object ${JSON.stringify(from)}`);\n    });\n}\n\n/**\n * A helper function to extract and convert an optional field from an object. Succeeds\n * and returns the converted value if the field exists in the supplied parameter and can\n * be converted. Succeeds with undefined if the parameter is an object but the named field\n * is not present. Fails if the supplied parameter is not an object.\n * @param name The name of the field to be extracted.\n * @param converter Converter used to convert the extracted field.\n */\nexport function optionalField<T>(name: string, converter: Converter<T>): Converter<T|undefined> {\n    return new BaseConverter((from: unknown) => {\n        if (typeof from === 'object' && from !== null) {\n            if (isKeyOf(name, from)) {\n                const result = converter.convert(from[name]).onFailure((message) => {\n                    return fail(`${name}: ${message}`);\n                });\n\n                // if conversion was successful or input was undefined we\n                // succeed with 'undefined', but we propagate actual\n                // failures.\n                if (result.isSuccess() || (from[name] !== undefined)) {\n                    return result;\n                }\n            }\n            return succeed(undefined);\n        }\n        return fail(`Cannot convert field \"${name}\" from non-object ${JSON.stringify(from)}`);\n    });\n}\n\nexport type FieldConverters<T> = { [ key in keyof T ]: Converter<T[key]> };\n\nexport class ObjectConverter<T> extends BaseConverter<T> {\n    public readonly fields: FieldConverters<T>;\n    public readonly optionalFields: (keyof T)[];\n\n    public constructor(fields: FieldConverters<T>, optional?: (keyof T)[]) {\n        super((from: unknown) => {\n            // eslint bug thinks key is used before defined\n            // eslint-disable-next-line no-use-before-define\n            const converted = {} as { [key in keyof T]: T[key] };\n            const errors: string[] = [];\n            for (const key in fields) {\n                if (fields[key]) {\n                    const isOptional = optional?.includes(key) ?? false;\n                    const result = isOptional\n                        ? optionalField(key, fields[key]).convert(from)\n                        : field(key, fields[key]).convert(from);\n                    if (result.isSuccess() && (result.value !== undefined)) {\n                        converted[key] = result.value;\n                    }\n                    else if (result.isFailure()) {\n                        errors.push(result.message);\n                    }\n                }\n            }\n            return (errors.length === 0) ? succeed(converted) : fail(errors.join('\\n'));\n        });\n\n        this.fields = fields;\n        this.optionalFields = optional ?? [];\n    }\n\n    public partial(optional?: (keyof T)[]): ObjectConverter<Partial<T>> {\n        return new ObjectConverter<Partial<T>>(this.fields as FieldConverters<Partial<T>>, optional);\n    }\n\n    public addPartial(addOptionalFields: (keyof T)[]): ObjectConverter<Partial<T>> {\n        return this.partial([...this.optionalFields, ...addOptionalFields]);\n    }\n}\n\n/**\n * Helper to convert an object without changing shape. The source parameter is an object with\n * key names that correspond to the target object and the appropriate corresponding converter\n * as the property value. If all of the requested fields exist and can be converted, returns a\n * new object with the converted values under the original key names.  If any fields do not exist\n * or cannot be converted, the entire conversion fails.\n *\n * Fields that succeed but convert to undefined are omitted from the result object but do not\n * fail the conversion.\n * @param fields An object containing defining the shape and converters to be applied.\n */\nexport function object<T>(fields: FieldConverters<T>, optional?: (keyof T)[]): ObjectConverter<T> {\n    return new ObjectConverter(fields, optional);\n}\n\n/**\n * Helper to convert an object to a new object with a different shape. The source parameter is\n * an object with key names that correspond to the target object, and an approriate _field_\n * converter that will extract and convert a single field from the source object.\n *\n * If all of the extracted fields exist and can be converted, returns a new object with the\n * converted values under the original key names.  If any fields to be extracted do not exist\n * or cannot be converted, the entire conversion fails.\n *\n * Fields that succeed but convert to undefined are omitted from the result object but do not\n * fail the conversion.\n *\n * @param fields An object defining the shape of the target object and the field converters\n * to be used to construct it.\n */\nexport function transform<T>(fields: FieldConverters<T>): Converter<T> {\n    return new BaseConverter((from: unknown) => {\n        // eslint bug thinks key is used before defined\n        // eslint-disable-next-line no-use-before-define\n        const converted = {} as { [ key in keyof T]: T[key] };\n        const errors: string[] = [];\n\n        for (const key in fields) {\n            if (fields[key]) {\n                const result = fields[key].convert(from);\n                if (result.isSuccess() && (result.value !== undefined)) {\n                    converted[key] = result.value;\n                }\n                else if (result.isFailure()) {\n                    errors.push(result.message);\n                }\n            }\n        }\n\n        return (errors.length === 0) ? succeed(converted) : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper wrapper to convert a range of some other comparable type\n * @param converter Converter used to convert min and max extent of the raid\n * @param constructor Optional static constructor to instantiate the object\n */\nexport function rangeTypeOf<T, RT extends RangeOf<T>>(converter: Converter<T>, constructor: (init: RangeOfProperties<T>) => Result<RT>): Converter<RT> {\n    return new BaseConverter((from: unknown) => {\n        const result = object({\n            min: converter,\n            max: converter,\n        }, ['min', 'max']).convert(from);\n        if (result.isSuccess()) {\n            return constructor({ min: result.value.min, max: result.value.max });\n        }\n        return fail(result.message);\n    });\n}\n\nexport function rangeOf<T>(converter: Converter<T>): Converter<RangeOf<T>> {\n    return rangeTypeOf<T, RangeOf<T>>(converter, RangeOf.createRange);\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"converters.js","sourceRoot":"","sources":["../src/converters.ts"],"names":[],"mappings":"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","sourcesContent":["\n/*\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 { BaseConverter, Converter } from './converter';\nimport { RangeOf, RangeOfProperties } from './rangeOf';\nimport { Result, captureResult, fail, succeed } from './result';\n\nimport { DateTime } from 'luxon';\nimport { ExtendedArray } from './extendedArray';\nimport Mustache from 'mustache';\nimport { isKeyOf } from './utils';\n\ntype OnError = 'failOnError' | 'ignoreErrors';\n\n/**\n * A converter to convert unknown to string. Values of type\n * string succeed.  Anything else fails.\n */\nexport const string = new BaseConverter<string>((from: unknown) => {\n    return typeof from === 'string'\n        ? succeed(from as string)\n        : fail(`Not a string: ${JSON.stringify(from)}`);\n});\n\n/**\n * Helper function to create a converter which converts unknown to string, applying\n * template conversions supplied at construction time or at runtime as context.\n * @param defaultContext optional default context to use for template values\n */\nexport function templateString(defaultContext?: unknown): Converter<string, unknown> {\n    return new BaseConverter<string, unknown>((from: unknown, _self: Converter<string, unknown>, context?: unknown) => {\n        if (typeof from !== 'string') {\n            return fail(`Not a string: ${JSON.stringify(from)}`);\n        }\n        return captureResult(() => Mustache.render(from, context));\n    }, defaultContext);\n}\n\n/**\n * A converter to convert unknown to one of a set of supplied enumerated values. Anything else fails.\n * Allowed enumerated values can also be supplied as context at conversion time.\n * @param values Array of allowed values\n */\nexport function enumeratedValue<T>(values: T[]): Converter<T, T[]> {\n    return new BaseConverter((from: unknown, _self: Converter<T, T[]>, context?: T[]): Result<T> => {\n        const v = context ?? values;\n        const index = v.indexOf(from as T);\n        return (index >= 0 ? succeed(v[index]) : fail(`Invalid enumerated value ${JSON.stringify(from)}`));\n    });\n}\n\n/**\n * A converter to convert unknown to a number.  Numbers and strings\n * with a numeric format succeed.  Anything else fails.\n */\nexport const number = new BaseConverter<number>((from: unknown) => {\n    if (typeof from !== 'number') {\n        const num: number = (typeof from === 'string' ? Number(from) : NaN);\n        return isNaN(num)\n            ? fail(`Not a number: ${JSON.stringify(from)}`)\n            : succeed(num);\n    }\n    return succeed(from);\n});\n\n/**\n * A converter to convert unknown to boolean. Boolean values or the\n * case-insensitive strings 'true' and 'false' succeed.  Anything\n * else fails.\n */\nexport const boolean = new BaseConverter<boolean>((from: unknown) => {\n    if (typeof from === 'boolean') {\n        return succeed(from as boolean);\n    }\n    else if (typeof from === 'string') {\n        switch (from.toLowerCase()) {\n            case 'true': return succeed(true);\n            case 'false': return succeed(false);\n        }\n    }\n    return fail(`Not a boolean: ${JSON.stringify(from)}`);\n});\n\n/**\n * A converter to convert an optional string value. Values of type string\n * are returned.  Anything else returns success with an undefined value.\n */\nexport const optionalString = string.optional();\n\n/**\n * Creates a converter which converts any string into an array of strings\n * by separating at a supplied delimiter. Delimeter may also be supplied\n * as context at conversion time.\n * @param delimiter The delimiter at which to split.\n */\nexport function delimitedString(delimiter: string, options: 'filtered'|'all' = 'filtered'): Converter<string[], string> {\n    return new BaseConverter<string[], string>((from: unknown, _self: Converter<string[], string>, context?: string) => {\n        const result = string.convert(from);\n        if (result.isSuccess()) {\n            let strings = result.value.split(context ?? delimiter);\n            if (options !== 'all') {\n                strings = strings.filter((s) => (s.trim().length > 0));\n            }\n            return succeed(strings);\n        }\n        return fail(result.message);\n    });\n}\n\n/**\n * A converter to convert an iso formatted string, a number or a Date object to a Date object\n */\nexport const isoDate = new BaseConverter<Date>((from: unknown) => {\n    if (typeof from === 'string') {\n        const dt = DateTime.fromISO(from);\n        if (dt.isValid) {\n            return succeed(dt.toJSDate());\n        }\n        return fail(`Invalid date: ${dt.invalidExplanation}`);\n    }\n    else if (typeof from === 'number') {\n        return succeed(new Date(from));\n    }\n    else if (from instanceof Date) {\n        return succeed(from);\n    }\n    return fail(`Cannot convert ${JSON.stringify(from)} to Date`);\n});\n\n/**\n * A converter to convert an optional number value. Values of type number\n * or numeric strings are converted and returned. Anything else returns\n * success with an undefined value.\n */\nexport const optionalNumber = number.optional();\n\n/**\n * A converter to convert an optional boolean value. Values of type boolean\n * or strings that match (case-insensitive) 'true' or 'false' are converted\n * and returned.  Anything else returns success with an undefined value.\n */\nexport const optionalBoolean = boolean.optional();\n\n/**\n * A helper wrapper for polymorphic fields. Invokes the wrapped converters\n * in sequence, returning the first successful result.  Returns an error\n * if none of the supplied converters can convert the value.\n *\n * If onError is 'ignoreErrors' (default), then errors from any of the\n * converters are ignored provided that some converter succeeds.  If\n * onError is 'failOnError', then an error from any converter fails the entire\n * conversion.\n *\n * @param converters An ordered list of converters to be considered\n * @param onError Specifies treatment of unconvertable elements\n */\nexport function oneOf<T>(converters: Array<Converter<T>>, onError: OnError = 'ignoreErrors'): Converter<T> {\n    return new BaseConverter((from: unknown) => {\n        const errors: string[] = [];\n        for (const converter of converters) {\n            const result = converter.convert(from);\n            if (result.isSuccess() && (result.value !== undefined)) {\n                return result;\n            }\n\n            if (result.isFailure()) {\n                if (onError === 'failOnError') {\n                    return result;\n                }\n                errors.push(result.message);\n            }\n        }\n        return fail(`No matching converter for ${JSON.stringify(from)}: ${errors.join('\\n')}`);\n    });\n}\n\n/**\n * A helper wrapper for converting an array of <T>.  If onError is 'failOnError' (default),\n * then the entire conversion fails if any element cannot be converted.  If onError\n * is 'ignoreErrors', failing elements are silently ignored.\n * @param converter Converter used to convert each item in the array\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function arrayOf<T, TC=undefined>(converter: Converter<T, TC>, onError: OnError = 'failOnError'): Converter<T[], TC> {\n    return new BaseConverter((from: unknown, _self: Converter<T[], TC>, context?: TC) => {\n        if (!Array.isArray(from)) {\n            return fail(`Not an array: ${JSON.stringify(from)}`);\n        }\n\n        const successes: T[] = [];\n        const errors: string[] = [];\n        for (const item of from) {\n            const result = converter.convert(item, context);\n            if (result.isSuccess() && result.value !== undefined) {\n                successes.push(result.value);\n            }\n            else if (result.isFailure()) {\n                errors.push(result.message);\n            }\n        }\n\n        return (errors.length === 0) || (onError === 'ignoreErrors')\n            ? succeed(successes)\n            : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper wrapper for converting to Itemrray<T>.  If onError is 'failOnError' (default),\n * then the entire conversion fails if any element cannot be converted.  If onError\n * is 'ignoreErrors', failing elements are silently ignored.\n * @param converter Converter used to convert each item in the array\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function extendedArrayOf<T, TC=undefined>(label: string, converter: Converter<T, TC>, onError: OnError = 'failOnError'): Converter<ExtendedArray<T>, TC> {\n    return arrayOf(converter, onError).map((items: T[]) => {\n        return captureResult(() => new ExtendedArray(label, ...items));\n    });\n}\n\n/**\n * A helper wrapper to convert the string-keyed properties of an object to a Record of T.\n * If onError is 'fail' (default),  then the entire conversion fails if any element\n * cannot be converted.  If onError is 'ignore' failing elements are silently ignored.\n * @param converter Converter used to convert each item in the record\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function recordOf<T, TC=undefined>(converter: Converter<T, TC>, onError: 'fail'|'ignore' = 'fail'): Converter<Record<string, T>, TC> {\n    return new BaseConverter((from: unknown, _self: Converter<Record<string, T>, TC>, context?: TC) => {\n        if ((typeof from !== 'object') || Array.isArray(from)) {\n            return fail(`Not a string-keyed object: ${JSON.stringify(from)}`);\n        }\n\n        const record: Record<string, T> = {};\n        const errors: string[] = [];\n\n        for (const key in from) {\n            if (isKeyOf(key, from)) {\n                const result = converter.convert(from[key] as unknown, context);\n                if (result.isSuccess()) {\n                    record[key] = result.value;\n                }\n                else {\n                    errors.push(result.message);\n                }\n            }\n        }\n\n        return (errors.length === 0) || (onError === 'ignore')\n            ? succeed(record)\n            : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper wrapper to convert the string-keyed properties of an object to a Map of T.\n * If onError is 'fail' (default),  then the entire conversion fails if any element\n * cannot be converted.  If onError is 'ignore' failing elements are silently ignored.\n * @param converter Converter used to convert each item in the record\n * @param ignoreErrors Specifies treatment of unconvertable elements\n */\nexport function mapOf<T, TC=undefined>(converter: Converter<T, TC>, onError: 'fail'|'ignore' = 'fail'): Converter<Map<string, T>, TC> {\n    return new BaseConverter((from: unknown, _self: Converter<Map<string, T>, TC>, context?: TC) => {\n        if ((typeof from !== 'object') || Array.isArray(from)) {\n            return fail(`Not a string-keyed object: ${JSON.stringify(from)}`);\n        }\n\n        const map = new Map<string, T>();\n        const errors: string[] = [];\n\n        for (const key in from) {\n            if (isKeyOf(key, from)) {\n                const result = converter.convert(from[key] as unknown, context);\n                if (result.isSuccess()) {\n                    map.set(key, result.value);\n                }\n                else {\n                    errors.push(result.message);\n                }\n            }\n        }\n\n        return (errors.length === 0) || (onError === 'ignore')\n            ? succeed(map)\n            : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper function to extract and convert a field from an object. Succeeds and returns\n * the converted value if the field exists in the supplied parameter and can be converted.\n * Fails otherwise.\n * @param name The name of the field to be extracted.\n * @param converter Converter used to convert the extracted field.\n */\nexport function field<T, TC=undefined>(name: string, converter: Converter<T, TC>): Converter<T, TC> {\n    return new BaseConverter((from: unknown, _self: Converter<T, TC>, context?: TC) => {\n        if (typeof from === 'object' && from !== null) {\n            if (isKeyOf(name, from)) {\n                return converter.convert(from[name], context).onFailure((message) => {\n                    return fail(`Field ${name}: ${message}`);\n                });\n            }\n            return fail(`Field ${name} not found in: ${JSON.stringify(from)}`);\n        }\n        return fail(`Cannot convert field \"${name}\" from non-object ${JSON.stringify(from)}`);\n    });\n}\n\n/**\n * A helper function to extract and convert an optional field from an object. Succeeds\n * and returns the converted value if the field exists in the supplied parameter and can\n * be converted. Succeeds with undefined if the parameter is an object but the named field\n * is not present. Fails if the supplied parameter is not an object.\n * @param name The name of the field to be extracted.\n * @param converter Converter used to convert the extracted field.\n */\nexport function optionalField<T, TC=undefined>(name: string, converter: Converter<T, TC>): Converter<T|undefined, TC> {\n    return new BaseConverter((from: unknown, _self: Converter<T|undefined, TC>, context?: TC) => {\n        if (typeof from === 'object' && from !== null) {\n            if (isKeyOf(name, from)) {\n                const result = converter.convert(from[name], context).onFailure((message) => {\n                    return fail(`${name}: ${message}`);\n                });\n\n                // if conversion was successful or input was undefined we\n                // succeed with 'undefined', but we propagate actual\n                // failures.\n                if (result.isSuccess() || (from[name] !== undefined)) {\n                    return result;\n                }\n            }\n            return succeed(undefined);\n        }\n        return fail(`Cannot convert field \"${name}\" from non-object ${JSON.stringify(from)}`);\n    });\n}\n\nexport type FieldConverters<T> = { [ key in keyof T ]: Converter<T[key]> };\n\nexport class ObjectConverter<T> extends BaseConverter<T> {\n    public readonly fields: FieldConverters<T>;\n    public readonly optionalFields: (keyof T)[];\n\n    public constructor(fields: FieldConverters<T>, optional?: (keyof T)[]) {\n        super((from: unknown) => {\n            // eslint bug thinks key is used before defined\n            // eslint-disable-next-line no-use-before-define\n            const converted = {} as { [key in keyof T]: T[key] };\n            const errors: string[] = [];\n            for (const key in fields) {\n                if (fields[key]) {\n                    const isOptional = optional?.includes(key) ?? false;\n                    const result = isOptional\n                        ? optionalField(key, fields[key]).convert(from)\n                        : field(key, fields[key]).convert(from);\n                    if (result.isSuccess() && (result.value !== undefined)) {\n                        converted[key] = result.value;\n                    }\n                    else if (result.isFailure()) {\n                        errors.push(result.message);\n                    }\n                }\n            }\n            return (errors.length === 0) ? succeed(converted) : fail(errors.join('\\n'));\n        });\n\n        this.fields = fields;\n        this.optionalFields = optional ?? [];\n    }\n\n    public partial(optional?: (keyof T)[]): ObjectConverter<Partial<T>> {\n        return new ObjectConverter<Partial<T>>(this.fields as FieldConverters<Partial<T>>, optional);\n    }\n\n    public addPartial(addOptionalFields: (keyof T)[]): ObjectConverter<Partial<T>> {\n        return this.partial([...this.optionalFields, ...addOptionalFields]);\n    }\n}\n\n/**\n * Helper to convert an object without changing shape. The source parameter is an object with\n * key names that correspond to the target object and the appropriate corresponding converter\n * as the property value. If all of the requested fields exist and can be converted, returns a\n * new object with the converted values under the original key names.  If any fields do not exist\n * or cannot be converted, the entire conversion fails.\n *\n * Fields that succeed but convert to undefined are omitted from the result object but do not\n * fail the conversion.\n * @param fields An object containing defining the shape and converters to be applied.\n */\nexport function object<T>(fields: FieldConverters<T>, optional?: (keyof T)[]): ObjectConverter<T> {\n    return new ObjectConverter(fields, optional);\n}\n\n/**\n * Helper to convert an object to a new object with a different shape. The source parameter is\n * an object with key names that correspond to the target object, and an approriate _field_\n * converter that will extract and convert a single field from the source object.\n *\n * If all of the extracted fields exist and can be converted, returns a new object with the\n * converted values under the original key names.  If any fields to be extracted do not exist\n * or cannot be converted, the entire conversion fails.\n *\n * Fields that succeed but convert to undefined are omitted from the result object but do not\n * fail the conversion.\n *\n * @param fields An object defining the shape of the target object and the field converters\n * to be used to construct it.\n */\nexport function transform<T>(fields: FieldConverters<T>): Converter<T> {\n    return new BaseConverter((from: unknown) => {\n        // eslint bug thinks key is used before defined\n        // eslint-disable-next-line no-use-before-define\n        const converted = {} as { [ key in keyof T]: T[key] };\n        const errors: string[] = [];\n\n        for (const key in fields) {\n            if (fields[key]) {\n                const result = fields[key].convert(from);\n                if (result.isSuccess() && (result.value !== undefined)) {\n                    converted[key] = result.value;\n                }\n                else if (result.isFailure()) {\n                    errors.push(result.message);\n                }\n            }\n        }\n\n        return (errors.length === 0) ? succeed(converted) : fail(errors.join('\\n'));\n    });\n}\n\n/**\n * A helper wrapper to convert a range of some other comparable type\n * @param converter Converter used to convert min and max extent of the raid\n * @param constructor Optional static constructor to instantiate the object\n */\nexport function rangeTypeOf<T, RT extends RangeOf<T>>(converter: Converter<T>, constructor: (init: RangeOfProperties<T>) => Result<RT>): Converter<RT> {\n    return new BaseConverter((from: unknown) => {\n        const result = object({\n            min: converter,\n            max: converter,\n        }, ['min', 'max']).convert(from);\n        if (result.isSuccess()) {\n            return constructor({ min: result.value.min, max: result.value.max });\n        }\n        return fail(result.message);\n    });\n}\n\nexport function rangeOf<T>(converter: Converter<T>): Converter<RangeOf<T>> {\n    return rangeTypeOf<T, RangeOf<T>>(converter, RangeOf.createRange);\n}\n"]}
{
"name": "@fgv/ts-utils",
"version": "0.3.4",
"version": "0.3.5",
"description": "Assorted Typescript Utilities",

@@ -5,0 +5,0 @@ "main": "index.js",

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc