@fgv/ts-json 1.9.4 → 1.9.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -256,4 +256,4 @@ class JsonEditor {
256
256
  }
257
257
  }
258
258
  exports.JsonEditor = JsonEditor;
259
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsonEditor.js","sourceRoot":"","sources":["../../src/jsonEditor/jsonEditor.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport {\n    ConditionalJsonEditorRule,\n    MultiValueJsonEditorRule,\n    ReferenceJsonEditorRule,\n    TemplatedJsonEditorRule,\n} from './rules';\nimport {\n    DetailedResult,\n    Result,\n    captureResult,\n    fail,\n    failWithDetail,\n    mapDetailedResults,\n    mapResults,\n    succeed,\n    succeedWithDetail,\n} from '@fgv/ts-utils';\n\nimport { JsonArray, JsonObject, JsonValue, isJsonArray, isJsonObject, isJsonPrimitive } from '../common';\nimport { JsonEditFailureReason, JsonEditorRule, JsonPropertyEditFailureReason } from './jsonEditorRule';\nimport { JsonEditorOptions, JsonEditorState, JsonEditorValidationOptions } from './jsonEditorState';\n\nimport { JsonContext } from '../jsonContext';\n\n/**\n * The JsonEditor can be used to edit JSON objects in place or to clone any JSON value,\n * applying a default context and optional set of editor rules that were supplied at\n * initialization.\n */\nexport class JsonEditor {\n    // eslint-disable-next-line no-use-before-define\n    protected static _default?: JsonEditor;\n\n    /**\n     * Full set of @see JsonEditorOptions in effect for this rule.\n     */\n    public options: JsonEditorOptions;\n    protected _rules: JsonEditorRule[];\n\n    protected constructor(options?: Partial<JsonEditorOptions>, rules?: JsonEditorRule[]) {\n        this.options = JsonEditor._getDefaultOptions(options).orThrow();\n        this._rules = rules || JsonEditor.getDefaultRules(this.options).orThrow();\n    }\n\n    /**\n     * Default singleton @see JsonEditor for simple use. Applies all rules\n     * but with no default context.\n     */\n    public static get default(): JsonEditor {\n        if (!JsonEditor._default) {\n            const rules = this.getDefaultRules().orDefault();\n            JsonEditor._default = new JsonEditor(undefined, rules);\n        }\n        return JsonEditor._default;\n    }\n\n    /**\n     * Constructs a new @see JsonEditor\n     * @param options Optional configuration an context for this editor\n     * @param rules Optional set of rules used by this editor\n     */\n    public static create(options?: Partial<JsonEditorOptions>, rules?: JsonEditorRule[]): Result<JsonEditor> {\n        return captureResult(() => new JsonEditor(options, rules));\n    }\n\n    /**\n     * Gets the default set of rules to be applied for a given set of options.\n     * By default, all available rules (templates, conditionals, multi-value and references)\n     * are applied.\n     * @param options The options used to initialize all rules\n     */\n    public static getDefaultRules(options?: JsonEditorOptions): Result<JsonEditorRule[]> {\n        return mapResults<JsonEditorRule>([\n            TemplatedJsonEditorRule.create(options),\n            ConditionalJsonEditorRule.create(options),\n            MultiValueJsonEditorRule.create(options),\n            ReferenceJsonEditorRule.create(options),\n        ]);\n    }\n\n    protected static _getDefaultOptions(options?: Partial<JsonEditorOptions>): Result<JsonEditorOptions> {\n        const context: JsonContext|undefined = options?.context;\n        let validation: JsonEditorValidationOptions|undefined = options?.validation;\n        if (validation === undefined) {\n            validation = {\n                onInvalidPropertyName: 'error',\n                onInvalidPropertyValue: 'error',\n                onUndefinedPropertyValue: 'ignore',\n            };\n        }\n        return succeed({ context, validation });\n    }\n\n    /**\n     * Merges a supplied source object into a supplied target, updating the target object.\n     * @param target The target object to be updated\n     * @param src The source object to be merged\n     * @param runtimeContext An optional @see JsonContext supplying variables and references\n     */\n    public mergeObjectInPlace(target: JsonObject, src: JsonObject, runtimeContext?: JsonContext): Result<JsonObject> {\n        const state = new JsonEditorState(this, this.options, runtimeContext);\n        return this._mergeObjectInPlace(target, src, state).onSuccess((merged) => {\n            return this._finalizeAndMerge(merged, state);\n        });\n    }\n\n    /**\n     * Merges multiple supplied source objects into a supplied target, updating the target\n     * object and using the default context supplied at creation time.\n     * @param target The target object to be updated\n     * @param srcObjects Objects to be merged into the target object, in the order supplied.\n     */\n    public mergeObjectsInPlace(target: JsonObject, srcObjects: JsonObject[]): Result<JsonObject> {\n        return this.mergeObjectsInPlaceWithContext(this.options.context, target, srcObjects);\n    }\n\n    /**\n     * Merges multiple supplied source objects into a supplied target, updating the target\n     * object and using an optional context supplied in the call.\n     * @param context An optional @see JsonContext supplying variables and references\n     * @param target The target object to be updated\n     * @param srcObjects Objects to be merged into the target object, in the order supplied.\n     */\n    public mergeObjectsInPlaceWithContext(context: JsonContext|undefined, base: JsonObject, srcObjects: JsonObject[]): Result<JsonObject> {\n        for (const src of srcObjects) {\n            const mergeResult = this.mergeObjectInPlace(base, src, context);\n            if (mergeResult.isFailure()) {\n                return mergeResult.withFailureDetail('error');\n            }\n        }\n        return succeedWithDetail(base);\n    }\n\n    /**\n     * Deep clones a supplied JSON value, applying all editor rules and a default\n     * or optionally supplied context\n     * @param src The @see JsonValue to be cloned\n     * @param context An optional @see JsonContext supplying variables and references\n     */\n    public clone(src: JsonValue, context?: JsonContext): DetailedResult<JsonValue, JsonEditFailureReason> {\n        const state = new JsonEditorState(this, this.options, context);\n        let value = src;\n        let valueResult = this._editValue(src, state);\n\n        while (valueResult.isSuccess()) {\n            value = valueResult.value;\n            valueResult = this._editValue(value, state);\n        }\n\n        if ((valueResult.detail === 'error') || (valueResult.detail === 'ignore')) {\n            return valueResult;\n        }\n\n        if (isJsonPrimitive(value) || (value === null)) {\n            return succeedWithDetail(value, 'edited');\n        }\n        else if (isJsonObject(value)) {\n            return this.mergeObjectInPlace({}, value, state.context).withFailureDetail('error');\n        }\n        else if (isJsonArray(value)) {\n            return this._cloneArray(value, state.context);\n        }\n        else if (value === undefined) {\n            return state.failValidation('undefinedPropertyValue');\n        }\n        return state.failValidation('invalidPropertyValue', `Cannot convert invalid JSON: \"${JSON.stringify(value)}\"`);\n    }\n\n    protected _mergeObjectInPlace(target: JsonObject, src: JsonObject, state: JsonEditorState): Result<JsonObject> {\n        for (const key in src) {\n            if (src.hasOwnProperty(key)) {\n                const propResult = this._editProperty(key, src[key], state);\n                if (propResult.isSuccess()) {\n                    if (propResult.detail === 'deferred') {\n                        state.defer(propResult.value);\n                    }\n                    else {\n                        const mergeResult = this._mergeObjectInPlace(target, propResult.value, state);\n                        if (mergeResult.isFailure()) {\n                            return mergeResult;\n                        }\n                    }\n                }\n                else if (propResult.detail === 'inapplicable') {\n                    const valueResult = this.clone(src[key], state.context).onSuccess((cloned) => {\n                        return this._mergeClonedProperty(target, key, cloned, state);\n                    });\n\n                    if (valueResult.isFailure() && (valueResult.detail === 'error')) {\n                        return fail(`${key}: ${valueResult.message}`);\n                    }\n                }\n                else if (propResult.detail !== 'ignore') {\n                    return fail(`${key}: ${propResult.message}`);\n                }\n            }\n            else {\n                return fail(`${key}: Cannot merge inherited properties`);\n            }\n        }\n        return succeed(target);\n    }\n\n    protected _cloneArray(src: JsonArray, context?: JsonContext): DetailedResult<JsonArray, JsonEditFailureReason> {\n        const results = src.map((v) => {\n            return this.clone(v, context);\n        });\n\n        return mapDetailedResults<JsonValue, JsonEditFailureReason>(results, ['ignore']).onSuccess((converted) => {\n            return succeed(converted);\n        }).withFailureDetail('error');\n    }\n\n    protected _mergeClonedProperty(target: JsonObject, key: string, newValue: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        const existing = target[key];\n\n        // merge is called right after clone so this should never happen\n        // since clone itself will have failed\n        // istanbul ignore else\n        if (isJsonPrimitive(newValue)) {\n            target[key] = newValue;\n            return succeedWithDetail(newValue, 'edited');\n        }\n        else if (isJsonObject(newValue)) {\n            if (isJsonObject(existing)) {\n                return this.mergeObjectInPlace(existing, newValue, state.context).withFailureDetail('error');\n            }\n            target[key] = newValue;\n            return succeedWithDetail(newValue, 'edited');\n        }\n        else if (isJsonArray(newValue)) {\n            if (isJsonArray(existing)) {\n                target[key] = existing.concat(...newValue);\n                return succeedWithDetail(target[key], 'edited');\n            }\n            target[key] = newValue;\n            return succeedWithDetail(newValue, 'edited');\n        }\n        else {\n            return failWithDetail(`Invalid JSON: ${JSON.stringify(newValue)}`, 'error');\n        }\n    }\n\n    protected _editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        for (const rule of this._rules) {\n            const ruleResult = rule.editProperty(key, value, state);\n            if (ruleResult.isSuccess() || (ruleResult.detail !== 'inapplicable')) {\n                return ruleResult;\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _editValue(value: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        for (const rule of this._rules) {\n            const ruleResult = rule.editValue(value, state);\n            if (ruleResult.isSuccess() || (ruleResult.detail !== 'inapplicable')) {\n                return ruleResult;\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _finalizeAndMerge(target: JsonObject, state: JsonEditorState): DetailedResult<JsonObject, JsonEditFailureReason> {\n        const deferred = state.deferred;\n        if (deferred.length > 0) {\n            for (const rule of this._rules) {\n                const ruleResult = rule.finalizeProperties(deferred, state);\n                if (ruleResult.isSuccess()) {\n                    return this.mergeObjectsInPlaceWithContext(state.context, target, ruleResult.value).withFailureDetail('error');\n                }\n                else if (ruleResult.detail === 'ignore') {\n                    succeedWithDetail(target, 'edited');\n                }\n                else if (ruleResult.detail !== 'inapplicable') {\n                    return failWithDetail(ruleResult.message, ruleResult.detail);\n                }\n            }\n        }\n        return succeedWithDetail(target, 'edited');\n    }\n}\n\n"]}
259
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsonEditor.js","sourceRoot":"","sources":["../../src/jsonEditor/jsonEditor.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;;;AAEH,mCAKiB;AACjB,4CAUuB;AAGvB,sCAAyG;AAIzG,uDAAoD;AAEpD;;;;GAIG;AACH,MAAa,UAAU;IAUnB,YAAsB,OAAoC,EAAE,KAAwB;QAChF,IAAI,CAAC,OAAO,GAAG,UAAU,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,CAAC;QAChE,IAAI,CAAC,MAAM,GAAG,KAAK,IAAI,UAAU,CAAC,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,CAAC;IAC9E,CAAC;IAED;;;OAGG;IACI,MAAM,KAAK,OAAO;QACrB,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE;YACtB,MAAM,KAAK,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,SAAS,EAAE,CAAC;YACjD,UAAU,CAAC,QAAQ,GAAG,IAAI,UAAU,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;SAC1D;QACD,OAAO,UAAU,CAAC,QAAQ,CAAC;IAC/B,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,MAAM,CAAC,OAAoC,EAAE,KAAwB;QAC/E,OAAO,IAAA,wBAAa,EAAC,GAAG,EAAE,CAAC,IAAI,UAAU,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC;IAC/D,CAAC;IAED;;;;;OAKG;IACI,MAAM,CAAC,eAAe,CAAC,OAA2B;QACrD,OAAO,IAAA,qBAAU,EAAiB;YAC9B,+BAAuB,CAAC,MAAM,CAAC,OAAO,CAAC;YACvC,iCAAyB,CAAC,MAAM,CAAC,OAAO,CAAC;YACzC,gCAAwB,CAAC,MAAM,CAAC,OAAO,CAAC;YACxC,+BAAuB,CAAC,MAAM,CAAC,OAAO,CAAC;SAC1C,CAAC,CAAC;IACP,CAAC;IAES,MAAM,CAAC,kBAAkB,CAAC,OAAoC;QACpE,MAAM,OAAO,GAA0B,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,OAAO,CAAC;QACxD,IAAI,UAAU,GAA0C,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,UAAU,CAAC;QAC5E,IAAI,UAAU,KAAK,SAAS,EAAE;YAC1B,UAAU,GAAG;gBACT,qBAAqB,EAAE,OAAO;gBAC9B,sBAAsB,EAAE,OAAO;gBAC/B,wBAAwB,EAAE,QAAQ;aACrC,CAAC;SACL;QACD,OAAO,IAAA,kBAAO,EAAC,EAAE,OAAO,EAAE,UAAU,EAAE,CAAC,CAAC;IAC5C,CAAC;IAED;;;;;OAKG;IACI,kBAAkB,CAAC,MAAkB,EAAE,GAAe,EAAE,cAA4B;QACvF,MAAM,KAAK,GAAG,IAAI,iCAAe,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;QACtE,OAAO,IAAI,CAAC,mBAAmB,CAAC,MAAM,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;YACrE,OAAO,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;;OAKG;IACI,mBAAmB,CAAC,MAAkB,EAAE,UAAwB;QACnE,OAAO,IAAI,CAAC,8BAA8B,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAM,EAAE,UAAU,CAAC,CAAC;IACzF,CAAC;IAED;;;;;;OAMG;IACI,8BAA8B,CAAC,OAA8B,EAAE,IAAgB,EAAE,UAAwB;QAC5G,KAAK,MAAM,GAAG,IAAI,UAAU,EAAE;YAC1B,MAAM,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,GAAG,EAAE,OAAO,CAAC,CAAC;YAChE,IAAI,WAAW,CAAC,SAAS,EAAE,EAAE;gBACzB,OAAO,WAAW,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;aACjD;SACJ;QACD,OAAO,IAAA,4BAAiB,EAAC,IAAI,CAAC,CAAC;IACnC,CAAC;IAED;;;;;OAKG;IACI,KAAK,CAAC,GAAc,EAAE,OAAqB;QAC9C,MAAM,KAAK,GAAG,IAAI,iCAAe,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC/D,IAAI,KAAK,GAAG,GAAG,CAAC;QAChB,IAAI,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QAE9C,OAAO,WAAW,CAAC,SAAS,EAAE,EAAE;YAC5B,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC;YAC1B,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;SAC/C;QAED,IAAI,CAAC,WAAW,CAAC,MAAM,KAAK,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,KAAK,QAAQ,CAAC,EAAE;YACvE,OAAO,WAAW,CAAC;SACtB;QAED,IAAI,IAAA,wBAAe,EAAC,KAAK,CAAC,IAAI,CAAC,KAAK,KAAK,IAAI,CAAC,EAAE;YAC5C,OAAO,IAAA,4BAAiB,EAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;SAC7C;aACI,IAAI,IAAA,qBAAY,EAAC,KAAK,CAAC,EAAE;YAC1B,OAAO,IAAI,CAAC,kBAAkB,CAAC,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;SACvF;aACI,IAAI,IAAA,oBAAW,EAAC,KAAK,CAAC,EAAE;YACzB,OAAO,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;SACjD;aACI,IAAI,KAAK,KAAK,SAAS,EAAE;YAC1B,OAAO,KAAK,CAAC,cAAc,CAAC,wBAAwB,CAAC,CAAC;SACzD;QACD,OAAO,KAAK,CAAC,cAAc,CAAC,sBAAsB,EAAE,iCAAiC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACnH,CAAC;IAES,mBAAmB,CAAC,MAAkB,EAAE,GAAe,EAAE,KAAsB;QACrF,KAAK,MAAM,GAAG,IAAI,GAAG,EAAE;YACnB,IAAI,GAAG,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE;gBACzB,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,EAAE,GAAG,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;gBAC5D,IAAI,UAAU,CAAC,SAAS,EAAE,EAAE;oBACxB,IAAI,UAAU,CAAC,MAAM,KAAK,UAAU,EAAE;wBAClC,KAAK,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;qBACjC;yBACI;wBACD,MAAM,WAAW,GAAG,IAAI,CAAC,mBAAmB,CAAC,MAAM,EAAE,UAAU,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;wBAC9E,IAAI,WAAW,CAAC,SAAS,EAAE,EAAE;4BACzB,OAAO,WAAW,CAAC;yBACtB;qBACJ;iBACJ;qBACI,IAAI,UAAU,CAAC,MAAM,KAAK,cAAc,EAAE;oBAC3C,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;wBACzE,OAAO,IAAI,CAAC,oBAAoB,CAAC,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC;oBACjE,CAAC,CAAC,CAAC;oBAEH,IAAI,WAAW,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,KAAK,OAAO,CAAC,EAAE;wBAC7D,OAAO,IAAA,eAAI,EAAC,GAAG,GAAG,KAAK,WAAW,CAAC,OAAO,EAAE,CAAC,CAAC;qBACjD;iBACJ;qBACI,IAAI,UAAU,CAAC,MAAM,KAAK,QAAQ,EAAE;oBACrC,OAAO,IAAA,eAAI,EAAC,GAAG,GAAG,KAAK,UAAU,CAAC,OAAO,EAAE,CAAC,CAAC;iBAChD;aACJ;iBACI;gBACD,OAAO,IAAA,eAAI,EAAC,GAAG,GAAG,qCAAqC,CAAC,CAAC;aAC5D;SACJ;QACD,OAAO,IAAA,kBAAO,EAAC,MAAM,CAAC,CAAC;IAC3B,CAAC;IAES,WAAW,CAAC,GAAc,EAAE,OAAqB;QACvD,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;YAC1B,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,OAAO,IAAA,6BAAkB,EAAmC,OAAO,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,SAAS,EAAE,EAAE;YACrG,OAAO,IAAA,kBAAO,EAAC,SAAS,CAAC,CAAC;QAC9B,CAAC,CAAC,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;IAClC,CAAC;IAES,oBAAoB,CAAC,MAAkB,EAAE,GAAW,EAAE,QAAmB,EAAE,KAAsB;QACvG,MAAM,QAAQ,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;QAE7B,gEAAgE;QAChE,sCAAsC;QACtC,uBAAuB;QACvB,IAAI,IAAA,wBAAe,EAAC,QAAQ,CAAC,EAAE;YAC3B,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC;YACvB,OAAO,IAAA,4BAAiB,EAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;SAChD;aACI,IAAI,IAAA,qBAAY,EAAC,QAAQ,CAAC,EAAE;YAC7B,IAAI,IAAA,qBAAY,EAAC,QAAQ,CAAC,EAAE;gBACxB,OAAO,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,QAAQ,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;aAChG;YACD,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC;YACvB,OAAO,IAAA,4BAAiB,EAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;SAChD;aACI,IAAI,IAAA,oBAAW,EAAC,QAAQ,CAAC,EAAE;YAC5B,IAAI,IAAA,oBAAW,EAAC,QAAQ,CAAC,EAAE;gBACvB,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC,MAAM,CAAC,GAAG,QAAQ,CAAC,CAAC;gBAC3C,OAAO,IAAA,4BAAiB,EAAC,MAAM,CAAC,GAAG,CAAC,EAAE,QAAQ,CAAC,CAAC;aACnD;YACD,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC;YACvB,OAAO,IAAA,4BAAiB,EAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;SAChD;aACI;YACD,OAAO,IAAA,yBAAc,EAAC,iBAAiB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE,EAAE,OAAO,CAAC,CAAC;SAC/E;IACL,CAAC;IAES,aAAa,CAAC,GAAW,EAAE,KAAgB,EAAE,KAAsB;QACzE,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,MAAM,EAAE;YAC5B,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;YACxD,IAAI,UAAU,CAAC,SAAS,EAAE,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,cAAc,CAAC,EAAE;gBAClE,OAAO,UAAU,CAAC;aACrB;SACJ;QACD,OAAO,IAAA,yBAAc,EAAC,cAAc,EAAE,cAAc,CAAC,CAAC;IAC1D,CAAC;IAES,UAAU,CAAC,KAAgB,EAAE,KAAsB;QACzD,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,MAAM,EAAE;YAC5B,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YAChD,IAAI,UAAU,CAAC,SAAS,EAAE,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,cAAc,CAAC,EAAE;gBAClE,OAAO,UAAU,CAAC;aACrB;SACJ;QACD,OAAO,IAAA,yBAAc,EAAC,cAAc,EAAE,cAAc,CAAC,CAAC;IAC1D,CAAC;IAES,iBAAiB,CAAC,MAAkB,EAAE,KAAsB;QAClE,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;QAChC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;YACrB,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,MAAM,EAAE;gBAC5B,MAAM,UAAU,GAAG,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;gBAC5D,IAAI,UAAU,CAAC,SAAS,EAAE,EAAE;oBACxB,OAAO,IAAI,CAAC,8BAA8B,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;iBAClH;qBACI,IAAI,UAAU,CAAC,MAAM,KAAK,QAAQ,EAAE;oBACrC,IAAA,4BAAiB,EAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;iBACvC;qBACI,IAAI,UAAU,CAAC,MAAM,KAAK,cAAc,EAAE;oBAC3C,OAAO,IAAA,yBAAc,EAAC,UAAU,CAAC,OAAO,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC;iBAChE;aACJ;SACJ;QACD,OAAO,IAAA,4BAAiB,EAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;IAC/C,CAAC;CACJ;AA5PD,gCA4PC","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport {\n    ConditionalJsonEditorRule,\n    MultiValueJsonEditorRule,\n    ReferenceJsonEditorRule,\n    TemplatedJsonEditorRule,\n} from './rules';\nimport {\n    DetailedResult,\n    Result,\n    captureResult,\n    fail,\n    failWithDetail,\n    mapDetailedResults,\n    mapResults,\n    succeed,\n    succeedWithDetail,\n} from '@fgv/ts-utils';\n\nimport { IJsonCloneEditor, JsonEditFailureReason, JsonEditorOptions, JsonEditorValidationOptions, JsonPropertyEditFailureReason } from './common';\nimport { JsonArray, JsonObject, JsonValue, isJsonArray, isJsonObject, isJsonPrimitive } from '../common';\n\nimport { JsonContext } from '../jsonContext';\nimport { JsonEditorRule } from './jsonEditorRule';\nimport { JsonEditorState } from './jsonEditorState';\n\n/**\n * The JsonEditor can be used to edit JSON objects in place or to clone any JSON value,\n * applying a default context and optional set of editor rules that were supplied at\n * initialization.\n */\nexport class JsonEditor implements IJsonCloneEditor {\n    // eslint-disable-next-line no-use-before-define\n    protected static _default?: JsonEditor;\n\n    /**\n     * Full set of @see JsonEditorOptions in effect for this rule.\n     */\n    public options: JsonEditorOptions;\n    protected _rules: JsonEditorRule[];\n\n    protected constructor(options?: Partial<JsonEditorOptions>, rules?: JsonEditorRule[]) {\n        this.options = JsonEditor._getDefaultOptions(options).orThrow();\n        this._rules = rules || JsonEditor.getDefaultRules(this.options).orThrow();\n    }\n\n    /**\n     * Default singleton @see JsonEditor for simple use. Applies all rules\n     * but with no default context.\n     */\n    public static get default(): JsonEditor {\n        if (!JsonEditor._default) {\n            const rules = this.getDefaultRules().orDefault();\n            JsonEditor._default = new JsonEditor(undefined, rules);\n        }\n        return JsonEditor._default;\n    }\n\n    /**\n     * Constructs a new @see JsonEditor\n     * @param options Optional configuration an context for this editor\n     * @param rules Optional set of rules used by this editor\n     */\n    public static create(options?: Partial<JsonEditorOptions>, rules?: JsonEditorRule[]): Result<JsonEditor> {\n        return captureResult(() => new JsonEditor(options, rules));\n    }\n\n    /**\n     * Gets the default set of rules to be applied for a given set of options.\n     * By default, all available rules (templates, conditionals, multi-value and references)\n     * are applied.\n     * @param options The options used to initialize all rules\n     */\n    public static getDefaultRules(options?: JsonEditorOptions): Result<JsonEditorRule[]> {\n        return mapResults<JsonEditorRule>([\n            TemplatedJsonEditorRule.create(options),\n            ConditionalJsonEditorRule.create(options),\n            MultiValueJsonEditorRule.create(options),\n            ReferenceJsonEditorRule.create(options),\n        ]);\n    }\n\n    protected static _getDefaultOptions(options?: Partial<JsonEditorOptions>): Result<JsonEditorOptions> {\n        const context: JsonContext|undefined = options?.context;\n        let validation: JsonEditorValidationOptions|undefined = options?.validation;\n        if (validation === undefined) {\n            validation = {\n                onInvalidPropertyName: 'error',\n                onInvalidPropertyValue: 'error',\n                onUndefinedPropertyValue: 'ignore',\n            };\n        }\n        return succeed({ context, validation });\n    }\n\n    /**\n     * Merges a supplied source object into a supplied target, updating the target object.\n     * @param target The target object to be updated\n     * @param src The source object to be merged\n     * @param runtimeContext An optional @see JsonContext supplying variables and references\n     */\n    public mergeObjectInPlace(target: JsonObject, src: JsonObject, runtimeContext?: JsonContext): Result<JsonObject> {\n        const state = new JsonEditorState(this, this.options, runtimeContext);\n        return this._mergeObjectInPlace(target, src, state).onSuccess((merged) => {\n            return this._finalizeAndMerge(merged, state);\n        });\n    }\n\n    /**\n     * Merges multiple supplied source objects into a supplied target, updating the target\n     * object and using the default context supplied at creation time.\n     * @param target The target object to be updated\n     * @param srcObjects Objects to be merged into the target object, in the order supplied.\n     */\n    public mergeObjectsInPlace(target: JsonObject, srcObjects: JsonObject[]): Result<JsonObject> {\n        return this.mergeObjectsInPlaceWithContext(this.options.context, target, srcObjects);\n    }\n\n    /**\n     * Merges multiple supplied source objects into a supplied target, updating the target\n     * object and using an optional context supplied in the call.\n     * @param context An optional @see JsonContext supplying variables and references\n     * @param target The target object to be updated\n     * @param srcObjects Objects to be merged into the target object, in the order supplied.\n     */\n    public mergeObjectsInPlaceWithContext(context: JsonContext|undefined, base: JsonObject, srcObjects: JsonObject[]): Result<JsonObject> {\n        for (const src of srcObjects) {\n            const mergeResult = this.mergeObjectInPlace(base, src, context);\n            if (mergeResult.isFailure()) {\n                return mergeResult.withFailureDetail('error');\n            }\n        }\n        return succeedWithDetail(base);\n    }\n\n    /**\n     * Deep clones a supplied JSON value, applying all editor rules and a default\n     * or optionally supplied context\n     * @param src The @see JsonValue to be cloned\n     * @param context An optional @see JsonContext supplying variables and references\n     */\n    public clone(src: JsonValue, context?: JsonContext): DetailedResult<JsonValue, JsonEditFailureReason> {\n        const state = new JsonEditorState(this, this.options, context);\n        let value = src;\n        let valueResult = this._editValue(src, state);\n\n        while (valueResult.isSuccess()) {\n            value = valueResult.value;\n            valueResult = this._editValue(value, state);\n        }\n\n        if ((valueResult.detail === 'error') || (valueResult.detail === 'ignore')) {\n            return valueResult;\n        }\n\n        if (isJsonPrimitive(value) || (value === null)) {\n            return succeedWithDetail(value, 'edited');\n        }\n        else if (isJsonObject(value)) {\n            return this.mergeObjectInPlace({}, value, state.context).withFailureDetail('error');\n        }\n        else if (isJsonArray(value)) {\n            return this._cloneArray(value, state.context);\n        }\n        else if (value === undefined) {\n            return state.failValidation('undefinedPropertyValue');\n        }\n        return state.failValidation('invalidPropertyValue', `Cannot convert invalid JSON: \"${JSON.stringify(value)}\"`);\n    }\n\n    protected _mergeObjectInPlace(target: JsonObject, src: JsonObject, state: JsonEditorState): Result<JsonObject> {\n        for (const key in src) {\n            if (src.hasOwnProperty(key)) {\n                const propResult = this._editProperty(key, src[key], state);\n                if (propResult.isSuccess()) {\n                    if (propResult.detail === 'deferred') {\n                        state.defer(propResult.value);\n                    }\n                    else {\n                        const mergeResult = this._mergeObjectInPlace(target, propResult.value, state);\n                        if (mergeResult.isFailure()) {\n                            return mergeResult;\n                        }\n                    }\n                }\n                else if (propResult.detail === 'inapplicable') {\n                    const valueResult = this.clone(src[key], state.context).onSuccess((cloned) => {\n                        return this._mergeClonedProperty(target, key, cloned, state);\n                    });\n\n                    if (valueResult.isFailure() && (valueResult.detail === 'error')) {\n                        return fail(`${key}: ${valueResult.message}`);\n                    }\n                }\n                else if (propResult.detail !== 'ignore') {\n                    return fail(`${key}: ${propResult.message}`);\n                }\n            }\n            else {\n                return fail(`${key}: Cannot merge inherited properties`);\n            }\n        }\n        return succeed(target);\n    }\n\n    protected _cloneArray(src: JsonArray, context?: JsonContext): DetailedResult<JsonArray, JsonEditFailureReason> {\n        const results = src.map((v) => {\n            return this.clone(v, context);\n        });\n\n        return mapDetailedResults<JsonValue, JsonEditFailureReason>(results, ['ignore']).onSuccess((converted) => {\n            return succeed(converted);\n        }).withFailureDetail('error');\n    }\n\n    protected _mergeClonedProperty(target: JsonObject, key: string, newValue: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        const existing = target[key];\n\n        // merge is called right after clone so this should never happen\n        // since clone itself will have failed\n        // istanbul ignore else\n        if (isJsonPrimitive(newValue)) {\n            target[key] = newValue;\n            return succeedWithDetail(newValue, 'edited');\n        }\n        else if (isJsonObject(newValue)) {\n            if (isJsonObject(existing)) {\n                return this.mergeObjectInPlace(existing, newValue, state.context).withFailureDetail('error');\n            }\n            target[key] = newValue;\n            return succeedWithDetail(newValue, 'edited');\n        }\n        else if (isJsonArray(newValue)) {\n            if (isJsonArray(existing)) {\n                target[key] = existing.concat(...newValue);\n                return succeedWithDetail(target[key], 'edited');\n            }\n            target[key] = newValue;\n            return succeedWithDetail(newValue, 'edited');\n        }\n        else {\n            return failWithDetail(`Invalid JSON: ${JSON.stringify(newValue)}`, 'error');\n        }\n    }\n\n    protected _editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        for (const rule of this._rules) {\n            const ruleResult = rule.editProperty(key, value, state);\n            if (ruleResult.isSuccess() || (ruleResult.detail !== 'inapplicable')) {\n                return ruleResult;\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _editValue(value: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        for (const rule of this._rules) {\n            const ruleResult = rule.editValue(value, state);\n            if (ruleResult.isSuccess() || (ruleResult.detail !== 'inapplicable')) {\n                return ruleResult;\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _finalizeAndMerge(target: JsonObject, state: JsonEditorState): DetailedResult<JsonObject, JsonEditFailureReason> {\n        const deferred = state.deferred;\n        if (deferred.length > 0) {\n            for (const rule of this._rules) {\n                const ruleResult = rule.finalizeProperties(deferred, state);\n                if (ruleResult.isSuccess()) {\n                    return this.mergeObjectsInPlaceWithContext(state.context, target, ruleResult.value).withFailureDetail('error');\n                }\n                else if (ruleResult.detail === 'ignore') {\n                    succeedWithDetail(target, 'edited');\n                }\n                else if (ruleResult.detail !== 'inapplicable') {\n                    return failWithDetail(ruleResult.message, ruleResult.detail);\n                }\n            }\n        }\n        return succeedWithDetail(target, 'edited');\n    }\n}\n\n"]}
@@ -1,8 +1,7 @@
1
1
  import { DetailedResult } from '@fgv/ts-utils';
2
+ import { JsonEditFailureReason, JsonPropertyEditFailureReason } from './common';
2
3
  import { JsonObject, JsonValue } from '../common';
3
4
  import { JsonEditorState } from './jsonEditorState';
4
- export type JsonEditFailureReason = 'ignore' | 'inapplicable' | 'edited' | 'error';
5
- export type JsonPropertyEditFailureReason = JsonEditFailureReason | 'deferred';
6
5
  export interface JsonEditorRule {
7
6
  /**
8
7
  * Called by a JSON editor to possibly edit one of the properties being merged into a target object.
@@ -41,4 +41,4 @@ class JsonEditorRuleBase {
41
41
  }
42
42
  }
43
43
  exports.JsonEditorRuleBase = JsonEditorRuleBase;
44
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoianNvbkVkaXRvclJ1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvanNvbkVkaXRvci9qc29uRWRpdG9yUnVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHOzs7QUFFSCw0Q0FBK0Q7QUE4Qy9EOzs7R0FHRztBQUNILE1BQWEsa0JBQWtCO0lBQzNCLHVCQUF1QjtJQUNoQixZQUFZLENBQUMsSUFBWSxFQUFFLE1BQWlCLEVBQUUsTUFBdUI7UUFDeEUsdUJBQXVCO1FBQ3ZCLE9BQU8sSUFBQSx5QkFBYyxFQUFDLGNBQWMsRUFBRSxjQUFjLENBQUMsQ0FBQztJQUMxRCxDQUFDO0lBRU0sU0FBUyxDQUFDLE1BQWlCLEVBQUUsTUFBdUI7UUFDdkQsT0FBTyxJQUFBLHlCQUFjLEVBQUMsY0FBYyxFQUFFLGNBQWMsQ0FBQyxDQUFDO0lBQzFELENBQUM7SUFFTSxrQkFBa0IsQ0FBQyxTQUF1QixFQUFFLE1BQXVCO1FBQ3RFLE9BQU8sSUFBQSx5QkFBYyxFQUFDLGNBQWMsRUFBRSxjQUFjLENBQUMsQ0FBQztJQUMxRCxDQUFDO0NBQ0o7QUFkRCxnREFjQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjAgRXJpayBGb3J0dW5lXG4gKlxuICogUGVybWlzc2lvbiBpcyBoZXJlYnkgZ3JhbnRlZCwgZnJlZSBvZiBjaGFyZ2UsIHRvIGFueSBwZXJzb24gb2J0YWluaW5nIGEgY29weVxuICogb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzICh0aGUgXCJTb2Z0d2FyZVwiKSwgdG8gZGVhbFxuICogaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0c1xuICogdG8gdXNlLCBjb3B5LCBtb2RpZnksIG1lcmdlLCBwdWJsaXNoLCBkaXN0cmlidXRlLCBzdWJsaWNlbnNlLCBhbmQvb3Igc2VsbFxuICogY29waWVzIG9mIHRoZSBTb2Z0d2FyZSwgYW5kIHRvIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIFNvZnR3YXJlIGlzXG4gKiBmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOlxuICpcbiAqIFRoZSBhYm92ZSBjb3B5cmlnaHQgbm90aWNlIGFuZCB0aGlzIHBlcm1pc3Npb24gbm90aWNlIHNoYWxsIGJlIGluY2x1ZGVkIGluIGFsbFxuICogY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS5cbiAqXG4gKiBUSEUgU09GVFdBUkUgSVMgUFJPVklERUQgXCJBUyBJU1wiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELCBFWFBSRVNTIE9SXG4gKiBJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GIE1FUkNIQU5UQUJJTElUWSxcbiAqIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFORCBOT05JTkZSSU5HRU1FTlQuIElOIE5PIEVWRU5UIFNIQUxMIFRIRVxuICogQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWSBDTEFJTSwgREFNQUdFUyBPUiBPVEhFUlxuICogTElBQklMSVRZLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwgVE9SVCBPUiBPVEhFUldJU0UsIEFSSVNJTkcgRlJPTSxcbiAqIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFIFNPRlRXQVJFIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4gVEhFXG4gKiBTT0ZUV0FSRS5cbiAqL1xuXG5pbXBvcnQgeyBEZXRhaWxlZFJlc3VsdCwgZmFpbFdpdGhEZXRhaWwgfSBmcm9tICdAZmd2L3RzLXV0aWxzJztcbmltcG9ydCB7IEpzb25PYmplY3QsIEpzb25WYWx1ZSB9IGZyb20gJy4uL2NvbW1vbic7XG5cbmltcG9ydCB7IEpzb25FZGl0b3JTdGF0ZSB9IGZyb20gJy4vanNvbkVkaXRvclN0YXRlJztcblxuZXhwb3J0IHR5cGUgSnNvbkVkaXRGYWlsdXJlUmVhc29uID0gJ2lnbm9yZSd8J2luYXBwbGljYWJsZSd8J2VkaXRlZCd8J2Vycm9yJztcbmV4cG9ydCB0eXBlIEpzb25Qcm9wZXJ0eUVkaXRGYWlsdXJlUmVhc29uID0gSnNvbkVkaXRGYWlsdXJlUmVhc29ufCdkZWZlcnJlZCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgSnNvbkVkaXRvclJ1bGUge1xuICAgIC8qKlxuICAgICAqIENhbGxlZCBieSBhIEpTT04gZWRpdG9yIHRvIHBvc3NpYmx5IGVkaXQgb25lIG9mIHRoZSBwcm9wZXJ0aWVzIGJlaW5nIG1lcmdlZCBpbnRvIGEgdGFyZ2V0IG9iamVjdC5cbiAgICAgKiBAcGFyYW0ga2V5IFRoZSBrZXkgb2YgdGhlIHByb3BlcnR5IHRvIGJlIGVkaXRlZFxuICAgICAqIEBwYXJhbSB2YWx1ZSBUaGUgdmFsdWUgb2YgdGhlIHByb3BlcnR5IHRvIGJlIGVkaXRlZFxuICAgICAqIEBwYXJhbSBzdGF0ZSBFZGl0b3Igc3RhdGUgd2hpY2ggYXBwbGllcyB0byB0aGUgZWRpdFxuICAgICAqIEByZXR1cm5zIElmIHRoZSBwcm9wZXJ0eSB3YXMgZWRpdGVkLCByZXR1cm5zIFN1Y2Nlc3Mgd2l0aCBhIEpTT04gb2JqZWN0IGNvbnRhaW5pbmcgdGhlIGVkaXRlZCByZXN1bHRzXG4gICAgICogYW5kIHdpdGggZGV0YWlsICdlZGl0ZWQnLiBJZiB0aGlzIHByb3BlcnR5IHNob3VsZCBiZSBkZWZlcnJlZCBmb3IgbGF0ZXIgY29uc2lkZXJhdGlvbiBvciBtZXJnLCBTdWNjZWVkc1xuICAgICAqIHdpdGggZGV0YWlsICdkZWZlcnJlZCcgYW5kIGFuIEpzb25PYmplY3QgdG8gYmUgZmluYWxpemVkLiAgSWYgdGhlIHJ1bGUgZG9lcyBub3QgYWZmZWN0IHRoaXMgcHJvcGVydHksXG4gICAgICogZmFpbHMgd2l0aCBkZXRhaWwgJ2luYXBwbGljYWJsZScuIElmIGFuIGVycm9yIG9jY3VycmVkIHdoaWxlIHByb2Nlc3NpbmcgdGhlIGVycm9yLCByZXR1cm5zIEZhaWx1cmUgd2l0aFxuICAgICAqIGRldGFpbCAnZXJyb3InLlxuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby11c2UtYmVmb3JlLWRlZmluZVxuICAgIGVkaXRQcm9wZXJ0eShrZXk6IHN0cmluZywgdmFsdWU6IEpzb25WYWx1ZSwgc3RhdGU6IEpzb25FZGl0b3JTdGF0ZSk6IERldGFpbGVkUmVzdWx0PEpzb25PYmplY3QsIEpzb25Qcm9wZXJ0eUVkaXRGYWlsdXJlUmVhc29uPjtcblxuICAgIC8qKlxuICAgICAqIENhbGxlZCBieSBhIEpTT04gZWRpdG9yIHRvIHBvc3NpYmx5IGVkaXQgYSBwcm9wZXJ0eSB2YWx1ZSBvciBhcnJheSBlbGVtZW50XG4gICAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSB0byBiZSBlZGl0ZWRcbiAgICAgKiBAcGFyYW0gc3RhdGUgRWRpdG9yIHN0YXRlIHdoaWNoIGFwcGxpZXMgdG8gdGhlIGVkaXRcbiAgICAgKiBAcmV0dXJucyBSZXR1cm5zIHN1Y2Nlc3Mgd2l0aCB0aGUgSnNvblZhbHVlIHRvIGJlIGluc2VydGVkLCB3aXRoIGRldGFpbCAnZWRpdGVkJyBpZiB0aGUgdmFsdWUgd2FzXG4gICAgICogZWRpdGVkLiAgUmV0dXJucyBmYWlsdXJlIHdpdGggJ2luYXBwbGljYWJsZScgaWYgdGhlIHJ1bGUgZG9lcyBub3QgYWZmZWN0IHRoaXMgdmFsdWUuICBGYWlscyB3aXRoXG4gICAgICogZGV0YWlsICdpZ25vcmUnIGlmIHRoZSB2YWx1ZSBpcyB0byBiZSBpZ25vcmVkLCBvciB3aXRoICdlcnJvcicgaWYgYW4gZXJyb3Igb2NjdXJzLlxuICAgICAqL1xuICAgIGVkaXRWYWx1ZSh2YWx1ZTogSnNvblZhbHVlLCBzdGF0ZTogSnNvbkVkaXRvclN0YXRlKTogRGV0YWlsZWRSZXN1bHQ8SnNvblZhbHVlLCBKc29uRWRpdEZhaWx1cmVSZWFzb24+O1xuXG4gICAgLyoqXG4gICAgICogQ2FsbGVkIGZvciBlYWNoIHJ1bGUgYWZ0ZXIgYWxsIHByb3BlcnRpZXMgaGF2ZSBiZWVuIG1lcmdlZC4gIEFueSBwcm9wZXJ0aWVzIHRoYXQgd2VyZSBkZWZlcnJlZFxuICAgICAqIGR1cmluZyB0aGUgaW5pdGlhbCBlZGl0IHBhc3MgYXJlIHN1cHBsaWVkIGFzIGlucHV0LlxuICAgICAqIEBwYXJhbSBkZWZlcnJlZCBBbnkgb2JqZWN0cyB0aGF0IHdlcmUgZGVmZXJyZWQgZHVyaW5nIHRoZSBmaXJzdCBlZGl0IHBhc3NcbiAgICAgKiBAcGFyYW0gc3RhdGUgRWRpdG9yIHN0YXRlIHdoaWNoIGFwcGxpZXMgdG8gdGhlIGVkaXRcbiAgICAgKiBAcmV0dXJucyBPbiBzdWNjZXNzZnVsIHJldHVybiwgYW55IHJldHVybmVkIG9iamVjdHMgYXJlIG1lcmdlZCBpbiBvcmRlciBhbmQgZmluYWxpemF0aW9uXG4gICAgICogaXMgc3RvcHBlZC4gRmluYWxpemF0aW9uIGlzIGFsc28gc3RvcHBlZCBvbiBGYWlsdXJlIHdpdGggZGV0YWlsICdpZ25vcmUuJyBPbiBmYWlsdXJlXG4gICAgICogd2l0aCBkZXRhaWwgJ2luYXBwbGljYWJsZScsIGZpbmFsaXphdGlvbiBjb250aW51ZXMgd2l0aCB0aGUgbmV4dCBydWxlLiBGYWlscyB3aXRoIGFuXG4gICAgICogZXJyb3IgZGV0YWlsICdlcnJvcicgYW5kIGFuIGluZm9ybWF0aXZlIG1lc3NhZ2UgaWYgYW4gZXJyb3Igb2NjdXJzLlxuICAgICAqL1xuICAgIGZpbmFsaXplUHJvcGVydGllcyhkZWZlcnJlZDogSnNvbk9iamVjdFtdLCBzdGF0ZTogSnNvbkVkaXRvclN0YXRlKTogRGV0YWlsZWRSZXN1bHQ8SnNvbk9iamVjdFtdLCBKc29uRWRpdEZhaWx1cmVSZWFzb24+O1xufVxuXG4vKipcbiAqIERlZmF1bHQgYmFzZSBpbXBsZW1lbnRhdGlvbiBvZiBKc29uRWRpdG9yIHJ1bGUgcmV0dXJucyBpbmFwcGxpY2FibGUgZm9yIGFsbCBvcGVyYXRpb25zIHNvIHRoYXRcbiAqIGRlcml2ZWQgY2xhc3NlcyBuZWVkIG9ubHkgaW1wbGVtZW50IHRoZSBvcGVyYXRpb25zIHRoZXkgYWN0dWFsbHkgc3VwcG9ydC5cbiAqL1xuZXhwb3J0IGNsYXNzIEpzb25FZGl0b3JSdWxlQmFzZSBpbXBsZW1lbnRzIEpzb25FZGl0b3JSdWxlIHtcbiAgICAvLyBpc3RhbmJ1bCBpZ25vcmUgbmV4dFxuICAgIHB1YmxpYyBlZGl0UHJvcGVydHkoX2tleTogc3RyaW5nLCBfdmFsdWU6IEpzb25WYWx1ZSwgX3N0YXRlOiBKc29uRWRpdG9yU3RhdGUpOiBEZXRhaWxlZFJlc3VsdDxKc29uT2JqZWN0LCBKc29uUHJvcGVydHlFZGl0RmFpbHVyZVJlYXNvbj4ge1xuICAgICAgICAvLyBpc3RhbmJ1bCBpZ25vcmUgbmV4dFxuICAgICAgICByZXR1cm4gZmFpbFdpdGhEZXRhaWwoJ2luYXBwbGljYWJsZScsICdpbmFwcGxpY2FibGUnKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZWRpdFZhbHVlKF92YWx1ZTogSnNvblZhbHVlLCBfc3RhdGU6IEpzb25FZGl0b3JTdGF0ZSk6IERldGFpbGVkUmVzdWx0PEpzb25WYWx1ZSwgSnNvbkVkaXRGYWlsdXJlUmVhc29uPiB7XG4gICAgICAgIHJldHVybiBmYWlsV2l0aERldGFpbCgnaW5hcHBsaWNhYmxlJywgJ2luYXBwbGljYWJsZScpO1xuICAgIH1cblxuICAgIHB1YmxpYyBmaW5hbGl6ZVByb3BlcnRpZXMoX2RlZmVycmVkOiBKc29uT2JqZWN0W10sIF9zdGF0ZTogSnNvbkVkaXRvclN0YXRlKTogRGV0YWlsZWRSZXN1bHQ8SnNvbk9iamVjdFtdLCBKc29uRWRpdEZhaWx1cmVSZWFzb24+IHtcbiAgICAgICAgcmV0dXJuIGZhaWxXaXRoRGV0YWlsKCdpbmFwcGxpY2FibGUnLCAnaW5hcHBsaWNhYmxlJyk7XG4gICAgfVxufVxuIl19
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoianNvbkVkaXRvclJ1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvanNvbkVkaXRvci9qc29uRWRpdG9yUnVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0JHOzs7QUFFSCw0Q0FBK0Q7QUE0Qy9EOzs7R0FHRztBQUNILE1BQWEsa0JBQWtCO0lBQzNCLHVCQUF1QjtJQUNoQixZQUFZLENBQUMsSUFBWSxFQUFFLE1BQWlCLEVBQUUsTUFBdUI7UUFDeEUsdUJBQXVCO1FBQ3ZCLE9BQU8sSUFBQSx5QkFBYyxFQUFDLGNBQWMsRUFBRSxjQUFjLENBQUMsQ0FBQztJQUMxRCxDQUFDO0lBRU0sU0FBUyxDQUFDLE1BQWlCLEVBQUUsTUFBdUI7UUFDdkQsT0FBTyxJQUFBLHlCQUFjLEVBQUMsY0FBYyxFQUFFLGNBQWMsQ0FBQyxDQUFDO0lBQzFELENBQUM7SUFFTSxrQkFBa0IsQ0FBQyxTQUF1QixFQUFFLE1BQXVCO1FBQ3RFLE9BQU8sSUFBQSx5QkFBYyxFQUFDLGNBQWMsRUFBRSxjQUFjLENBQUMsQ0FBQztJQUMxRCxDQUFDO0NBQ0o7QUFkRCxnREFjQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjAgRXJpayBGb3J0dW5lXG4gKlxuICogUGVybWlzc2lvbiBpcyBoZXJlYnkgZ3JhbnRlZCwgZnJlZSBvZiBjaGFyZ2UsIHRvIGFueSBwZXJzb24gb2J0YWluaW5nIGEgY29weVxuICogb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzICh0aGUgXCJTb2Z0d2FyZVwiKSwgdG8gZGVhbFxuICogaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0c1xuICogdG8gdXNlLCBjb3B5LCBtb2RpZnksIG1lcmdlLCBwdWJsaXNoLCBkaXN0cmlidXRlLCBzdWJsaWNlbnNlLCBhbmQvb3Igc2VsbFxuICogY29waWVzIG9mIHRoZSBTb2Z0d2FyZSwgYW5kIHRvIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIFNvZnR3YXJlIGlzXG4gKiBmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOlxuICpcbiAqIFRoZSBhYm92ZSBjb3B5cmlnaHQgbm90aWNlIGFuZCB0aGlzIHBlcm1pc3Npb24gbm90aWNlIHNoYWxsIGJlIGluY2x1ZGVkIGluIGFsbFxuICogY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS5cbiAqXG4gKiBUSEUgU09GVFdBUkUgSVMgUFJPVklERUQgXCJBUyBJU1wiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELCBFWFBSRVNTIE9SXG4gKiBJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GIE1FUkNIQU5UQUJJTElUWSxcbiAqIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFORCBOT05JTkZSSU5HRU1FTlQuIElOIE5PIEVWRU5UIFNIQUxMIFRIRVxuICogQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWSBDTEFJTSwgREFNQUdFUyBPUiBPVEhFUlxuICogTElBQklMSVRZLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwgVE9SVCBPUiBPVEhFUldJU0UsIEFSSVNJTkcgRlJPTSxcbiAqIE9VVCBPRiBPUiBJTiBDT05ORUNUSU9OIFdJVEggVEhFIFNPRlRXQVJFIE9SIFRIRSBVU0UgT1IgT1RIRVIgREVBTElOR1MgSU4gVEhFXG4gKiBTT0ZUV0FSRS5cbiAqL1xuXG5pbXBvcnQgeyBEZXRhaWxlZFJlc3VsdCwgZmFpbFdpdGhEZXRhaWwgfSBmcm9tICdAZmd2L3RzLXV0aWxzJztcbmltcG9ydCB7IEpzb25FZGl0RmFpbHVyZVJlYXNvbiwgSnNvblByb3BlcnR5RWRpdEZhaWx1cmVSZWFzb24gfSBmcm9tICcuL2NvbW1vbic7XG5pbXBvcnQgeyBKc29uT2JqZWN0LCBKc29uVmFsdWUgfSBmcm9tICcuLi9jb21tb24nO1xuXG5pbXBvcnQgeyBKc29uRWRpdG9yU3RhdGUgfSBmcm9tICcuL2pzb25FZGl0b3JTdGF0ZSc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgSnNvbkVkaXRvclJ1bGUge1xuICAgIC8qKlxuICAgICAqIENhbGxlZCBieSBhIEpTT04gZWRpdG9yIHRvIHBvc3NpYmx5IGVkaXQgb25lIG9mIHRoZSBwcm9wZXJ0aWVzIGJlaW5nIG1lcmdlZCBpbnRvIGEgdGFyZ2V0IG9iamVjdC5cbiAgICAgKiBAcGFyYW0ga2V5IFRoZSBrZXkgb2YgdGhlIHByb3BlcnR5IHRvIGJlIGVkaXRlZFxuICAgICAqIEBwYXJhbSB2YWx1ZSBUaGUgdmFsdWUgb2YgdGhlIHByb3BlcnR5IHRvIGJlIGVkaXRlZFxuICAgICAqIEBwYXJhbSBzdGF0ZSBFZGl0b3Igc3RhdGUgd2hpY2ggYXBwbGllcyB0byB0aGUgZWRpdFxuICAgICAqIEByZXR1cm5zIElmIHRoZSBwcm9wZXJ0eSB3YXMgZWRpdGVkLCByZXR1cm5zIFN1Y2Nlc3Mgd2l0aCBhIEpTT04gb2JqZWN0IGNvbnRhaW5pbmcgdGhlIGVkaXRlZCByZXN1bHRzXG4gICAgICogYW5kIHdpdGggZGV0YWlsICdlZGl0ZWQnLiBJZiB0aGlzIHByb3BlcnR5IHNob3VsZCBiZSBkZWZlcnJlZCBmb3IgbGF0ZXIgY29uc2lkZXJhdGlvbiBvciBtZXJnLCBTdWNjZWVkc1xuICAgICAqIHdpdGggZGV0YWlsICdkZWZlcnJlZCcgYW5kIGFuIEpzb25PYmplY3QgdG8gYmUgZmluYWxpemVkLiAgSWYgdGhlIHJ1bGUgZG9lcyBub3QgYWZmZWN0IHRoaXMgcHJvcGVydHksXG4gICAgICogZmFpbHMgd2l0aCBkZXRhaWwgJ2luYXBwbGljYWJsZScuIElmIGFuIGVycm9yIG9jY3VycmVkIHdoaWxlIHByb2Nlc3NpbmcgdGhlIGVycm9yLCByZXR1cm5zIEZhaWx1cmUgd2l0aFxuICAgICAqIGRldGFpbCAnZXJyb3InLlxuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby11c2UtYmVmb3JlLWRlZmluZVxuICAgIGVkaXRQcm9wZXJ0eShrZXk6IHN0cmluZywgdmFsdWU6IEpzb25WYWx1ZSwgc3RhdGU6IEpzb25FZGl0b3JTdGF0ZSk6IERldGFpbGVkUmVzdWx0PEpzb25PYmplY3QsIEpzb25Qcm9wZXJ0eUVkaXRGYWlsdXJlUmVhc29uPjtcblxuICAgIC8qKlxuICAgICAqIENhbGxlZCBieSBhIEpTT04gZWRpdG9yIHRvIHBvc3NpYmx5IGVkaXQgYSBwcm9wZXJ0eSB2YWx1ZSBvciBhcnJheSBlbGVtZW50XG4gICAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSB0byBiZSBlZGl0ZWRcbiAgICAgKiBAcGFyYW0gc3RhdGUgRWRpdG9yIHN0YXRlIHdoaWNoIGFwcGxpZXMgdG8gdGhlIGVkaXRcbiAgICAgKiBAcmV0dXJucyBSZXR1cm5zIHN1Y2Nlc3Mgd2l0aCB0aGUgSnNvblZhbHVlIHRvIGJlIGluc2VydGVkLCB3aXRoIGRldGFpbCAnZWRpdGVkJyBpZiB0aGUgdmFsdWUgd2FzXG4gICAgICogZWRpdGVkLiAgUmV0dXJucyBmYWlsdXJlIHdpdGggJ2luYXBwbGljYWJsZScgaWYgdGhlIHJ1bGUgZG9lcyBub3QgYWZmZWN0IHRoaXMgdmFsdWUuICBGYWlscyB3aXRoXG4gICAgICogZGV0YWlsICdpZ25vcmUnIGlmIHRoZSB2YWx1ZSBpcyB0byBiZSBpZ25vcmVkLCBvciB3aXRoICdlcnJvcicgaWYgYW4gZXJyb3Igb2NjdXJzLlxuICAgICAqL1xuICAgIGVkaXRWYWx1ZSh2YWx1ZTogSnNvblZhbHVlLCBzdGF0ZTogSnNvbkVkaXRvclN0YXRlKTogRGV0YWlsZWRSZXN1bHQ8SnNvblZhbHVlLCBKc29uRWRpdEZhaWx1cmVSZWFzb24+O1xuXG4gICAgLyoqXG4gICAgICogQ2FsbGVkIGZvciBlYWNoIHJ1bGUgYWZ0ZXIgYWxsIHByb3BlcnRpZXMgaGF2ZSBiZWVuIG1lcmdlZC4gIEFueSBwcm9wZXJ0aWVzIHRoYXQgd2VyZSBkZWZlcnJlZFxuICAgICAqIGR1cmluZyB0aGUgaW5pdGlhbCBlZGl0IHBhc3MgYXJlIHN1cHBsaWVkIGFzIGlucHV0LlxuICAgICAqIEBwYXJhbSBkZWZlcnJlZCBBbnkgb2JqZWN0cyB0aGF0IHdlcmUgZGVmZXJyZWQgZHVyaW5nIHRoZSBmaXJzdCBlZGl0IHBhc3NcbiAgICAgKiBAcGFyYW0gc3RhdGUgRWRpdG9yIHN0YXRlIHdoaWNoIGFwcGxpZXMgdG8gdGhlIGVkaXRcbiAgICAgKiBAcmV0dXJucyBPbiBzdWNjZXNzZnVsIHJldHVybiwgYW55IHJldHVybmVkIG9iamVjdHMgYXJlIG1lcmdlZCBpbiBvcmRlciBhbmQgZmluYWxpemF0aW9uXG4gICAgICogaXMgc3RvcHBlZC4gRmluYWxpemF0aW9uIGlzIGFsc28gc3RvcHBlZCBvbiBGYWlsdXJlIHdpdGggZGV0YWlsICdpZ25vcmUuJyBPbiBmYWlsdXJlXG4gICAgICogd2l0aCBkZXRhaWwgJ2luYXBwbGljYWJsZScsIGZpbmFsaXphdGlvbiBjb250aW51ZXMgd2l0aCB0aGUgbmV4dCBydWxlLiBGYWlscyB3aXRoIGFuXG4gICAgICogZXJyb3IgZGV0YWlsICdlcnJvcicgYW5kIGFuIGluZm9ybWF0aXZlIG1lc3NhZ2UgaWYgYW4gZXJyb3Igb2NjdXJzLlxuICAgICAqL1xuICAgIGZpbmFsaXplUHJvcGVydGllcyhkZWZlcnJlZDogSnNvbk9iamVjdFtdLCBzdGF0ZTogSnNvbkVkaXRvclN0YXRlKTogRGV0YWlsZWRSZXN1bHQ8SnNvbk9iamVjdFtdLCBKc29uRWRpdEZhaWx1cmVSZWFzb24+O1xufVxuXG4vKipcbiAqIERlZmF1bHQgYmFzZSBpbXBsZW1lbnRhdGlvbiBvZiBKc29uRWRpdG9yIHJ1bGUgcmV0dXJucyBpbmFwcGxpY2FibGUgZm9yIGFsbCBvcGVyYXRpb25zIHNvIHRoYXRcbiAqIGRlcml2ZWQgY2xhc3NlcyBuZWVkIG9ubHkgaW1wbGVtZW50IHRoZSBvcGVyYXRpb25zIHRoZXkgYWN0dWFsbHkgc3VwcG9ydC5cbiAqL1xuZXhwb3J0IGNsYXNzIEpzb25FZGl0b3JSdWxlQmFzZSBpbXBsZW1lbnRzIEpzb25FZGl0b3JSdWxlIHtcbiAgICAvLyBpc3RhbmJ1bCBpZ25vcmUgbmV4dFxuICAgIHB1YmxpYyBlZGl0UHJvcGVydHkoX2tleTogc3RyaW5nLCBfdmFsdWU6IEpzb25WYWx1ZSwgX3N0YXRlOiBKc29uRWRpdG9yU3RhdGUpOiBEZXRhaWxlZFJlc3VsdDxKc29uT2JqZWN0LCBKc29uUHJvcGVydHlFZGl0RmFpbHVyZVJlYXNvbj4ge1xuICAgICAgICAvLyBpc3RhbmJ1bCBpZ25vcmUgbmV4dFxuICAgICAgICByZXR1cm4gZmFpbFdpdGhEZXRhaWwoJ2luYXBwbGljYWJsZScsICdpbmFwcGxpY2FibGUnKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZWRpdFZhbHVlKF92YWx1ZTogSnNvblZhbHVlLCBfc3RhdGU6IEpzb25FZGl0b3JTdGF0ZSk6IERldGFpbGVkUmVzdWx0PEpzb25WYWx1ZSwgSnNvbkVkaXRGYWlsdXJlUmVhc29uPiB7XG4gICAgICAgIHJldHVybiBmYWlsV2l0aERldGFpbCgnaW5hcHBsaWNhYmxlJywgJ2luYXBwbGljYWJsZScpO1xuICAgIH1cblxuICAgIHB1YmxpYyBmaW5hbGl6ZVByb3BlcnRpZXMoX2RlZmVycmVkOiBKc29uT2JqZWN0W10sIF9zdGF0ZTogSnNvbkVkaXRvclN0YXRlKTogRGV0YWlsZWRSZXN1bHQ8SnNvbk9iamVjdFtdLCBKc29uRWRpdEZhaWx1cmVSZWFzb24+IHtcbiAgICAgICAgcmV0dXJuIGZhaWxXaXRoRGV0YWlsKCdpbmFwcGxpY2FibGUnLCAnaW5hcHBsaWNhYmxlJyk7XG4gICAgfVxufVxuIl19
@@ -1,43 +1,14 @@
1
1
  import { DetailedFailure, Result } from '@fgv/ts-utils';
2
+ import { IJsonCloneEditor, JsonEditFailureReason, JsonEditorOptions, JsonEditorValidationOptions, JsonEditorValidationRules } from './common';
2
3
  import { JsonContext, JsonReferenceMap, TemplateVars, VariableValue } from '../jsonContext';
3
- import { JsonEditFailureReason } from './jsonEditorRule';
4
- import { JsonEditor } from './jsonEditor';
5
4
  import { JsonObject } from '../common';
6
- export type JsonEditorValidationRules = 'invalidPropertyName' | 'invalidPropertyValue' | 'undefinedPropertyValue';
7
- export interface JsonEditorValidationOptions {
8
- /**
9
- * If onInvalidPropertyName is 'error' (default) then any property name
10
- * that is invalid after template rendering causes an error and stops
11
- * conversion. If onInvalidPropertyName is 'ignore', then names which
12
- * are invalid after template rendering are passed through unchanged.
13
- */
14
- onInvalidPropertyName: 'error' | 'ignore';
15
- /**
16
- * If onInvalidPropertyValue is 'error' (default) then any illegal
17
- * property value other than undefined causes an error and stops
18
- * conversion. If onInvalidPropertyValue is 'ignore' then any
19
- * invalid property values are silently ignored.
20
- */
21
- onInvalidPropertyValue: 'error' | 'ignore';
22
- /**
23
- * If onUnknownPropertyValue is error, then any property with
24
- * value undefined will cause an error and stop conversion. If
25
- * onUndefinedPropertyValue is 'ignore' (default) then any
26
- * property with value undefined is silently ignored.
27
- */
28
- onUndefinedPropertyValue: 'error' | 'ignore';
29
- }
30
- export interface JsonEditorOptions {
31
- context?: JsonContext;
32
- validation: JsonEditorValidationOptions;
33
- }
34
5
  export declare class JsonEditorState {
35
6
  protected static _nextId: number;
36
- readonly editor: JsonEditor;
7
+ readonly editor: IJsonCloneEditor;
37
8
  readonly options: JsonEditorOptions;
38
9
  protected readonly _deferred: JsonObject[];
39
10
  protected readonly _id: number;
40
- constructor(editor: JsonEditor, baseOptions: JsonEditorOptions, runtimeContext?: JsonContext);
11
+ constructor(editor: IJsonCloneEditor, baseOptions: JsonEditorOptions, runtimeContext?: JsonContext);
41
12
  get context(): JsonContext | undefined;
42
13
  get deferred(): JsonObject[];
43
14
  protected static _getEffectiveOptions(options: JsonEditorOptions, context?: JsonContext): Result<JsonEditorOptions>;
@@ -83,4 +83,4 @@ class JsonEditorState {
83
83
  }
84
84
  exports.JsonEditorState = JsonEditorState;
85
85
  JsonEditorState._nextId = 0;
86
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsonEditorState.js","sourceRoot":"","sources":["../../src/jsonEditor/jsonEditorState.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedFailure, Result, failWithDetail, succeed } from '@fgv/ts-utils';\nimport { JsonContext, JsonReferenceMap, TemplateVars, VariableValue } from '../jsonContext';\nimport { JsonEditFailureReason, JsonPropertyEditFailureReason } from './jsonEditorRule';\n\nimport { JsonContextHelper } from '../contextHelpers';\nimport { JsonEditor } from './jsonEditor';\nimport { JsonObject } from '../common';\n\nexport type JsonEditorValidationRules = 'invalidPropertyName'|'invalidPropertyValue'|'undefinedPropertyValue';\n\nexport interface JsonEditorValidationOptions {\n    /**\n     * If onInvalidPropertyName is 'error' (default) then any property name\n     * that is invalid after template rendering causes an error and stops\n     * conversion.  If onInvalidPropertyName is 'ignore', then names which\n     * are invalid after template rendering are passed through unchanged.\n     */\n    onInvalidPropertyName: 'error'|'ignore';\n\n    /**\n     * If onInvalidPropertyValue is 'error' (default) then any illegal\n     * property value other than undefined causes an error and stops\n     * conversion.  If onInvalidPropertyValue is 'ignore' then any\n     * invalid property values are silently ignored.\n     */\n    onInvalidPropertyValue: 'error'|'ignore';\n\n    /**\n     * If onUnknownPropertyValue is error, then any property with\n     * value undefined will cause an error and stop conversion.  If\n     * onUndefinedPropertyValue is 'ignore' (default) then any\n     * property with value undefined is silently ignored.\n     */\n    onUndefinedPropertyValue: 'error'|'ignore';\n}\n\nexport interface JsonEditorOptions {\n    context?: JsonContext;\n    validation: JsonEditorValidationOptions;\n}\n\nexport class JsonEditorState {\n    protected static _nextId = 0;\n\n    public readonly editor: JsonEditor;\n\n    public readonly options: JsonEditorOptions;\n    protected readonly _deferred: JsonObject[] = [];\n    protected readonly _id: number;\n\n    public constructor(editor: JsonEditor, baseOptions: JsonEditorOptions, runtimeContext?: JsonContext) {\n        this.editor = editor;\n        this.options = JsonEditorState._getEffectiveOptions(baseOptions, runtimeContext).orThrow();\n        this._id = JsonEditorState._nextId++;\n    }\n\n    public get context(): JsonContext|undefined { return this.options.context; }\n\n    public get deferred(): JsonObject[] {\n        return this._deferred;\n    }\n\n    protected static _getEffectiveOptions(options: JsonEditorOptions, context?: JsonContext): Result<JsonEditorOptions> {\n        if (!context) {\n            return succeed(options);\n        }\n        return JsonContextHelper.mergeContext(options.context, context).onSuccess((merged) => {\n            return succeed({ context: merged, validation: options.validation });\n        });\n    }\n\n    public defer(obj: JsonObject): void {\n        this._deferred.push(obj);\n    }\n\n    public getVars(defaultContext?: JsonContext): TemplateVars|undefined {\n        return this.options.context?.vars ?? defaultContext?.vars;\n    }\n\n    public getRefs(defaultContext?: JsonContext): JsonReferenceMap|undefined {\n        return this.options.context?.refs ?? defaultContext?.refs;\n    }\n\n    public getContext(defaultContext?: JsonContext): JsonContext|undefined {\n        return JsonContextHelper.mergeContext(defaultContext, this.options.context).orDefault();\n    }\n\n    public extendContext(baseContext: JsonContext|undefined, add: { vars?: VariableValue[], refs?: JsonReferenceMap[] }): Result<JsonContext|undefined> {\n        const context = this.getContext(baseContext);\n        return JsonContextHelper.extendContext(context, add);\n    }\n\n    public failValidation<T=JsonObject>(\n        rule: JsonEditorValidationRules,\n        message?: string,\n        validation?: JsonEditorValidationOptions,\n    ): DetailedFailure<T, JsonEditFailureReason> {\n        let detail: JsonPropertyEditFailureReason = 'error';\n        const effective = validation ?? this.options.validation;\n        switch (rule) {\n            case 'invalidPropertyName':\n                detail = (effective.onInvalidPropertyName !== 'ignore') ? 'error' : 'inapplicable';\n                break;\n            case 'invalidPropertyValue':\n                detail = (effective.onInvalidPropertyValue !== 'ignore') ? 'error' : 'ignore';\n                break;\n            case 'undefinedPropertyValue':\n                detail = (effective.onUndefinedPropertyValue !== 'error') ? 'ignore' : 'error';\n                // istanbul ignore next\n                message = message ?? 'Cannot convert undefined to JSON';\n                break;\n        }\n        // istanbul ignore next\n        return failWithDetail(message ?? rule, detail);\n    }\n}\n"]}
86
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsonEditorState.js","sourceRoot":"","sources":["../../src/jsonEditor/jsonEditorState.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedFailure, Result, failWithDetail, succeed } from '@fgv/ts-utils';\nimport { IJsonCloneEditor, JsonEditFailureReason, JsonEditorOptions, JsonEditorValidationOptions, JsonEditorValidationRules, JsonPropertyEditFailureReason } from './common';\nimport { JsonContext, JsonReferenceMap, TemplateVars, VariableValue } from '../jsonContext';\n\nimport { JsonContextHelper } from '../contextHelpers';\nimport { JsonObject } from '../common';\n\nexport class JsonEditorState {\n    protected static _nextId = 0;\n\n    public readonly editor: IJsonCloneEditor;\n\n    public readonly options: JsonEditorOptions;\n    protected readonly _deferred: JsonObject[] = [];\n    protected readonly _id: number;\n\n    public constructor(editor: IJsonCloneEditor, baseOptions: JsonEditorOptions, runtimeContext?: JsonContext) {\n        this.editor = editor;\n        this.options = JsonEditorState._getEffectiveOptions(baseOptions, runtimeContext).orThrow();\n        this._id = JsonEditorState._nextId++;\n    }\n\n    public get context(): JsonContext|undefined { return this.options.context; }\n\n    public get deferred(): JsonObject[] {\n        return this._deferred;\n    }\n\n    protected static _getEffectiveOptions(options: JsonEditorOptions, context?: JsonContext): Result<JsonEditorOptions> {\n        if (!context) {\n            return succeed(options);\n        }\n        return JsonContextHelper.mergeContext(options.context, context).onSuccess((merged) => {\n            return succeed({ context: merged, validation: options.validation });\n        });\n    }\n\n    public defer(obj: JsonObject): void {\n        this._deferred.push(obj);\n    }\n\n    public getVars(defaultContext?: JsonContext): TemplateVars|undefined {\n        return this.options.context?.vars ?? defaultContext?.vars;\n    }\n\n    public getRefs(defaultContext?: JsonContext): JsonReferenceMap|undefined {\n        return this.options.context?.refs ?? defaultContext?.refs;\n    }\n\n    public getContext(defaultContext?: JsonContext): JsonContext|undefined {\n        return JsonContextHelper.mergeContext(defaultContext, this.options.context).orDefault();\n    }\n\n    public extendContext(baseContext: JsonContext|undefined, add: { vars?: VariableValue[], refs?: JsonReferenceMap[] }): Result<JsonContext|undefined> {\n        const context = this.getContext(baseContext);\n        return JsonContextHelper.extendContext(context, add);\n    }\n\n    public failValidation<T=JsonObject>(\n        rule: JsonEditorValidationRules,\n        message?: string,\n        validation?: JsonEditorValidationOptions,\n    ): DetailedFailure<T, JsonEditFailureReason> {\n        let detail: JsonPropertyEditFailureReason = 'error';\n        const effective = validation ?? this.options.validation;\n        switch (rule) {\n            case 'invalidPropertyName':\n                detail = (effective.onInvalidPropertyName !== 'ignore') ? 'error' : 'inapplicable';\n                break;\n            case 'invalidPropertyValue':\n                detail = (effective.onInvalidPropertyValue !== 'ignore') ? 'error' : 'ignore';\n                break;\n            case 'undefinedPropertyValue':\n                detail = (effective.onUndefinedPropertyValue !== 'error') ? 'ignore' : 'error';\n                // istanbul ignore next\n                message = message ?? 'Cannot convert undefined to JSON';\n                break;\n        }\n        // istanbul ignore next\n        return failWithDetail(message ?? rule, detail);\n    }\n}\n"]}
@@ -1,7 +1,8 @@
1
1
  import { DetailedResult, Result } from '@fgv/ts-utils';
2
- import { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';
3
- import { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';
2
+ import { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';
4
3
  import { JsonObject, JsonValue } from '../../common';
4
+ import { JsonEditorRuleBase } from '../jsonEditorRule';
5
+ import { JsonEditorState } from '../jsonEditorState';
5
6
  /**
6
7
  * Returned by the _tryMatch method of the @see ConditionalJsonEditorRule
7
8
  * to indicate whether a successful match was due to a matching condition
@@ -23,8 +23,8 @@
23
23
  Object.defineProperty(exports, "__esModule", { value: true });
24
24
  exports.ConditionalJsonEditorRule = void 0;
25
25
  const ts_utils_1 = require("@fgv/ts-utils");
26
- const jsonEditorRule_1 = require("../jsonEditorRule");
27
26
  const common_1 = require("../../common");
27
+ const jsonEditorRule_1 = require("../jsonEditorRule");
28
28
  /**
29
29
  * The Conditional JSON editor rule evaluates properties with conditional keys,
30
30
  * omitting non-matching keys and merging keys that match, or default keys only
@@ -143,4 +143,4 @@ class ConditionalJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
143
143
  }
144
144
  }
145
145
  exports.ConditionalJsonEditorRule = ConditionalJsonEditorRule;
146
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"conditional.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/conditional.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedResult, Result, captureResult, failWithDetail, succeedWithDetail } from '@fgv/ts-utils';\nimport { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';\nimport { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';\nimport { JsonObject, JsonValue, isJsonObject } from '../../common';\n\n/**\n * Returned by the _tryMatch method of the @see ConditionalJsonEditorRule\n * to indicate whether a successful match was due to a matching condition\n * or a default value.\n */\nexport interface ConditionalJsonKeyResult extends JsonObject {\n    matchType: 'default'|'match'|'unconditional';\n}\n\n/**\n * On a successful match, the @see ConditionalJsonEditorRule stores a\n * ConditionalJsonDeferredObject describing the matching result, to\n * be resolved at finalization time.\n */\nexport interface ConditionalJsonDeferredObject extends ConditionalJsonKeyResult{\n    value: JsonValue;\n}\n\n/**\n * Configuration options for the Conditional JSON editor rule\n */\nexport interface ConditionalJsonRuleOptions extends Partial<JsonEditorOptions> {\n    /**\n     * If true (default) then properties with unconditional names\n     * (which start with !) are flattened.\n     */\n    flattenUnconditionalValues?: boolean;\n}\n\n/**\n * The Conditional JSON editor rule evaluates properties with conditional keys,\n * omitting non-matching keys and merging keys that match, or default keys only\n * if no other keys match.\n *\n * The default syntax for a conditional key is:\n *    \"?value1=value2\" - matches if value1 and value2 are the same, is ignored otherwise.\n *    \"?value\" - matches if value is a non-empty, non-whitespace string. Is ignored otherwise.\n *    \"?default\" - matches only if no other conditional blocks in the same object were matched\n */\nexport class ConditionalJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: ConditionalJsonRuleOptions;\n\n    /**\n     * Creates a new @see ConditionalJsonEditorRule\n     * @param options Optional configuration options used for this rule\n     */\n    public constructor(options?: ConditionalJsonRuleOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new @see ConditionalJsonEditorRule\n     * @param options Optional configuration options used for this rule\n     */\n    public static create(options?: ConditionalJsonRuleOptions): Result<ConditionalJsonEditorRule> {\n        return captureResult(() => new ConditionalJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property for conditional application.\n     * @param key The key of the property to be considered\n     * @param value The value of the property to be considered\n     * @param state The editor state for the object being edited\n     * @returns Returns Success with detail 'deferred' and a @see ConditionalJsonDeferredObject\n     * for a matching, default or unconditional key. Fails with detail 'ignore' for a\n     * non-matching conditional and with detail 'error' if an error occurs. Otherwise\n     * fails with detail 'inapplicable'.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        const result = this._tryParseCondition(key, state).onSuccess((deferred) => {\n            if (isJsonObject(value)) {\n                const rtrn: ConditionalJsonDeferredObject = { ...deferred, value };\n                return succeedWithDetail(rtrn, 'deferred');\n            }\n            return failWithDetail<JsonObject, JsonPropertyEditFailureReason>(`${key}: conditional body must be object`, 'error');\n        });\n\n        if (result.isFailure() && (result.detail === 'error')) {\n            return state.failValidation('invalidPropertyName', result.message, this._options?.validation);\n        }\n\n        return result;\n    }\n\n    /**\n     * Finalizes any deferred conditional properties. If the only deferred property is\n     * default, that property is emitted. Otherwise all matching properties are emitted.\n     * @param finalized The deferred properties to be considered for merge\n     * @param _state The editor state for the object being edited\n     */\n    public finalizeProperties(finalized: JsonObject[], _state: JsonEditorState): DetailedResult<JsonObject[], JsonEditFailureReason> {\n        let toMerge = finalized;\n        if (finalized.length > 1) {\n            if (finalized.find((o) => o.matchType === 'match') !== undefined) {\n                toMerge = finalized.filter((o) => (o.matchType === 'match') || (o.matchType === 'unconditional'));\n            }\n        }\n        return succeedWithDetail(toMerge.map((o) => o.value).filter(isJsonObject), 'edited');\n    }\n\n    /**\n     * Determines if a given property key is conditional. Derived classes can override this\n     * method to use a different format for conditional properties.\n     * @param key The key of the property to consider.\n     * @param state The editor state of the object being edited.\n     * @returns Success with detail 'deferred' and a @see ConditionalJsonKeyResult describing the\n     * match for a default or matching conditional property.  Fails with detail 'ignore' for\n     * a non-matching conditional property. Fails with detail 'error' if an error occurs\n     * or with detail 'inapplicable' if the key does not represent a conditional property.\n     */\n    protected _tryParseCondition(key: string, state: JsonEditorState): DetailedResult<ConditionalJsonKeyResult, JsonPropertyEditFailureReason> {\n        if (key.startsWith('?')) {\n            // ignore everything after any #\n            key = key.split('#')[0].trim();\n\n            if (key === '?default') {\n                return succeedWithDetail({ matchType: 'default' }, 'deferred');\n            }\n\n            const parts = key.substring(1).split(/(=|>=|<=|>|<|!=)/);\n            if (parts.length === 3) {\n                if (!this._compare(parts[0].trim(), parts[2].trim(), parts[1])) {\n                    return failWithDetail(`Condition ${key} does not match`, 'ignore');\n                }\n                return succeedWithDetail({ matchType: 'match' }, 'deferred');\n            }\n            else if (parts.length === 1) {\n                if (parts[0].trim().length === 0) {\n                    return failWithDetail(`Condition ${key} does not match`, 'ignore');\n                }\n                return succeedWithDetail({ matchType: 'match' }, 'deferred');\n            }\n            const message = `Malformed condition token ${key}`;\n            return state.failValidation('invalidPropertyName', message, this._options?.validation);\n        }\n        else if ((this._options?.flattenUnconditionalValues !== false) && key.startsWith('!')) {\n            return succeedWithDetail({ matchType: 'unconditional' }, 'deferred');\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _compare(left: string, right: string, operator: string): boolean {\n        switch (operator) {\n            case '=': return left === right;\n            case '>': return left > right;\n            case '<': return left < right;\n            case '>=': return left >= right;\n            case '<=': return left <= right;\n            case '!=': return left !== right;\n        }\n        // istanbul ignore next: unreachable\n        return false;\n    }\n}\n"]}
146
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"conditional.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/conditional.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;;;AAEH,4CAAyG;AAEzG,yCAAmE;AACnE,sDAAuD;AAgCvD;;;;;;;;;GASG;AACH,MAAa,yBAA0B,SAAQ,mCAAkB;IAG7D;;;OAGG;IACH,YAAmB,OAAoC;QACnD,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACI,MAAM,CAAC,MAAM,CAAC,OAAoC;QACrD,OAAO,IAAA,wBAAa,EAAC,GAAG,EAAE,CAAC,IAAI,yBAAyB,CAAC,OAAO,CAAC,CAAC,CAAC;IACvE,CAAC;IAED;;;;;;;;;OASG;IACI,YAAY,CAAC,GAAW,EAAE,KAAgB,EAAE,KAAsB;;QACrE,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC,SAAS,CAAC,CAAC,QAAQ,EAAE,EAAE;YACtE,IAAI,IAAA,qBAAY,EAAC,KAAK,CAAC,EAAE;gBACrB,MAAM,IAAI,GAAkC,EAAE,GAAG,QAAQ,EAAE,KAAK,EAAE,CAAC;gBACnE,OAAO,IAAA,4BAAiB,EAAC,IAAI,EAAE,UAAU,CAAC,CAAC;aAC9C;YACD,OAAO,IAAA,yBAAc,EAA4C,GAAG,GAAG,mCAAmC,EAAE,OAAO,CAAC,CAAC;QACzH,CAAC,CAAC,CAAC;QAEH,IAAI,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,KAAK,OAAO,CAAC,EAAE;YACnD,OAAO,KAAK,CAAC,cAAc,CAAC,qBAAqB,EAAE,MAAM,CAAC,OAAO,EAAE,MAAA,IAAI,CAAC,QAAQ,0CAAE,UAAU,CAAC,CAAC;SACjG;QAED,OAAO,MAAM,CAAC;IAClB,CAAC;IAED;;;;;OAKG;IACI,kBAAkB,CAAC,SAAuB,EAAE,MAAuB;QACtE,IAAI,OAAO,GAAG,SAAS,CAAC;QACxB,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE;YACtB,IAAI,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,KAAK,OAAO,CAAC,KAAK,SAAS,EAAE;gBAC9D,OAAO,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,KAAK,eAAe,CAAC,CAAC,CAAC;aACrG;SACJ;QACD,OAAO,IAAA,4BAAiB,EAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,qBAAY,CAAC,EAAE,QAAQ,CAAC,CAAC;IACzF,CAAC;IAED;;;;;;;;;OASG;IACO,kBAAkB,CAAC,GAAW,EAAE,KAAsB;;QAC5D,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE;YACrB,gCAAgC;YAChC,GAAG,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAE/B,IAAI,GAAG,KAAK,UAAU,EAAE;gBACpB,OAAO,IAAA,4BAAiB,EAAC,EAAE,SAAS,EAAE,SAAS,EAAE,EAAE,UAAU,CAAC,CAAC;aAClE;YAED,MAAM,KAAK,GAAG,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;YACzD,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE;gBACpB,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE;oBAC5D,OAAO,IAAA,yBAAc,EAAC,aAAa,GAAG,iBAAiB,EAAE,QAAQ,CAAC,CAAC;iBACtE;gBACD,OAAO,IAAA,4BAAiB,EAAC,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE,UAAU,CAAC,CAAC;aAChE;iBACI,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE;gBACzB,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE;oBAC9B,OAAO,IAAA,yBAAc,EAAC,aAAa,GAAG,iBAAiB,EAAE,QAAQ,CAAC,CAAC;iBACtE;gBACD,OAAO,IAAA,4BAAiB,EAAC,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE,UAAU,CAAC,CAAC;aAChE;YACD,MAAM,OAAO,GAAG,6BAA6B,GAAG,EAAE,CAAC;YACnD,OAAO,KAAK,CAAC,cAAc,CAAC,qBAAqB,EAAE,OAAO,EAAE,MAAA,IAAI,CAAC,QAAQ,0CAAE,UAAU,CAAC,CAAC;SAC1F;aACI,IAAI,CAAC,CAAA,MAAA,IAAI,CAAC,QAAQ,0CAAE,0BAA0B,MAAK,KAAK,CAAC,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE;YACnF,OAAO,IAAA,4BAAiB,EAAC,EAAE,SAAS,EAAE,eAAe,EAAE,EAAE,UAAU,CAAC,CAAC;SACxE;QACD,OAAO,IAAA,yBAAc,EAAC,cAAc,EAAE,cAAc,CAAC,CAAC;IAC1D,CAAC;IAES,QAAQ,CAAC,IAAY,EAAE,KAAa,EAAE,QAAgB;QAC5D,QAAQ,QAAQ,EAAE;YACd,KAAK,GAAG,CAAC,CAAC,OAAO,IAAI,KAAK,KAAK,CAAC;YAChC,KAAK,GAAG,CAAC,CAAC,OAAO,IAAI,GAAG,KAAK,CAAC;YAC9B,KAAK,GAAG,CAAC,CAAC,OAAO,IAAI,GAAG,KAAK,CAAC;YAC9B,KAAK,IAAI,CAAC,CAAC,OAAO,IAAI,IAAI,KAAK,CAAC;YAChC,KAAK,IAAI,CAAC,CAAC,OAAO,IAAI,IAAI,KAAK,CAAC;YAChC,KAAK,IAAI,CAAC,CAAC,OAAO,IAAI,KAAK,KAAK,CAAC;SACpC;QACD,oCAAoC;QACpC,OAAO,KAAK,CAAC;IACjB,CAAC;CACJ;AAnHD,8DAmHC","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedResult, Result, captureResult, failWithDetail, succeedWithDetail } from '@fgv/ts-utils';\nimport { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';\nimport { JsonObject, JsonValue, isJsonObject } from '../../common';\nimport { JsonEditorRuleBase } from '../jsonEditorRule';\nimport { JsonEditorState } from '../jsonEditorState';\n\n/**\n * Returned by the _tryMatch method of the @see ConditionalJsonEditorRule\n * to indicate whether a successful match was due to a matching condition\n * or a default value.\n */\nexport interface ConditionalJsonKeyResult extends JsonObject {\n    matchType: 'default'|'match'|'unconditional';\n}\n\n/**\n * On a successful match, the @see ConditionalJsonEditorRule stores a\n * ConditionalJsonDeferredObject describing the matching result, to\n * be resolved at finalization time.\n */\nexport interface ConditionalJsonDeferredObject extends ConditionalJsonKeyResult{\n    value: JsonValue;\n}\n\n/**\n * Configuration options for the Conditional JSON editor rule\n */\nexport interface ConditionalJsonRuleOptions extends Partial<JsonEditorOptions> {\n    /**\n     * If true (default) then properties with unconditional names\n     * (which start with !) are flattened.\n     */\n    flattenUnconditionalValues?: boolean;\n}\n\n/**\n * The Conditional JSON editor rule evaluates properties with conditional keys,\n * omitting non-matching keys and merging keys that match, or default keys only\n * if no other keys match.\n *\n * The default syntax for a conditional key is:\n *    \"?value1=value2\" - matches if value1 and value2 are the same, is ignored otherwise.\n *    \"?value\" - matches if value is a non-empty, non-whitespace string. Is ignored otherwise.\n *    \"?default\" - matches only if no other conditional blocks in the same object were matched\n */\nexport class ConditionalJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: ConditionalJsonRuleOptions;\n\n    /**\n     * Creates a new @see ConditionalJsonEditorRule\n     * @param options Optional configuration options used for this rule\n     */\n    public constructor(options?: ConditionalJsonRuleOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new @see ConditionalJsonEditorRule\n     * @param options Optional configuration options used for this rule\n     */\n    public static create(options?: ConditionalJsonRuleOptions): Result<ConditionalJsonEditorRule> {\n        return captureResult(() => new ConditionalJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property for conditional application.\n     * @param key The key of the property to be considered\n     * @param value The value of the property to be considered\n     * @param state The editor state for the object being edited\n     * @returns Returns Success with detail 'deferred' and a @see ConditionalJsonDeferredObject\n     * for a matching, default or unconditional key. Fails with detail 'ignore' for a\n     * non-matching conditional and with detail 'error' if an error occurs. Otherwise\n     * fails with detail 'inapplicable'.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        const result = this._tryParseCondition(key, state).onSuccess((deferred) => {\n            if (isJsonObject(value)) {\n                const rtrn: ConditionalJsonDeferredObject = { ...deferred, value };\n                return succeedWithDetail(rtrn, 'deferred');\n            }\n            return failWithDetail<JsonObject, JsonPropertyEditFailureReason>(`${key}: conditional body must be object`, 'error');\n        });\n\n        if (result.isFailure() && (result.detail === 'error')) {\n            return state.failValidation('invalidPropertyName', result.message, this._options?.validation);\n        }\n\n        return result;\n    }\n\n    /**\n     * Finalizes any deferred conditional properties. If the only deferred property is\n     * default, that property is emitted. Otherwise all matching properties are emitted.\n     * @param finalized The deferred properties to be considered for merge\n     * @param _state The editor state for the object being edited\n     */\n    public finalizeProperties(finalized: JsonObject[], _state: JsonEditorState): DetailedResult<JsonObject[], JsonEditFailureReason> {\n        let toMerge = finalized;\n        if (finalized.length > 1) {\n            if (finalized.find((o) => o.matchType === 'match') !== undefined) {\n                toMerge = finalized.filter((o) => (o.matchType === 'match') || (o.matchType === 'unconditional'));\n            }\n        }\n        return succeedWithDetail(toMerge.map((o) => o.value).filter(isJsonObject), 'edited');\n    }\n\n    /**\n     * Determines if a given property key is conditional. Derived classes can override this\n     * method to use a different format for conditional properties.\n     * @param key The key of the property to consider.\n     * @param state The editor state of the object being edited.\n     * @returns Success with detail 'deferred' and a @see ConditionalJsonKeyResult describing the\n     * match for a default or matching conditional property.  Fails with detail 'ignore' for\n     * a non-matching conditional property. Fails with detail 'error' if an error occurs\n     * or with detail 'inapplicable' if the key does not represent a conditional property.\n     */\n    protected _tryParseCondition(key: string, state: JsonEditorState): DetailedResult<ConditionalJsonKeyResult, JsonPropertyEditFailureReason> {\n        if (key.startsWith('?')) {\n            // ignore everything after any #\n            key = key.split('#')[0].trim();\n\n            if (key === '?default') {\n                return succeedWithDetail({ matchType: 'default' }, 'deferred');\n            }\n\n            const parts = key.substring(1).split(/(=|>=|<=|>|<|!=)/);\n            if (parts.length === 3) {\n                if (!this._compare(parts[0].trim(), parts[2].trim(), parts[1])) {\n                    return failWithDetail(`Condition ${key} does not match`, 'ignore');\n                }\n                return succeedWithDetail({ matchType: 'match' }, 'deferred');\n            }\n            else if (parts.length === 1) {\n                if (parts[0].trim().length === 0) {\n                    return failWithDetail(`Condition ${key} does not match`, 'ignore');\n                }\n                return succeedWithDetail({ matchType: 'match' }, 'deferred');\n            }\n            const message = `Malformed condition token ${key}`;\n            return state.failValidation('invalidPropertyName', message, this._options?.validation);\n        }\n        else if ((this._options?.flattenUnconditionalValues !== false) && key.startsWith('!')) {\n            return succeedWithDetail({ matchType: 'unconditional' }, 'deferred');\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _compare(left: string, right: string, operator: string): boolean {\n        switch (operator) {\n            case '=': return left === right;\n            case '>': return left > right;\n            case '<': return left < right;\n            case '>=': return left >= right;\n            case '<=': return left <= right;\n            case '!=': return left !== right;\n        }\n        // istanbul ignore next: unreachable\n        return false;\n    }\n}\n"]}
@@ -1,8 +1,9 @@
1
1
  import { DetailedResult, Result } from '@fgv/ts-utils';
2
2
  import { JsonContext, VariableValue } from '../../jsonContext';
3
- import { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';
4
- import { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';
3
+ import { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';
5
4
  import { JsonObject, JsonValue } from '../../common';
5
+ import { JsonEditorRuleBase } from '../jsonEditorRule';
6
+ import { JsonEditorState } from '../jsonEditorState';
6
7
  /**
7
8
  * Represents the parts of a multi-value property key.
8
9
  */
@@ -126,4 +126,4 @@ class MultiValueJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
126
126
  }
127
127
  }
128
128
  exports.MultiValueJsonEditorRule = MultiValueJsonEditorRule;
129
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multivalue.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/multivalue.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedResult, Result, allSucceed, captureResult, failWithDetail, succeed, succeedWithDetail } from '@fgv/ts-utils';\nimport { JsonContext, VariableValue } from '../../jsonContext';\nimport { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';\nimport { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';\nimport { JsonObject, JsonValue } from '../../common';\n\n/**\n * Represents the parts of a multi-value property key.\n */\nexport interface MultiValuePropertyParts {\n    /**\n     * The original matched token\n     */\n    readonly token: string;\n\n    /**\n     * The name of the variable used to project each possible\n     * property value into the child values or objects being\n     * resolved.\n     */\n    readonly propertyVariable: string;\n\n    /**\n     * The set of property values to be expanded\n     */\n    readonly propertyValues: string[];\n\n    /**\n     * If true, the resolved values are added as an array\n     * with the name of the propertyVariable. If false,\n     * values are added as individual properties with names\n     * that correspond the value.\n     */\n    readonly asArray: boolean;\n}\n\n/**\n * The Multi-Value JSON editor rule expands matching keys multiple\n * times, projecting the value into the template context for any\n * child objects rendered by the rule.\n *\n * The default syntax for a multi-value key is:\n *  \"[[var]]=value1,value2,value3\"\n * Where \"var\" is the name of the variable that will be passed to\n * child template resolution, and \"value1,value2,value3\" is a\n * comma-separated list of values to be expanded.\n */\nexport class MultiValueJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: JsonEditorOptions;\n\n    /**\n     * Creates a new MultiValueJsonEditorRule.\n     * @param options Optional configuration options\n     */\n    public constructor(options?: JsonEditorOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new MultiValueJsonEditorRule.\n     * @param options Optional configuration options\n     */\n    public static create(options?: JsonEditorOptions): Result<MultiValueJsonEditorRule> {\n        return captureResult(() => new MultiValueJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property for multi-value expansion.\n     * @param key The key of the property to be considered\n     * @param value The value of the property to be considered\n     * @param state The editor state for the object being edited\n     * @returns Returns Success with an object containing the fully-resolved child\n     * values to be merged for matching multi-value property. Fails with\n     * detail 'error' if an error occurs or with detail 'inapplicable' if\n     * the property key is not a conditional property.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        const json: JsonObject = {};\n        const result = this._tryParse(key, state).onSuccess((parts) => {\n            return allSucceed(parts.propertyValues.map((pv) => {\n                return this._deriveContext(state, [parts.propertyVariable, pv]).onSuccess((ctx) => {\n                    return state.editor.clone(value, ctx).onSuccess((cloned) => {\n                        json[pv] = cloned;\n                        return succeedWithDetail(cloned);\n                    });\n                });\n            }), json).onSuccess(() => {\n                if (parts.asArray) {\n                    const arrayRtrn: JsonObject = {};\n                    arrayRtrn[parts.propertyVariable] = Array.from(Object.values(json));\n                    return succeed(arrayRtrn);\n                }\n                return succeed(json);\n            }).withFailureDetail('error');\n        });\n\n        if (result.isFailure() && (result.detail === 'error')) {\n            return state.failValidation('invalidPropertyName', result.message);\n        }\n        return result;\n    }\n\n    protected _deriveContext(state: JsonEditorState, ...values: VariableValue[]): Result<JsonContext|undefined> {\n        return state.extendContext(this._options?.context, { vars: values });\n    }\n\n    /**\n     * Determines if a given property key is multi-value. Derived classes can override this\n     * method to use a different format for multi-value properties.\n     * @param key The key of the property to consider.\n     * @param state The editor state of the object being edited.\n     * @returns Success with detail 'deferred' and a @see MultiValuePropertyParts describing the\n     * match for matching multi-value property.  Fails with detail 'error' if an error occurs\n     * or with detail 'inapplicable' if the key does not represent a multi-value property.\n     */\n    protected _tryParse(token: string, state: JsonEditorState): DetailedResult<MultiValuePropertyParts, JsonEditFailureReason> {\n        let parts: string[] = [];\n        let asArray = false;\n\n        if (token.startsWith('[[')) {\n            parts = token.substring(2).split(']]=');\n            asArray = true;\n        }\n        else if (token.startsWith('*')) {\n            parts = token.substring(1).split('=');\n            asArray = false;\n        }\n        else {\n            return failWithDetail(token, 'inapplicable');\n        }\n\n        if (parts.length !== 2) {\n            const message = `Malformed multi-value property: ${token}`;\n            return state.failValidation('invalidPropertyName', message, this._options?.validation);\n        }\n\n        if (parts[1].includes('{{')) {\n            return failWithDetail('unresolved template', 'inapplicable');\n        }\n\n        const propertyVariable = parts[0];\n        const propertyValues = parts[1].split(',');\n        return succeedWithDetail({ token, propertyVariable, propertyValues, asArray });\n    }\n}\n"]}
129
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multivalue.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/multivalue.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedResult, Result, allSucceed, captureResult, failWithDetail, succeed, succeedWithDetail } from '@fgv/ts-utils';\nimport { JsonContext, VariableValue } from '../../jsonContext';\nimport { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';\nimport { JsonObject, JsonValue } from '../../common';\n\nimport { JsonEditorRuleBase } from '../jsonEditorRule';\nimport { JsonEditorState } from '../jsonEditorState';\n\n/**\n * Represents the parts of a multi-value property key.\n */\nexport interface MultiValuePropertyParts {\n    /**\n     * The original matched token\n     */\n    readonly token: string;\n\n    /**\n     * The name of the variable used to project each possible\n     * property value into the child values or objects being\n     * resolved.\n     */\n    readonly propertyVariable: string;\n\n    /**\n     * The set of property values to be expanded\n     */\n    readonly propertyValues: string[];\n\n    /**\n     * If true, the resolved values are added as an array\n     * with the name of the propertyVariable. If false,\n     * values are added as individual properties with names\n     * that correspond the value.\n     */\n    readonly asArray: boolean;\n}\n\n/**\n * The Multi-Value JSON editor rule expands matching keys multiple\n * times, projecting the value into the template context for any\n * child objects rendered by the rule.\n *\n * The default syntax for a multi-value key is:\n *  \"[[var]]=value1,value2,value3\"\n * Where \"var\" is the name of the variable that will be passed to\n * child template resolution, and \"value1,value2,value3\" is a\n * comma-separated list of values to be expanded.\n */\nexport class MultiValueJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: JsonEditorOptions;\n\n    /**\n     * Creates a new MultiValueJsonEditorRule.\n     * @param options Optional configuration options\n     */\n    public constructor(options?: JsonEditorOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new MultiValueJsonEditorRule.\n     * @param options Optional configuration options\n     */\n    public static create(options?: JsonEditorOptions): Result<MultiValueJsonEditorRule> {\n        return captureResult(() => new MultiValueJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property for multi-value expansion.\n     * @param key The key of the property to be considered\n     * @param value The value of the property to be considered\n     * @param state The editor state for the object being edited\n     * @returns Returns Success with an object containing the fully-resolved child\n     * values to be merged for matching multi-value property. Fails with\n     * detail 'error' if an error occurs or with detail 'inapplicable' if\n     * the property key is not a conditional property.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        const json: JsonObject = {};\n        const result = this._tryParse(key, state).onSuccess((parts) => {\n            return allSucceed(parts.propertyValues.map((pv) => {\n                return this._deriveContext(state, [parts.propertyVariable, pv]).onSuccess((ctx) => {\n                    return state.editor.clone(value, ctx).onSuccess((cloned) => {\n                        json[pv] = cloned;\n                        return succeedWithDetail(cloned);\n                    });\n                });\n            }), json).onSuccess(() => {\n                if (parts.asArray) {\n                    const arrayRtrn: JsonObject = {};\n                    arrayRtrn[parts.propertyVariable] = Array.from(Object.values(json));\n                    return succeed(arrayRtrn);\n                }\n                return succeed(json);\n            }).withFailureDetail('error');\n        });\n\n        if (result.isFailure() && (result.detail === 'error')) {\n            return state.failValidation('invalidPropertyName', result.message);\n        }\n        return result;\n    }\n\n    protected _deriveContext(state: JsonEditorState, ...values: VariableValue[]): Result<JsonContext|undefined> {\n        return state.extendContext(this._options?.context, { vars: values });\n    }\n\n    /**\n     * Determines if a given property key is multi-value. Derived classes can override this\n     * method to use a different format for multi-value properties.\n     * @param key The key of the property to consider.\n     * @param state The editor state of the object being edited.\n     * @returns Success with detail 'deferred' and a @see MultiValuePropertyParts describing the\n     * match for matching multi-value property.  Fails with detail 'error' if an error occurs\n     * or with detail 'inapplicable' if the key does not represent a multi-value property.\n     */\n    protected _tryParse(token: string, state: JsonEditorState): DetailedResult<MultiValuePropertyParts, JsonEditFailureReason> {\n        let parts: string[] = [];\n        let asArray = false;\n\n        if (token.startsWith('[[')) {\n            parts = token.substring(2).split(']]=');\n            asArray = true;\n        }\n        else if (token.startsWith('*')) {\n            parts = token.substring(1).split('=');\n            asArray = false;\n        }\n        else {\n            return failWithDetail(token, 'inapplicable');\n        }\n\n        if (parts.length !== 2) {\n            const message = `Malformed multi-value property: ${token}`;\n            return state.failValidation('invalidPropertyName', message, this._options?.validation);\n        }\n\n        if (parts[1].includes('{{')) {\n            return failWithDetail('unresolved template', 'inapplicable');\n        }\n\n        const propertyVariable = parts[0];\n        const propertyValues = parts[1].split(',');\n        return succeedWithDetail({ token, propertyVariable, propertyValues, asArray });\n    }\n}\n"]}
@@ -1,8 +1,9 @@
1
1
  import { DetailedResult, Result } from '@fgv/ts-utils';
2
- import { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';
3
- import { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';
2
+ import { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';
4
3
  import { JsonObject, JsonValue } from '../../common';
5
4
  import { JsonContext } from '../../jsonContext';
5
+ import { JsonEditorRuleBase } from '../jsonEditorRule';
6
+ import { JsonEditorState } from '../jsonEditorState';
6
7
  /**
7
8
  * The Reference JSON editor rule replaces property keys or values that match
8
9
  * some known object with a copy of that referenced object, formatted according
@@ -23,8 +23,8 @@
23
23
  Object.defineProperty(exports, "__esModule", { value: true });
24
24
  exports.ReferenceJsonEditorRule = void 0;
25
25
  const ts_utils_1 = require("@fgv/ts-utils");
26
- const jsonEditorRule_1 = require("../jsonEditorRule");
27
26
  const common_1 = require("../../common");
27
+ const jsonEditorRule_1 = require("../jsonEditorRule");
28
28
  /**
29
29
  * The Reference JSON editor rule replaces property keys or values that match
30
30
  * some known object with a copy of that referenced object, formatted according
@@ -155,4 +155,4 @@ class ReferenceJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
155
155
  }
156
156
  }
157
157
  exports.ReferenceJsonEditorRule = ReferenceJsonEditorRule;
158
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"references.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/references.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedResult, Result, captureResult, fail, failWithDetail, succeedWithDetail } from '@fgv/ts-utils';\nimport { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';\nimport { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';\nimport { JsonObject, JsonValue, isJsonObject, pickJsonObject } from '../../common';\nimport { JsonContext } from '../../jsonContext';\n\n/**\n * The Reference JSON editor rule replaces property keys or values that match\n * some known object with a copy of that referenced object, formatted according\n * to the current context.\n *\n * A property key is matched if it matches any known referenceable value.\n * - If the value of the matched key is 'default', then the entire object is formatted\n *   with the current context, flattened and merged into the current object.\n * - If the value of the matched key is some other string, then the entire\n *   object is formatted with the current context, and the child of the resulting\n *   object at the specified path is flattened and merged into the current object.\n * - If the value of the matched key is an object, then the entire object is\n *   formatted with the current context extended to include any properties of\n *   that object, flattened, and merged into the current object.\n * - It is an error if the referenced value is not an object.\n *\n * Any property, array or literal value is matched if it matches any known\n * referenceable value. The referenced value is replaced by the referenced\n * value, formatted using the current editor context.\n */\nexport class ReferenceJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: JsonEditorOptions;\n\n    /**\n     * Creates a new @see ReferenceJsonEditorRule.\n     * @param options Optional configuration options for this rule\n     */\n    public constructor(options?: JsonEditorOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new @see ReferenceJsonEditorRule.\n     * @param options Optional configuration options for this rule\n     */\n    public static create(options?: JsonEditorOptions): Result<ReferenceJsonEditorRule> {\n        return captureResult(() => new ReferenceJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property for reference expansion.\n     * @param key The key of the property to be considered\n     * @param value The value of the property to be considered\n     * @param state The editor state for the object being edited\n     * @returns If the reference is successful, returns Success with a JsonObject to\n     * be flattened and merged into the current object. Fails with detail 'inapplicable'\n     * for non-reference keys or with detail 'error' if an error occurs.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        // istanbul ignore next\n        const validation = this._options?.validation;\n        // istanbul ignore next\n        const refs = state.getRefs(this._options?.context);\n        if (refs?.has(key)) {\n            // need to apply any rules to the value before we evaluate it\n            const cloneResult = state.editor.clone(value, state.context);\n            if (cloneResult.isSuccess()) {\n                value = cloneResult.value;\n            }\n            else {\n                const message = `${key}: ${cloneResult.message}`;\n                return state.failValidation('invalidPropertyName', message, validation);\n            }\n\n            const contextResult = this._extendContext(state, value);\n            if (contextResult.isSuccess()) {\n                const objResult = refs.getJsonObject(key, contextResult.value);\n                // guarded by the has above so should never happen\n                // istanbul ignore else\n                if (objResult.isSuccess()) {\n                    if ((typeof value !== 'string') || (value === 'default')) {\n                        return succeedWithDetail<JsonObject, JsonEditFailureReason>(objResult.value, 'edited');\n                    }\n                    const pickResult = pickJsonObject(objResult.value, value);\n                    if (pickResult.isFailure()) {\n                        const message = `${key}: ${pickResult.message}`;\n                        return state.failValidation('invalidPropertyName', message, validation);\n                    }\n                    return pickResult.withDetail('edited');\n                }\n                else if (objResult.detail !== 'unknown') {\n                    const message = `${key}: ${objResult.message}`;\n                    return state.failValidation('invalidPropertyName', message, validation);\n                }\n            }\n            else {\n                const message = `${key}: ${contextResult.message}`;\n                return state.failValidation('invalidPropertyName', message, validation);\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    /**\n     * Evaluates a property, array or literal value for reference replacement.\n     * @param value The value to be evaluated\n     * @param state The editor state for the object being edited\n     */\n    public editValue(value: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        // istanbul ignore next\n        const refs = state.getRefs(this._options?.context);\n\n        if (refs && (typeof value === 'string')) {\n            // istanbul ignore next\n            const context = state.getContext(this._options?.context);\n            const result = refs.getJsonValue(value, context);\n            if (result.isSuccess()) {\n                return succeedWithDetail(result.value, 'edited');\n            }\n            else if (result.detail === 'error') {\n                return state.failValidation('invalidPropertyValue', result.message, this._options?.validation);\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    /**\n     * Gets the template variables to use given the value of some property whose name matched a\n     * resource plus the base template context.\n     * @param supplied The string or object supplied in the source json\n     * @param baseVars The context in effect at the point of resolution\n     */\n    protected _extendContext(state: JsonEditorState, supplied: JsonValue): Result<JsonContext|undefined> {\n        const add: Record<string, unknown> = {};\n        if (isJsonObject(supplied)) {\n            add.vars = Object.entries(supplied);\n        }\n        else if (typeof supplied !== 'string') {\n            return fail(`Invalid template path or context: \"${JSON.stringify(supplied)}\"`);\n        }\n        return state.extendContext(this._options?.context, add);\n    }\n}\n"]}
158
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"references.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/references.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;;;AAEH,4CAA+G;AAE/G,yCAAmF;AAEnF,sDAAuD;AAGvD;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAa,uBAAwB,SAAQ,mCAAkB;IAG3D;;;OAGG;IACH,YAAmB,OAA2B;QAC1C,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACI,MAAM,CAAC,MAAM,CAAC,OAA2B;QAC5C,OAAO,IAAA,wBAAa,EAAC,GAAG,EAAE,CAAC,IAAI,uBAAuB,CAAC,OAAO,CAAC,CAAC,CAAC;IACrE,CAAC;IAED;;;;;;;;OAQG;IACI,YAAY,CAAC,GAAW,EAAE,KAAgB,EAAE,KAAsB;;QACrE,uBAAuB;QACvB,MAAM,UAAU,GAAG,MAAA,IAAI,CAAC,QAAQ,0CAAE,UAAU,CAAC;QAC7C,uBAAuB;QACvB,MAAM,IAAI,GAAG,KAAK,CAAC,OAAO,CAAC,MAAA,IAAI,CAAC,QAAQ,0CAAE,OAAO,CAAC,CAAC;QACnD,IAAI,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,GAAG,CAAC,GAAG,CAAC,EAAE;YAChB,6DAA6D;YAC7D,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;YAC7D,IAAI,WAAW,CAAC,SAAS,EAAE,EAAE;gBACzB,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC;aAC7B;iBACI;gBACD,MAAM,OAAO,GAAG,GAAG,GAAG,KAAK,WAAW,CAAC,OAAO,EAAE,CAAC;gBACjD,OAAO,KAAK,CAAC,cAAc,CAAC,qBAAqB,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;aAC3E;YAED,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YACxD,IAAI,aAAa,CAAC,SAAS,EAAE,EAAE;gBAC3B,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,EAAE,aAAa,CAAC,KAAK,CAAC,CAAC;gBAC/D,kDAAkD;gBAClD,uBAAuB;gBACvB,IAAI,SAAS,CAAC,SAAS,EAAE,EAAE;oBACvB,IAAI,CAAC,OAAO,KAAK,KAAK,QAAQ,CAAC,IAAI,CAAC,KAAK,KAAK,SAAS,CAAC,EAAE;wBACtD,OAAO,IAAA,4BAAiB,EAAoC,SAAS,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;qBAC1F;oBACD,MAAM,UAAU,GAAG,IAAA,uBAAc,EAAC,SAAS,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;oBAC1D,IAAI,UAAU,CAAC,SAAS,EAAE,EAAE;wBACxB,MAAM,OAAO,GAAG,GAAG,GAAG,KAAK,UAAU,CAAC,OAAO,EAAE,CAAC;wBAChD,OAAO,KAAK,CAAC,cAAc,CAAC,qBAAqB,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;qBAC3E;oBACD,OAAO,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;iBAC1C;qBACI,IAAI,SAAS,CAAC,MAAM,KAAK,SAAS,EAAE;oBACrC,MAAM,OAAO,GAAG,GAAG,GAAG,KAAK,SAAS,CAAC,OAAO,EAAE,CAAC;oBAC/C,OAAO,KAAK,CAAC,cAAc,CAAC,qBAAqB,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;iBAC3E;aACJ;iBACI;gBACD,MAAM,OAAO,GAAG,GAAG,GAAG,KAAK,aAAa,CAAC,OAAO,EAAE,CAAC;gBACnD,OAAO,KAAK,CAAC,cAAc,CAAC,qBAAqB,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;aAC3E;SACJ;QACD,OAAO,IAAA,yBAAc,EAAC,cAAc,EAAE,cAAc,CAAC,CAAC;IAC1D,CAAC;IAED;;;;OAIG;IACI,SAAS,CAAC,KAAgB,EAAE,KAAsB;;QACrD,uBAAuB;QACvB,MAAM,IAAI,GAAG,KAAK,CAAC,OAAO,CAAC,MAAA,IAAI,CAAC,QAAQ,0CAAE,OAAO,CAAC,CAAC;QAEnD,IAAI,IAAI,IAAI,CAAC,OAAO,KAAK,KAAK,QAAQ,CAAC,EAAE;YACrC,uBAAuB;YACvB,MAAM,OAAO,GAAG,KAAK,CAAC,UAAU,CAAC,MAAA,IAAI,CAAC,QAAQ,0CAAE,OAAO,CAAC,CAAC;YACzD,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;YACjD,IAAI,MAAM,CAAC,SAAS,EAAE,EAAE;gBACpB,OAAO,IAAA,4BAAiB,EAAC,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;aACpD;iBACI,IAAI,MAAM,CAAC,MAAM,KAAK,OAAO,EAAE;gBAChC,OAAO,KAAK,CAAC,cAAc,CAAC,sBAAsB,EAAE,MAAM,CAAC,OAAO,EAAE,MAAA,IAAI,CAAC,QAAQ,0CAAE,UAAU,CAAC,CAAC;aAClG;SACJ;QACD,OAAO,IAAA,yBAAc,EAAC,cAAc,EAAE,cAAc,CAAC,CAAC;IAC1D,CAAC;IAED;;;;;OAKG;IACO,cAAc,CAAC,KAAsB,EAAE,QAAmB;;QAChE,MAAM,GAAG,GAA4B,EAAE,CAAC;QACxC,IAAI,IAAA,qBAAY,EAAC,QAAQ,CAAC,EAAE;YACxB,GAAG,CAAC,IAAI,GAAG,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;SACvC;aACI,IAAI,OAAO,QAAQ,KAAK,QAAQ,EAAE;YACnC,OAAO,IAAA,eAAI,EAAC,sCAAsC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;SAClF;QACD,OAAO,KAAK,CAAC,aAAa,CAAC,MAAA,IAAI,CAAC,QAAQ,0CAAE,OAAO,EAAE,GAAG,CAAC,CAAC;IAC5D,CAAC;CACJ;AAjHD,0DAiHC","sourcesContent":["/*\n * Copyright (c) 2020 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport { DetailedResult, Result, captureResult, fail, failWithDetail, succeedWithDetail } from '@fgv/ts-utils';\nimport { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';\nimport { JsonObject, JsonValue, isJsonObject, pickJsonObject } from '../../common';\nimport { JsonContext } from '../../jsonContext';\nimport { JsonEditorRuleBase } from '../jsonEditorRule';\nimport { JsonEditorState } from '../jsonEditorState';\n\n/**\n * The Reference JSON editor rule replaces property keys or values that match\n * some known object with a copy of that referenced object, formatted according\n * to the current context.\n *\n * A property key is matched if it matches any known referenceable value.\n * - If the value of the matched key is 'default', then the entire object is formatted\n *   with the current context, flattened and merged into the current object.\n * - If the value of the matched key is some other string, then the entire\n *   object is formatted with the current context, and the child of the resulting\n *   object at the specified path is flattened and merged into the current object.\n * - If the value of the matched key is an object, then the entire object is\n *   formatted with the current context extended to include any properties of\n *   that object, flattened, and merged into the current object.\n * - It is an error if the referenced value is not an object.\n *\n * Any property, array or literal value is matched if it matches any known\n * referenceable value. The referenced value is replaced by the referenced\n * value, formatted using the current editor context.\n */\nexport class ReferenceJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: JsonEditorOptions;\n\n    /**\n     * Creates a new @see ReferenceJsonEditorRule.\n     * @param options Optional configuration options for this rule\n     */\n    public constructor(options?: JsonEditorOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new @see ReferenceJsonEditorRule.\n     * @param options Optional configuration options for this rule\n     */\n    public static create(options?: JsonEditorOptions): Result<ReferenceJsonEditorRule> {\n        return captureResult(() => new ReferenceJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property for reference expansion.\n     * @param key The key of the property to be considered\n     * @param value The value of the property to be considered\n     * @param state The editor state for the object being edited\n     * @returns If the reference is successful, returns Success with a JsonObject to\n     * be flattened and merged into the current object. Fails with detail 'inapplicable'\n     * for non-reference keys or with detail 'error' if an error occurs.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        // istanbul ignore next\n        const validation = this._options?.validation;\n        // istanbul ignore next\n        const refs = state.getRefs(this._options?.context);\n        if (refs?.has(key)) {\n            // need to apply any rules to the value before we evaluate it\n            const cloneResult = state.editor.clone(value, state.context);\n            if (cloneResult.isSuccess()) {\n                value = cloneResult.value;\n            }\n            else {\n                const message = `${key}: ${cloneResult.message}`;\n                return state.failValidation('invalidPropertyName', message, validation);\n            }\n\n            const contextResult = this._extendContext(state, value);\n            if (contextResult.isSuccess()) {\n                const objResult = refs.getJsonObject(key, contextResult.value);\n                // guarded by the has above so should never happen\n                // istanbul ignore else\n                if (objResult.isSuccess()) {\n                    if ((typeof value !== 'string') || (value === 'default')) {\n                        return succeedWithDetail<JsonObject, JsonEditFailureReason>(objResult.value, 'edited');\n                    }\n                    const pickResult = pickJsonObject(objResult.value, value);\n                    if (pickResult.isFailure()) {\n                        const message = `${key}: ${pickResult.message}`;\n                        return state.failValidation('invalidPropertyName', message, validation);\n                    }\n                    return pickResult.withDetail('edited');\n                }\n                else if (objResult.detail !== 'unknown') {\n                    const message = `${key}: ${objResult.message}`;\n                    return state.failValidation('invalidPropertyName', message, validation);\n                }\n            }\n            else {\n                const message = `${key}: ${contextResult.message}`;\n                return state.failValidation('invalidPropertyName', message, validation);\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    /**\n     * Evaluates a property, array or literal value for reference replacement.\n     * @param value The value to be evaluated\n     * @param state The editor state for the object being edited\n     */\n    public editValue(value: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        // istanbul ignore next\n        const refs = state.getRefs(this._options?.context);\n\n        if (refs && (typeof value === 'string')) {\n            // istanbul ignore next\n            const context = state.getContext(this._options?.context);\n            const result = refs.getJsonValue(value, context);\n            if (result.isSuccess()) {\n                return succeedWithDetail(result.value, 'edited');\n            }\n            else if (result.detail === 'error') {\n                return state.failValidation('invalidPropertyValue', result.message, this._options?.validation);\n            }\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    /**\n     * Gets the template variables to use given the value of some property whose name matched a\n     * resource plus the base template context.\n     * @param supplied The string or object supplied in the source json\n     * @param baseVars The context in effect at the point of resolution\n     */\n    protected _extendContext(state: JsonEditorState, supplied: JsonValue): Result<JsonContext|undefined> {\n        const add: Record<string, unknown> = {};\n        if (isJsonObject(supplied)) {\n            add.vars = Object.entries(supplied);\n        }\n        else if (typeof supplied !== 'string') {\n            return fail(`Invalid template path or context: \"${JSON.stringify(supplied)}\"`);\n        }\n        return state.extendContext(this._options?.context, add);\n    }\n}\n"]}
@@ -1,7 +1,8 @@
1
1
  import { DetailedResult, Result } from '@fgv/ts-utils';
2
- import { JsonEditFailureReason, JsonEditorRuleBase, JsonPropertyEditFailureReason } from '../jsonEditorRule';
3
- import { JsonEditorOptions, JsonEditorState } from '../jsonEditorState';
2
+ import { JsonEditFailureReason, JsonEditorOptions, JsonPropertyEditFailureReason } from '../common';
4
3
  import { JsonObject, JsonValue } from '../../common';
4
+ import { JsonEditorRuleBase } from '../jsonEditorRule';
5
+ import { JsonEditorState } from '../jsonEditorState';
5
6
  /**
6
7
  * Configuration options for the Templated JSON editor rule
7
8
  */