@fgv/ts-json 1.0.1 → 1.8.1
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.
- package/LICENSE +0 -0
- package/README.md +0 -0
- package/common.d.ts +3 -3
- package/common.js +10 -10
- package/contextHelpers.d.ts +0 -0
- package/contextHelpers.js +10 -10
- package/converters.d.ts +0 -0
- package/converters.js +0 -0
- package/file.d.ts +1 -1
- package/file.js +16 -12
- package/index.d.ts +0 -0
- package/index.js +6 -2
- package/jsonContext.d.ts +4 -4
- package/jsonContext.js +2 -2
- package/jsonConverter.d.ts +5 -5
- package/jsonConverter.js +16 -16
- package/jsonEditor/index.d.ts +0 -0
- package/jsonEditor/index.js +6 -2
- package/jsonEditor/jsonEditor.d.ts +5 -5
- package/jsonEditor/jsonEditor.js +43 -43
- package/jsonEditor/jsonEditorRule.d.ts +2 -2
- package/jsonEditor/jsonEditorRule.js +4 -4
- package/jsonEditor/jsonEditorState.d.ts +3 -3
- package/jsonEditor/jsonEditorState.js +9 -9
- package/jsonEditor/rules/conditional.d.ts +0 -0
- package/jsonEditor/rules/conditional.js +13 -13
- package/jsonEditor/rules/index.d.ts +0 -0
- package/jsonEditor/rules/index.js +6 -2
- package/jsonEditor/rules/multivalue.d.ts +0 -0
- package/jsonEditor/rules/multivalue.js +9 -9
- package/jsonEditor/rules/references.d.ts +0 -0
- package/jsonEditor/rules/references.js +9 -9
- package/jsonEditor/rules/templates.d.ts +0 -0
- package/jsonEditor/rules/templates.js +8 -8
- package/jsonReferenceMap.d.ts +1 -1
- package/jsonReferenceMap.js +25 -25
- package/package.json +14 -14
|
@@ -59,7 +59,7 @@ class ReferenceJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
59
59
|
* @param options Optional configuration options for this rule
|
|
60
60
|
*/
|
|
61
61
|
static create(options) {
|
|
62
|
-
return ts_utils_1.captureResult(() => new ReferenceJsonEditorRule(options));
|
|
62
|
+
return (0, ts_utils_1.captureResult)(() => new ReferenceJsonEditorRule(options));
|
|
63
63
|
}
|
|
64
64
|
/**
|
|
65
65
|
* Evaluates a property for reference expansion.
|
|
@@ -93,9 +93,9 @@ class ReferenceJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
93
93
|
// istanbul ignore else
|
|
94
94
|
if (objResult.isSuccess()) {
|
|
95
95
|
if ((typeof value !== 'string') || (value === 'default')) {
|
|
96
|
-
return ts_utils_1.succeedWithDetail(objResult.value, 'edited');
|
|
96
|
+
return (0, ts_utils_1.succeedWithDetail)(objResult.value, 'edited');
|
|
97
97
|
}
|
|
98
|
-
const pickResult = common_1.pickJsonObject(objResult.value, value);
|
|
98
|
+
const pickResult = (0, common_1.pickJsonObject)(objResult.value, value);
|
|
99
99
|
if (pickResult.isFailure()) {
|
|
100
100
|
const message = `${key}: ${pickResult.message}`;
|
|
101
101
|
return state.failValidation('invalidPropertyName', message, validation);
|
|
@@ -112,7 +112,7 @@ class ReferenceJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
112
112
|
return state.failValidation('invalidPropertyName', message, validation);
|
|
113
113
|
}
|
|
114
114
|
}
|
|
115
|
-
return ts_utils_1.failWithDetail('inapplicable', 'inapplicable');
|
|
115
|
+
return (0, ts_utils_1.failWithDetail)('inapplicable', 'inapplicable');
|
|
116
116
|
}
|
|
117
117
|
/**
|
|
118
118
|
* Evaluates a property, array or literal value for reference replacement.
|
|
@@ -128,13 +128,13 @@ class ReferenceJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
128
128
|
const context = state.getContext((_b = this._options) === null || _b === void 0 ? void 0 : _b.context);
|
|
129
129
|
const result = refs.getJsonValue(value, context);
|
|
130
130
|
if (result.isSuccess()) {
|
|
131
|
-
return ts_utils_1.succeedWithDetail(result.value, 'edited');
|
|
131
|
+
return (0, ts_utils_1.succeedWithDetail)(result.value, 'edited');
|
|
132
132
|
}
|
|
133
133
|
else if (result.detail === 'error') {
|
|
134
134
|
return state.failValidation('invalidPropertyValue', result.message, (_c = this._options) === null || _c === void 0 ? void 0 : _c.validation);
|
|
135
135
|
}
|
|
136
136
|
}
|
|
137
|
-
return ts_utils_1.failWithDetail('inapplicable', 'inapplicable');
|
|
137
|
+
return (0, ts_utils_1.failWithDetail)('inapplicable', 'inapplicable');
|
|
138
138
|
}
|
|
139
139
|
/**
|
|
140
140
|
* Gets the template variables to use given the value of some property whose name matched a
|
|
@@ -145,14 +145,14 @@ class ReferenceJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
145
145
|
_extendContext(state, supplied) {
|
|
146
146
|
var _a;
|
|
147
147
|
const add = {};
|
|
148
|
-
if (common_1.isJsonObject(supplied)) {
|
|
148
|
+
if ((0, common_1.isJsonObject)(supplied)) {
|
|
149
149
|
add.vars = Object.entries(supplied);
|
|
150
150
|
}
|
|
151
151
|
else if (typeof supplied !== 'string') {
|
|
152
|
-
return ts_utils_1.fail(`Invalid template path or context: "${JSON.stringify(supplied)}"`);
|
|
152
|
+
return (0, ts_utils_1.fail)(`Invalid template path or context: "${JSON.stringify(supplied)}"`);
|
|
153
153
|
}
|
|
154
154
|
return state.extendContext((_a = this._options) === null || _a === void 0 ? void 0 : _a.context, add);
|
|
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":"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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"]}
|
|
File without changes
|
|
@@ -46,7 +46,7 @@ class TemplatedJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
46
46
|
* @param options Optional configuration options for this rule
|
|
47
47
|
*/
|
|
48
48
|
static create(options) {
|
|
49
|
-
return ts_utils_1.captureResult(() => new TemplatedJsonEditorRule(options));
|
|
49
|
+
return (0, ts_utils_1.captureResult)(() => new TemplatedJsonEditorRule(options));
|
|
50
50
|
}
|
|
51
51
|
/**
|
|
52
52
|
* Evaluates a property name for template rendering.
|
|
@@ -69,7 +69,7 @@ class TemplatedJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
69
69
|
}
|
|
70
70
|
const rtrn = {};
|
|
71
71
|
rtrn[newKey] = value;
|
|
72
|
-
return ts_utils_1.succeedWithDetail(rtrn, 'edited');
|
|
72
|
+
return (0, ts_utils_1.succeedWithDetail)(rtrn, 'edited');
|
|
73
73
|
});
|
|
74
74
|
if ((result.isFailure() && result.detail === 'error')) {
|
|
75
75
|
const message = `Cannot render name ${key}: ${result.message}`;
|
|
@@ -77,7 +77,7 @@ class TemplatedJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
77
77
|
}
|
|
78
78
|
return result;
|
|
79
79
|
}
|
|
80
|
-
return ts_utils_1.failWithDetail('inapplicable', 'inapplicable');
|
|
80
|
+
return (0, ts_utils_1.failWithDetail)('inapplicable', 'inapplicable');
|
|
81
81
|
}
|
|
82
82
|
/**
|
|
83
83
|
* Evaluates a property, array or literal value for template rendering
|
|
@@ -91,7 +91,7 @@ class TemplatedJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
91
91
|
var _a, _b;
|
|
92
92
|
if ((((_a = this._options) === null || _a === void 0 ? void 0 : _a.useValueTemplates) !== false) && (typeof value === 'string') && value.includes('{{')) {
|
|
93
93
|
const renderResult = this._render(value, state).onSuccess((newValue) => {
|
|
94
|
-
return ts_utils_1.succeedWithDetail(newValue, 'edited');
|
|
94
|
+
return (0, ts_utils_1.succeedWithDetail)(newValue, 'edited');
|
|
95
95
|
});
|
|
96
96
|
if (renderResult.isFailure() && (renderResult.detail === 'error')) {
|
|
97
97
|
const message = `Cannot render value: ${renderResult.message}`;
|
|
@@ -100,16 +100,16 @@ class TemplatedJsonEditorRule extends jsonEditorRule_1.JsonEditorRuleBase {
|
|
|
100
100
|
}
|
|
101
101
|
return renderResult;
|
|
102
102
|
}
|
|
103
|
-
return ts_utils_1.failWithDetail('inapplicable', 'inapplicable');
|
|
103
|
+
return (0, ts_utils_1.failWithDetail)('inapplicable', 'inapplicable');
|
|
104
104
|
}
|
|
105
105
|
_render(template, state) {
|
|
106
106
|
var _a;
|
|
107
107
|
const vars = state.getVars((_a = this._options) === null || _a === void 0 ? void 0 : _a.context);
|
|
108
108
|
if (vars && template.includes('{{')) {
|
|
109
|
-
return ts_utils_1.captureResult(() => mustache_1.default.render(template, vars)).withDetail('error', 'edited');
|
|
109
|
+
return (0, ts_utils_1.captureResult)(() => mustache_1.default.render(template, vars)).withDetail('error', 'edited');
|
|
110
110
|
}
|
|
111
|
-
return ts_utils_1.failWithDetail('inapplicable', 'inapplicable');
|
|
111
|
+
return (0, ts_utils_1.failWithDetail)('inapplicable', 'inapplicable');
|
|
112
112
|
}
|
|
113
113
|
}
|
|
114
114
|
exports.TemplatedJsonEditorRule = TemplatedJsonEditorRule;
|
|
115
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"templates.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/templates.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 } from '../../common';\n\nimport Mustache from 'mustache';\n\n/**\n * Configuration options for the Templated JSON editor rule\n */\nexport interface TemplatedJsonRuleOptions extends Partial<JsonEditorOptions> {\n    /**\n     * If true (default) then templates in property names are rendered\n     */\n    useNameTemplates?: boolean;\n    /**\n     * If true (default) then templates in property values are rendered\n     */\n    useValueTemplates?: boolean;\n}\n\n/**\n * The Templated JSON editor rule applies mustache rendering as appropriate\n * to any keys or values in the object being edited.\n */\nexport class TemplatedJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: TemplatedJsonRuleOptions;\n\n    /**\n     * Creates a new @see TemplatedJsonEditorRule\n     * @param options Optional configuration options for this rule\n     */\n    public constructor(options?: TemplatedJsonRuleOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new @see TemplatedJsonEditorRule\n     * @param options Optional configuration options for this rule\n     */\n    public static create(options?: TemplatedJsonRuleOptions): Result<TemplatedJsonEditorRule> {\n        return captureResult(() => new TemplatedJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property name for template rendering.\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 Succeeds with detail 'edited' and an object to be flattened and merged\n     * if the key contained a template. Fails with detail 'error' if an error occurred\n     * or with detail 'inapplicable' if the property key does not contain a template\n     * or if name rendering is disabled.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        // istanbul ignore next\n        const validation = this._options?.validation;\n\n        if (this._options?.useNameTemplates !== false) {\n            const result = this._render(key, state).onSuccess((newKey) => {\n                if (newKey.length < 1) {\n                    return state.failValidation('invalidPropertyName', `Template \"${key}\" renders empty name.`);\n                }\n\n                const rtrn: JsonObject = {};\n                rtrn[newKey] = value;\n                return succeedWithDetail<JsonObject, JsonEditFailureReason>(rtrn, 'edited');\n            });\n\n            if ((result.isFailure() && result.detail === 'error')) {\n                const message = `Cannot render name ${key}: ${result.message}`;\n                return state.failValidation('invalidPropertyName', message, validation);\n            }\n            return result;\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    /**\n     * Evaluates a property, array or literal value for template rendering\n     * @param value The value to be edited\n     * @param state The editor state for the object being edited\n     * @returns Succeeds with detail 'edited' if the value contained a template and was edited.\n     * Fails with 'ignore' if the rendered value should be ignored, with 'error' if an error occurs\n     * or with 'inapplicable' if the value was not a string with a template.\n     */\n    public editValue(value: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        if ((this._options?.useValueTemplates !== false) && (typeof value === 'string') && value.includes('{{')) {\n            const renderResult = this._render(value, state).onSuccess((newValue) => {\n                return succeedWithDetail(newValue, 'edited');\n            });\n\n            if (renderResult.isFailure() && (renderResult.detail === 'error')) {\n                const message = `Cannot render value: ${renderResult.message}`;\n                // istanbul ignore next\n                return state.failValidation('invalidPropertyValue', message, this._options?.validation);\n            }\n            return renderResult;\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _render(template: string, state: JsonEditorState): DetailedResult<string, JsonEditFailureReason> {\n        const vars = state.getVars(this._options?.context);\n        if (vars && template.includes('{{')) {\n            return captureResult(() => Mustache.render(template, vars)).withDetail('error', 'edited');\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n}\n"]}
|
|
115
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"templates.js","sourceRoot":"","sources":["../../../src/jsonEditor/rules/templates.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 } from '../../common';\n\nimport Mustache from 'mustache';\n\n/**\n * Configuration options for the Templated JSON editor rule\n */\nexport interface TemplatedJsonRuleOptions extends Partial<JsonEditorOptions> {\n    /**\n     * If true (default) then templates in property names are rendered\n     */\n    useNameTemplates?: boolean;\n    /**\n     * If true (default) then templates in property values are rendered\n     */\n    useValueTemplates?: boolean;\n}\n\n/**\n * The Templated JSON editor rule applies mustache rendering as appropriate\n * to any keys or values in the object being edited.\n */\nexport class TemplatedJsonEditorRule extends JsonEditorRuleBase {\n    protected _options?: TemplatedJsonRuleOptions;\n\n    /**\n     * Creates a new @see TemplatedJsonEditorRule\n     * @param options Optional configuration options for this rule\n     */\n    public constructor(options?: TemplatedJsonRuleOptions) {\n        super();\n        this._options = options;\n    }\n\n    /**\n     * Creates a new @see TemplatedJsonEditorRule\n     * @param options Optional configuration options for this rule\n     */\n    public static create(options?: TemplatedJsonRuleOptions): Result<TemplatedJsonEditorRule> {\n        return captureResult(() => new TemplatedJsonEditorRule(options));\n    }\n\n    /**\n     * Evaluates a property name for template rendering.\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 Succeeds with detail 'edited' and an object to be flattened and merged\n     * if the key contained a template. Fails with detail 'error' if an error occurred\n     * or with detail 'inapplicable' if the property key does not contain a template\n     * or if name rendering is disabled.\n     */\n    public editProperty(key: string, value: JsonValue, state: JsonEditorState): DetailedResult<JsonObject, JsonPropertyEditFailureReason> {\n        // istanbul ignore next\n        const validation = this._options?.validation;\n\n        if (this._options?.useNameTemplates !== false) {\n            const result = this._render(key, state).onSuccess((newKey) => {\n                if (newKey.length < 1) {\n                    return state.failValidation('invalidPropertyName', `Template \"${key}\" renders empty name.`);\n                }\n\n                const rtrn: JsonObject = {};\n                rtrn[newKey] = value;\n                return succeedWithDetail<JsonObject, JsonEditFailureReason>(rtrn, 'edited');\n            });\n\n            if ((result.isFailure() && result.detail === 'error')) {\n                const message = `Cannot render name ${key}: ${result.message}`;\n                return state.failValidation('invalidPropertyName', message, validation);\n            }\n            return result;\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    /**\n     * Evaluates a property, array or literal value for template rendering\n     * @param value The value to be edited\n     * @param state The editor state for the object being edited\n     * @returns Succeeds with detail 'edited' if the value contained a template and was edited.\n     * Fails with 'ignore' if the rendered value should be ignored, with 'error' if an error occurs\n     * or with 'inapplicable' if the value was not a string with a template.\n     */\n    public editValue(value: JsonValue, state: JsonEditorState): DetailedResult<JsonValue, JsonEditFailureReason> {\n        if ((this._options?.useValueTemplates !== false) && (typeof value === 'string') && value.includes('{{')) {\n            const renderResult = this._render(value, state).onSuccess((newValue) => {\n                return succeedWithDetail(newValue, 'edited');\n            });\n\n            if (renderResult.isFailure() && (renderResult.detail === 'error')) {\n                const message = `Cannot render value: ${renderResult.message}`;\n                // istanbul ignore next\n                return state.failValidation('invalidPropertyValue', message, this._options?.validation);\n            }\n            return renderResult;\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n\n    protected _render(template: string, state: JsonEditorState): DetailedResult<string, JsonEditFailureReason> {\n        const vars = state.getVars(this._options?.context);\n        if (vars && template.includes('{{')) {\n            return captureResult(() => Mustache.render(template, vars)).withDetail('error', 'edited');\n        }\n        return failWithDetail('inapplicable', 'inapplicable');\n    }\n}\n"]}
|
package/jsonReferenceMap.d.ts
CHANGED
|
@@ -25,7 +25,7 @@ export declare class PrefixKeyPolicy<T> extends ReferenceMapKeyPolicy<T> {
|
|
|
25
25
|
isValid(key: string, _item?: T): boolean;
|
|
26
26
|
validate(key: string, item?: T, options?: ReferenceMapKeyPolicyValidateOptions): Result<string>;
|
|
27
27
|
}
|
|
28
|
-
export
|
|
28
|
+
export type MapOrRecord<T> = Map<string, T> | Record<string, T>;
|
|
29
29
|
/**
|
|
30
30
|
* A SimpleJsonMap presents a view of a simple map of @see JsonValue
|
|
31
31
|
*/
|
package/jsonReferenceMap.js
CHANGED
|
@@ -41,18 +41,18 @@ class ReferenceMapKeyPolicy {
|
|
|
41
41
|
return this._isValid(key, item);
|
|
42
42
|
}
|
|
43
43
|
validate(key, item, _options) {
|
|
44
|
-
return this.isValid(key, item) ? ts_utils_1.succeed(key) : ts_utils_1.fail(`${key}: invalid key`);
|
|
44
|
+
return this.isValid(key, item) ? (0, ts_utils_1.succeed)(key) : (0, ts_utils_1.fail)(`${key}: invalid key`);
|
|
45
45
|
}
|
|
46
46
|
validateItems(items, options) {
|
|
47
|
-
return ts_utils_1.mapResults(items.map((item) => {
|
|
47
|
+
return (0, ts_utils_1.mapResults)(items.map((item) => {
|
|
48
48
|
return this.validate(...item, options).onSuccess((valid) => {
|
|
49
|
-
return ts_utils_1.succeed([valid, item[1]]);
|
|
49
|
+
return (0, ts_utils_1.succeed)([valid, item[1]]);
|
|
50
50
|
});
|
|
51
51
|
}));
|
|
52
52
|
}
|
|
53
53
|
validateMap(map, options) {
|
|
54
54
|
return this.validateItems(Array.from(map.entries()), options).onSuccess((valid) => {
|
|
55
|
-
return ts_utils_1.captureResult(() => new Map(valid));
|
|
55
|
+
return (0, ts_utils_1.captureResult)(() => new Map(valid));
|
|
56
56
|
});
|
|
57
57
|
}
|
|
58
58
|
}
|
|
@@ -70,12 +70,12 @@ class PrefixKeyPolicy extends ReferenceMapKeyPolicy {
|
|
|
70
70
|
// istanbul ignore next
|
|
71
71
|
const makeValid = ((_a = (options !== null && options !== void 0 ? options : this._defaultOptions)) === null || _a === void 0 ? void 0 : _a.makeValid) === true;
|
|
72
72
|
if (this.isValid(key, item)) {
|
|
73
|
-
return ts_utils_1.succeed(key);
|
|
73
|
+
return (0, ts_utils_1.succeed)(key);
|
|
74
74
|
}
|
|
75
75
|
else if (makeValid && ReferenceMapKeyPolicy.defaultKeyPredicate(key)) {
|
|
76
|
-
return ts_utils_1.succeed(`${this.prefix}${key}`);
|
|
76
|
+
return (0, ts_utils_1.succeed)(`${this.prefix}${key}`);
|
|
77
77
|
}
|
|
78
|
-
return ts_utils_1.fail(`${key}: invalid key`);
|
|
78
|
+
return (0, ts_utils_1.fail)(`${key}: invalid key`);
|
|
79
79
|
}
|
|
80
80
|
}
|
|
81
81
|
exports.PrefixKeyPolicy = PrefixKeyPolicy;
|
|
@@ -84,19 +84,19 @@ exports.PrefixKeyPolicy = PrefixKeyPolicy;
|
|
|
84
84
|
*/
|
|
85
85
|
class SimpleJsonMapBase {
|
|
86
86
|
constructor(values, context, keyPolicy) {
|
|
87
|
-
values = SimpleJsonMapBase._toMap(values).
|
|
87
|
+
values = SimpleJsonMapBase._toMap(values).orThrow();
|
|
88
88
|
this._keyPolicy = keyPolicy !== null && keyPolicy !== void 0 ? keyPolicy : new ReferenceMapKeyPolicy();
|
|
89
|
-
this._values = this._keyPolicy.validateMap(values).
|
|
89
|
+
this._values = this._keyPolicy.validateMap(values).orThrow();
|
|
90
90
|
this._context = context;
|
|
91
91
|
}
|
|
92
92
|
static _toMap(values) {
|
|
93
93
|
if (values === undefined) {
|
|
94
|
-
return ts_utils_1.captureResult(() => new Map());
|
|
94
|
+
return (0, ts_utils_1.captureResult)(() => new Map());
|
|
95
95
|
}
|
|
96
96
|
else if (!(values instanceof Map)) {
|
|
97
|
-
return ts_utils_1.recordToMap(values, (_k, v) => ts_utils_1.succeed(v));
|
|
97
|
+
return (0, ts_utils_1.recordToMap)(values, (_k, v) => (0, ts_utils_1.succeed)(v));
|
|
98
98
|
}
|
|
99
|
-
return ts_utils_1.succeed(values);
|
|
99
|
+
return (0, ts_utils_1.succeed)(values);
|
|
100
100
|
}
|
|
101
101
|
/**
|
|
102
102
|
* Determine if a key might be valid for this map but does not determine if key actually
|
|
@@ -125,10 +125,10 @@ class SimpleJsonMapBase {
|
|
|
125
125
|
*/
|
|
126
126
|
getJsonObject(key, context) {
|
|
127
127
|
return this.getJsonValue(key, context).onSuccess((jv) => {
|
|
128
|
-
if (!common_1.isJsonObject(jv)) {
|
|
129
|
-
return ts_utils_1.failWithDetail(`${key}: not an object`, 'error');
|
|
128
|
+
if (!(0, common_1.isJsonObject)(jv)) {
|
|
129
|
+
return (0, ts_utils_1.failWithDetail)(`${key}: not an object`, 'error');
|
|
130
130
|
}
|
|
131
|
-
return ts_utils_1.succeedWithDetail(jv);
|
|
131
|
+
return (0, ts_utils_1.succeedWithDetail)(jv);
|
|
132
132
|
});
|
|
133
133
|
}
|
|
134
134
|
}
|
|
@@ -148,7 +148,7 @@ class SimpleJsonMap extends SimpleJsonMapBase {
|
|
|
148
148
|
* @param options Optional @see SimpleJsonMapOptions for initialization
|
|
149
149
|
*/
|
|
150
150
|
static createSimple(values, context, options) {
|
|
151
|
-
return ts_utils_1.captureResult(() => new SimpleJsonMap(values, context, options));
|
|
151
|
+
return (0, ts_utils_1.captureResult)(() => new SimpleJsonMap(values, context, options));
|
|
152
152
|
}
|
|
153
153
|
/**
|
|
154
154
|
* Gets a JSON value specified by key.
|
|
@@ -163,7 +163,7 @@ class SimpleJsonMap extends SimpleJsonMapBase {
|
|
|
163
163
|
context = context !== null && context !== void 0 ? context : this._context;
|
|
164
164
|
const value = this._values.get(key);
|
|
165
165
|
if (!value) {
|
|
166
|
-
return ts_utils_1.failWithDetail(`${key}: JSON value not found`, 'unknown');
|
|
166
|
+
return (0, ts_utils_1.failWithDetail)(`${key}: JSON value not found`, 'unknown');
|
|
167
167
|
}
|
|
168
168
|
return this._clone(value, context);
|
|
169
169
|
}
|
|
@@ -172,7 +172,7 @@ class SimpleJsonMap extends SimpleJsonMapBase {
|
|
|
172
172
|
const result = jsonEditor_1.JsonEditor.create();
|
|
173
173
|
// istanbul ignore next: nearly impossible to reproduce
|
|
174
174
|
if (result.isFailure()) {
|
|
175
|
-
return ts_utils_1.failWithDetail(result.message, 'error');
|
|
175
|
+
return (0, ts_utils_1.failWithDetail)(result.message, 'error');
|
|
176
176
|
}
|
|
177
177
|
this._editor = result.value;
|
|
178
178
|
}
|
|
@@ -189,7 +189,7 @@ class PrefixedJsonMap extends SimpleJsonMap {
|
|
|
189
189
|
super(values, context, options);
|
|
190
190
|
}
|
|
191
191
|
static createPrefixed(prefixOptions, values, context, editor) {
|
|
192
|
-
return ts_utils_1.captureResult(() => new PrefixedJsonMap(values, context, { keyPolicy: this._toPolicy(prefixOptions), editor }));
|
|
192
|
+
return (0, ts_utils_1.captureResult)(() => new PrefixedJsonMap(values, context, { keyPolicy: this._toPolicy(prefixOptions), editor }));
|
|
193
193
|
}
|
|
194
194
|
static _toPolicy(prefixOptions) {
|
|
195
195
|
if (typeof prefixOptions === 'string') {
|
|
@@ -212,7 +212,7 @@ class CompositeJsonMap {
|
|
|
212
212
|
* @param maps one or more object maps to be composed
|
|
213
213
|
*/
|
|
214
214
|
static create(maps) {
|
|
215
|
-
return ts_utils_1.captureResult(() => new CompositeJsonMap(maps));
|
|
215
|
+
return (0, ts_utils_1.captureResult)(() => new CompositeJsonMap(maps));
|
|
216
216
|
}
|
|
217
217
|
/**
|
|
218
218
|
* Determine if a key might be valid for this map but does not determine
|
|
@@ -241,10 +241,10 @@ class CompositeJsonMap {
|
|
|
241
241
|
*/
|
|
242
242
|
getJsonObject(key, context) {
|
|
243
243
|
return this.getJsonValue(key, context).onSuccess((jv) => {
|
|
244
|
-
if (!common_1.isJsonObject(jv)) {
|
|
245
|
-
return ts_utils_1.failWithDetail(`${key}: not an object`, 'error');
|
|
244
|
+
if (!(0, common_1.isJsonObject)(jv)) {
|
|
245
|
+
return (0, ts_utils_1.failWithDetail)(`${key}: not an object`, 'error');
|
|
246
246
|
}
|
|
247
|
-
return ts_utils_1.succeedWithDetail(jv);
|
|
247
|
+
return (0, ts_utils_1.succeedWithDetail)(jv);
|
|
248
248
|
});
|
|
249
249
|
}
|
|
250
250
|
/**
|
|
@@ -265,8 +265,8 @@ class CompositeJsonMap {
|
|
|
265
265
|
}
|
|
266
266
|
}
|
|
267
267
|
}
|
|
268
|
-
return ts_utils_1.failWithDetail(`${key}: value not found`, 'unknown');
|
|
268
|
+
return (0, ts_utils_1.failWithDetail)(`${key}: value not found`, 'unknown');
|
|
269
269
|
}
|
|
270
270
|
}
|
|
271
271
|
exports.CompositeJsonMap = CompositeJsonMap;
|
|
272
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsonReferenceMap.js","sourceRoot":"","sources":["../src/jsonReferenceMap.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    DetailedResult,\n    Result,\n    captureResult,\n    fail,\n    failWithDetail,\n    mapResults,\n    recordToMap,\n    succeed,\n    succeedWithDetail,\n} from '@fgv/ts-utils';\n\nimport { JsonContext, JsonReferenceMap, JsonReferenceMapFailureReason } from './jsonContext';\nimport { JsonObject, JsonValue, isJsonObject } from './common';\nimport { JsonEditor } from './jsonEditor';\n\nexport interface ReferenceMapKeyPolicyValidateOptions {\n    makeValid?: boolean;\n}\n\n/**\n * Policy object responsible for validating or correcting\n * keys in a reference map\n */\nexport class ReferenceMapKeyPolicy<T> {\n    protected readonly _defaultOptions?: ReferenceMapKeyPolicyValidateOptions;\n    protected readonly _isValid: (key: string, item?: T) => boolean;\n\n    public constructor(options?: ReferenceMapKeyPolicyValidateOptions, isValid?: (key: string, item?: T) => boolean) {\n        this._defaultOptions = options;\n        this._isValid = isValid ?? ReferenceMapKeyPolicy.defaultKeyPredicate;\n    }\n\n    public static defaultKeyPredicate(key: string): boolean {\n        return (key.length > 0) && (!key.includes('{{')) && (!key.startsWith('?'));\n    }\n\n    public isValid(key: string, item?: T): boolean {\n        return this._isValid(key, item);\n    }\n\n    public validate(key: string, item?: T, _options?: ReferenceMapKeyPolicyValidateOptions): Result<string> {\n        return this.isValid(key, item) ? succeed(key) : fail(`${key}: invalid key`);\n    }\n\n    public validateItems(items: [string, T][], options?: ReferenceMapKeyPolicyValidateOptions): Result<[string, T][]> {\n        return mapResults(items.map((item) => {\n            return this.validate(...item, options).onSuccess((valid) => {\n                return succeed([valid, item[1]]);\n            });\n        }));\n    }\n\n    public validateMap(map: Map<string, T>, options?: ReferenceMapKeyPolicyValidateOptions): Result<Map<string, T>> {\n        return this.validateItems(Array.from(map.entries()), options).onSuccess((valid) => {\n            return captureResult(() => new Map(valid));\n        });\n    }\n}\n\nexport class PrefixKeyPolicy<T> extends ReferenceMapKeyPolicy<T> {\n    public readonly prefix: string;\n\n    public constructor(prefix: string, options?: ReferenceMapKeyPolicyValidateOptions) {\n        super(options);\n        this.prefix = prefix;\n    }\n\n    public isValid(key: string, _item?: T): boolean {\n        return key.startsWith(this.prefix) && (key !== this.prefix) && ReferenceMapKeyPolicy.defaultKeyPredicate(key);\n    }\n\n    public validate(key: string, item?: T, options?: ReferenceMapKeyPolicyValidateOptions): Result<string> {\n        // istanbul ignore next\n        const makeValid = (options ?? this._defaultOptions)?.makeValid === true;\n        if (this.isValid(key, item)) {\n            return succeed(key);\n        }\n        else if (makeValid && ReferenceMapKeyPolicy.defaultKeyPredicate(key)) {\n            return succeed(`${this.prefix}${key}`);\n        }\n        return fail(`${key}: invalid key`);\n    }\n}\n\nexport type MapOrRecord<T> = Map<string, T>|Record<string, T>;\n\n/**\n * A SimpleJsonMap presents a view of a simple map of @see JsonValue\n */\nexport abstract class SimpleJsonMapBase<T> implements JsonReferenceMap {\n    protected readonly _keyPolicy: ReferenceMapKeyPolicy<T>;\n    protected readonly _values: Map<string, T>;\n    protected readonly _context?: JsonContext;\n\n    protected constructor(values?: MapOrRecord<T>, context?: JsonContext, keyPolicy?: ReferenceMapKeyPolicy<T>) {\n        values = SimpleJsonMapBase._toMap(values).getValueOrThrow();\n        this._keyPolicy = keyPolicy ?? new ReferenceMapKeyPolicy();\n        this._values = this._keyPolicy.validateMap(values).getValueOrThrow();\n        this._context = context;\n    }\n\n    protected static _toMap<T>(values?: MapOrRecord<T>): Result<Map<string, T>> {\n        if (values === undefined) {\n            return captureResult(() => new Map<string, T>());\n        }\n        else if (!(values instanceof Map)) {\n            return recordToMap(values, (_k, v) => succeed(v));\n        }\n        return succeed(values);\n    }\n\n    /**\n     * Determine if a key might be valid for this map but does not determine if key actually\n     * exists. Allows key range to be constrained.\n     * @param key key to be tested\n     * @returns true if the key is in the valid range, false otherwise.\n     */\n    public keyIsInRange(key: string): boolean {\n        return this._keyPolicy.isValid(key);\n    }\n\n    /**\n     * Determines if an object with the specified key actually exists in the map.\n     * @param key key to be tested\n     * @returns true if an object with the specified key exists, false otherwise.\n     */\n    public has(key: string): boolean {\n        return this._values.has(key);\n    }\n\n    /**\n     * Gets a JSON object specified by key.\n     * @param key key of the object to be retrieved\n     * @param context optional @see JsonContext used to format the object\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    public getJsonObject(key: string, context?: JsonContext): DetailedResult<JsonObject, JsonReferenceMapFailureReason> {\n        return this.getJsonValue(key, context).onSuccess((jv) => {\n            if (!isJsonObject(jv)) {\n                return failWithDetail(`${key}: not an object`, 'error');\n            }\n            return succeedWithDetail(jv);\n        });\n    }\n\n    /**\n     * Gets a JSON value specified by key.\n     * @param key key of the object to be retrieved\n     * @param context Optional @see JsonContext used to format the value\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    // eslint-disable-next-line no-use-before-define\n    public abstract getJsonValue(key: string, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason>;\n}\n\n/**\n * Initialization options for a @see SimpleJsonMap\n */\nexport interface SimpleJsonMapOptions {\n    keyPolicy?: ReferenceMapKeyPolicy<JsonValue>,\n    editor?: JsonEditor,\n}\n\n/**\n * A SimpleJsonMap presents a view of a simple map of @see JsonValue\n */\nexport class SimpleJsonMap extends SimpleJsonMapBase<JsonValue> {\n    protected _editor?: JsonEditor;\n\n    protected constructor(values?: MapOrRecord<JsonValue>, context?: JsonContext, options?: SimpleJsonMapOptions) {\n        super(values, context, options?.keyPolicy);\n        this._editor = options?.editor;\n    }\n\n    /**\n     * Creates a new @see SimpleJsonMap from the supplied objects\n     * @param values A string-keyed Map or Record of the @see JsonObject to be returned\n     * @param context Optional @see JsonContext used to format returned values\n     * @param options Optional @see SimpleJsonMapOptions for initialization\n     */\n    public static createSimple(values?: MapOrRecord<JsonValue>, context?: JsonContext, options?: SimpleJsonMapOptions): Result<SimpleJsonMap> {\n        return captureResult(() => new SimpleJsonMap(values, context, options));\n    }\n\n    /**\n     * Gets a JSON value specified by key.\n     * @param key key of the object to be retrieved\n     * @param context Optional @see JsonContext used to format the value\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    // eslint-disable-next-line no-use-before-define\n    public getJsonValue(key: string, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason> {\n        context = context ?? this._context;\n        const value = this._values.get(key);\n        if (!value) {\n            return failWithDetail(`${key}: JSON value not found`, 'unknown');\n        }\n        return this._clone(value, context);\n    }\n\n    protected _clone(value: JsonValue, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason> {\n        if (!this._editor) {\n            const result = JsonEditor.create();\n            // istanbul ignore next: nearly impossible to reproduce\n            if (result.isFailure()) {\n                return failWithDetail(result.message, 'error');\n            }\n            this._editor = result.value;\n        }\n        return this._editor.clone(value, context).withFailureDetail('error');\n    }\n}\n\n/**\n * Initialization options for a PrefixedJsonMap\n */\nexport interface KeyPrefixOptions {\n    /**\n     * Indicates whether the prefix should be added automatically as needed (default true)\n     */\n    addPrefix?: boolean;\n\n    /**\n     * The prefix to be enforced\n     */\n    prefix: string;\n}\n\n/**\n * A PrefixedJsonMap enforces a supplied prefix for all contained values, optionally\n * adding the prefix as necessary (default true).\n */\nexport class PrefixedJsonMap extends SimpleJsonMap {\n    protected constructor(values?: MapOrRecord<JsonValue>, context?: JsonContext, options?: SimpleJsonMapOptions) {\n        super(values, context, options);\n    }\n\n    /**\n     * Creates a new @see PrefixedJsonMap from the supplied values\n     * @param prefix A string prefix to be enforced for and added to key names as necessary\n     * @param values A string-keyed Map or Record of the @see JsonValue to be returned\n     * @param context Optional @see JsonContext used to format returned values\n     * @param editor Optional @see JsonEditor used to format returned values\n     */\n    public static createPrefixed(prefix: string, values?: MapOrRecord<JsonValue>, context?: JsonContext, editor?: JsonEditor): Result<PrefixedJsonMap>;\n\n    /**\n     * Creates a new @see PrefixedJsonMap from the supplied values\n     * @param prefixOptions A KeyPrefixOptions indicating the prefix to enforce and whether that prefix should\n     * be added automatically if necessary (default true)\n     * @param values A string-keyed Map or record of the @see JsonValue to be returned\n     * @param context Optional @see JsonContext used to format returned values\n     * @param editor Optional @see JsonEditor used to format returned values\n     */\n    public static createPrefixed(prefixOptions: KeyPrefixOptions, values?: MapOrRecord<JsonValue>, context?: JsonContext, editor?: JsonEditor): Result<PrefixedJsonMap>;\n    public static createPrefixed(prefixOptions: string|KeyPrefixOptions, values?: MapOrRecord<JsonValue>, context?: JsonContext, editor?: JsonEditor): Result<PrefixedJsonMap> {\n        return captureResult(() => new PrefixedJsonMap(values, context, { keyPolicy: this._toPolicy(prefixOptions), editor }));\n    }\n\n    protected static _toPolicy(prefixOptions: string|KeyPrefixOptions): ReferenceMapKeyPolicy<JsonValue> {\n        if (typeof prefixOptions === 'string') {\n            return new PrefixKeyPolicy(prefixOptions, { makeValid: true });\n        }\n        return new PrefixKeyPolicy(prefixOptions.prefix, { makeValid: (prefixOptions.addPrefix !== false) });\n    }\n}\n\n/**\n * A CompositeJsonMap presents a composed view of one or more other\n * JsonReferenceMaps.\n */\nexport class CompositeJsonMap implements JsonReferenceMap {\n    protected _maps: JsonReferenceMap[];\n\n    protected constructor(maps: JsonReferenceMap[]) {\n        this._maps = maps;\n    }\n\n    /**\n     * Creates a new @see CompositeJsonMap from the supplied maps\n     * @param maps one or more object maps to be composed\n     */\n    public static create(maps: JsonReferenceMap[]): Result<CompositeJsonMap> {\n        return captureResult(() => new CompositeJsonMap(maps));\n    }\n\n    /**\n     * Determine if a key might be valid for this map but does not determine\n     * if key actually exists. Allows key range to be constrained.\n     * @param key key to be tested\n     * @returns true if the key is in the valid range, false otherwise.\n     */\n    public keyIsInRange(key: string): boolean {\n        return this._maps.find((map) => map.keyIsInRange(key)) !== undefined;\n    }\n\n    /**\n     * Determines if an object with the specified key actually exists in the map.\n     * @param key key to be tested\n     * @returns true if an object with the specified key exists, false otherwise.\n     */\n    public has(key: string): boolean {\n        return this._maps.find((map) => map.has(key)) !== undefined;\n    }\n\n    /**\n     * Gets a JSON object specified by key.\n     * @param key key of the object to be retrieved\n     * @param context optional @see JsonContext used to format the object\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    public getJsonObject(key: string, context?: JsonContext): DetailedResult<JsonObject, JsonReferenceMapFailureReason> {\n        return this.getJsonValue(key, context).onSuccess((jv) => {\n            if (!isJsonObject(jv)) {\n                return failWithDetail(`${key}: not an object`, 'error');\n            }\n            return succeedWithDetail(jv);\n        });\n    }\n\n    /**\n     * Gets a JSON value specified by key.\n     * @param key key of the object to be retrieved\n     * @param context Optional @see JsonContext used to format the value\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    // eslint-disable-next-line no-use-before-define\n    public getJsonValue(key: string, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason> {\n        for (const map of this._maps) {\n            if (map.keyIsInRange(key)) {\n                const result = map.getJsonValue(key, context);\n                if (result.isSuccess() || (result.detail === 'error')) {\n                    return result;\n                }\n            }\n        }\n        return failWithDetail(`${key}: value not found`, 'unknown');\n    }\n}\n"]}
|
|
272
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsonReferenceMap.js","sourceRoot":"","sources":["../src/jsonReferenceMap.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    DetailedResult,\n    Result,\n    captureResult,\n    fail,\n    failWithDetail,\n    mapResults,\n    recordToMap,\n    succeed,\n    succeedWithDetail,\n} from '@fgv/ts-utils';\n\nimport { JsonContext, JsonReferenceMap, JsonReferenceMapFailureReason } from './jsonContext';\nimport { JsonObject, JsonValue, isJsonObject } from './common';\nimport { JsonEditor } from './jsonEditor';\n\nexport interface ReferenceMapKeyPolicyValidateOptions {\n    makeValid?: boolean;\n}\n\n/**\n * Policy object responsible for validating or correcting\n * keys in a reference map\n */\nexport class ReferenceMapKeyPolicy<T> {\n    protected readonly _defaultOptions?: ReferenceMapKeyPolicyValidateOptions;\n    protected readonly _isValid: (key: string, item?: T) => boolean;\n\n    public constructor(options?: ReferenceMapKeyPolicyValidateOptions, isValid?: (key: string, item?: T) => boolean) {\n        this._defaultOptions = options;\n        this._isValid = isValid ?? ReferenceMapKeyPolicy.defaultKeyPredicate;\n    }\n\n    public static defaultKeyPredicate(key: string): boolean {\n        return (key.length > 0) && (!key.includes('{{')) && (!key.startsWith('?'));\n    }\n\n    public isValid(key: string, item?: T): boolean {\n        return this._isValid(key, item);\n    }\n\n    public validate(key: string, item?: T, _options?: ReferenceMapKeyPolicyValidateOptions): Result<string> {\n        return this.isValid(key, item) ? succeed(key) : fail(`${key}: invalid key`);\n    }\n\n    public validateItems(items: [string, T][], options?: ReferenceMapKeyPolicyValidateOptions): Result<[string, T][]> {\n        return mapResults(items.map((item) => {\n            return this.validate(...item, options).onSuccess((valid) => {\n                return succeed([valid, item[1]]);\n            });\n        }));\n    }\n\n    public validateMap(map: Map<string, T>, options?: ReferenceMapKeyPolicyValidateOptions): Result<Map<string, T>> {\n        return this.validateItems(Array.from(map.entries()), options).onSuccess((valid) => {\n            return captureResult(() => new Map(valid));\n        });\n    }\n}\n\nexport class PrefixKeyPolicy<T> extends ReferenceMapKeyPolicy<T> {\n    public readonly prefix: string;\n\n    public constructor(prefix: string, options?: ReferenceMapKeyPolicyValidateOptions) {\n        super(options);\n        this.prefix = prefix;\n    }\n\n    public isValid(key: string, _item?: T): boolean {\n        return key.startsWith(this.prefix) && (key !== this.prefix) && ReferenceMapKeyPolicy.defaultKeyPredicate(key);\n    }\n\n    public validate(key: string, item?: T, options?: ReferenceMapKeyPolicyValidateOptions): Result<string> {\n        // istanbul ignore next\n        const makeValid = (options ?? this._defaultOptions)?.makeValid === true;\n        if (this.isValid(key, item)) {\n            return succeed(key);\n        }\n        else if (makeValid && ReferenceMapKeyPolicy.defaultKeyPredicate(key)) {\n            return succeed(`${this.prefix}${key}`);\n        }\n        return fail(`${key}: invalid key`);\n    }\n}\n\nexport type MapOrRecord<T> = Map<string, T>|Record<string, T>;\n\n/**\n * A SimpleJsonMap presents a view of a simple map of @see JsonValue\n */\nexport abstract class SimpleJsonMapBase<T> implements JsonReferenceMap {\n    protected readonly _keyPolicy: ReferenceMapKeyPolicy<T>;\n    protected readonly _values: Map<string, T>;\n    protected readonly _context?: JsonContext;\n\n    protected constructor(values?: MapOrRecord<T>, context?: JsonContext, keyPolicy?: ReferenceMapKeyPolicy<T>) {\n        values = SimpleJsonMapBase._toMap(values).orThrow();\n        this._keyPolicy = keyPolicy ?? new ReferenceMapKeyPolicy();\n        this._values = this._keyPolicy.validateMap(values).orThrow();\n        this._context = context;\n    }\n\n    protected static _toMap<T>(values?: MapOrRecord<T>): Result<Map<string, T>> {\n        if (values === undefined) {\n            return captureResult(() => new Map<string, T>());\n        }\n        else if (!(values instanceof Map)) {\n            return recordToMap(values, (_k, v) => succeed(v));\n        }\n        return succeed(values);\n    }\n\n    /**\n     * Determine if a key might be valid for this map but does not determine if key actually\n     * exists. Allows key range to be constrained.\n     * @param key key to be tested\n     * @returns true if the key is in the valid range, false otherwise.\n     */\n    public keyIsInRange(key: string): boolean {\n        return this._keyPolicy.isValid(key);\n    }\n\n    /**\n     * Determines if an object with the specified key actually exists in the map.\n     * @param key key to be tested\n     * @returns true if an object with the specified key exists, false otherwise.\n     */\n    public has(key: string): boolean {\n        return this._values.has(key);\n    }\n\n    /**\n     * Gets a JSON object specified by key.\n     * @param key key of the object to be retrieved\n     * @param context optional @see JsonContext used to format the object\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    public getJsonObject(key: string, context?: JsonContext): DetailedResult<JsonObject, JsonReferenceMapFailureReason> {\n        return this.getJsonValue(key, context).onSuccess((jv) => {\n            if (!isJsonObject(jv)) {\n                return failWithDetail(`${key}: not an object`, 'error');\n            }\n            return succeedWithDetail(jv);\n        });\n    }\n\n    /**\n     * Gets a JSON value specified by key.\n     * @param key key of the object to be retrieved\n     * @param context Optional @see JsonContext used to format the value\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    // eslint-disable-next-line no-use-before-define\n    public abstract getJsonValue(key: string, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason>;\n}\n\n/**\n * Initialization options for a @see SimpleJsonMap\n */\nexport interface SimpleJsonMapOptions {\n    keyPolicy?: ReferenceMapKeyPolicy<JsonValue>,\n    editor?: JsonEditor,\n}\n\n/**\n * A SimpleJsonMap presents a view of a simple map of @see JsonValue\n */\nexport class SimpleJsonMap extends SimpleJsonMapBase<JsonValue> {\n    protected _editor?: JsonEditor;\n\n    protected constructor(values?: MapOrRecord<JsonValue>, context?: JsonContext, options?: SimpleJsonMapOptions) {\n        super(values, context, options?.keyPolicy);\n        this._editor = options?.editor;\n    }\n\n    /**\n     * Creates a new @see SimpleJsonMap from the supplied objects\n     * @param values A string-keyed Map or Record of the @see JsonObject to be returned\n     * @param context Optional @see JsonContext used to format returned values\n     * @param options Optional @see SimpleJsonMapOptions for initialization\n     */\n    public static createSimple(values?: MapOrRecord<JsonValue>, context?: JsonContext, options?: SimpleJsonMapOptions): Result<SimpleJsonMap> {\n        return captureResult(() => new SimpleJsonMap(values, context, options));\n    }\n\n    /**\n     * Gets a JSON value specified by key.\n     * @param key key of the object to be retrieved\n     * @param context Optional @see JsonContext used to format the value\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    // eslint-disable-next-line no-use-before-define\n    public getJsonValue(key: string, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason> {\n        context = context ?? this._context;\n        const value = this._values.get(key);\n        if (!value) {\n            return failWithDetail(`${key}: JSON value not found`, 'unknown');\n        }\n        return this._clone(value, context);\n    }\n\n    protected _clone(value: JsonValue, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason> {\n        if (!this._editor) {\n            const result = JsonEditor.create();\n            // istanbul ignore next: nearly impossible to reproduce\n            if (result.isFailure()) {\n                return failWithDetail(result.message, 'error');\n            }\n            this._editor = result.value;\n        }\n        return this._editor.clone(value, context).withFailureDetail('error');\n    }\n}\n\n/**\n * Initialization options for a PrefixedJsonMap\n */\nexport interface KeyPrefixOptions {\n    /**\n     * Indicates whether the prefix should be added automatically as needed (default true)\n     */\n    addPrefix?: boolean;\n\n    /**\n     * The prefix to be enforced\n     */\n    prefix: string;\n}\n\n/**\n * A PrefixedJsonMap enforces a supplied prefix for all contained values, optionally\n * adding the prefix as necessary (default true).\n */\nexport class PrefixedJsonMap extends SimpleJsonMap {\n    protected constructor(values?: MapOrRecord<JsonValue>, context?: JsonContext, options?: SimpleJsonMapOptions) {\n        super(values, context, options);\n    }\n\n    /**\n     * Creates a new @see PrefixedJsonMap from the supplied values\n     * @param prefix A string prefix to be enforced for and added to key names as necessary\n     * @param values A string-keyed Map or Record of the @see JsonValue to be returned\n     * @param context Optional @see JsonContext used to format returned values\n     * @param editor Optional @see JsonEditor used to format returned values\n     */\n    public static createPrefixed(prefix: string, values?: MapOrRecord<JsonValue>, context?: JsonContext, editor?: JsonEditor): Result<PrefixedJsonMap>;\n\n    /**\n     * Creates a new @see PrefixedJsonMap from the supplied values\n     * @param prefixOptions A KeyPrefixOptions indicating the prefix to enforce and whether that prefix should\n     * be added automatically if necessary (default true)\n     * @param values A string-keyed Map or record of the @see JsonValue to be returned\n     * @param context Optional @see JsonContext used to format returned values\n     * @param editor Optional @see JsonEditor used to format returned values\n     */\n    public static createPrefixed(prefixOptions: KeyPrefixOptions, values?: MapOrRecord<JsonValue>, context?: JsonContext, editor?: JsonEditor): Result<PrefixedJsonMap>;\n    public static createPrefixed(prefixOptions: string|KeyPrefixOptions, values?: MapOrRecord<JsonValue>, context?: JsonContext, editor?: JsonEditor): Result<PrefixedJsonMap> {\n        return captureResult(() => new PrefixedJsonMap(values, context, { keyPolicy: this._toPolicy(prefixOptions), editor }));\n    }\n\n    protected static _toPolicy(prefixOptions: string|KeyPrefixOptions): ReferenceMapKeyPolicy<JsonValue> {\n        if (typeof prefixOptions === 'string') {\n            return new PrefixKeyPolicy(prefixOptions, { makeValid: true });\n        }\n        return new PrefixKeyPolicy(prefixOptions.prefix, { makeValid: (prefixOptions.addPrefix !== false) });\n    }\n}\n\n/**\n * A CompositeJsonMap presents a composed view of one or more other\n * JsonReferenceMaps.\n */\nexport class CompositeJsonMap implements JsonReferenceMap {\n    protected _maps: JsonReferenceMap[];\n\n    protected constructor(maps: JsonReferenceMap[]) {\n        this._maps = maps;\n    }\n\n    /**\n     * Creates a new @see CompositeJsonMap from the supplied maps\n     * @param maps one or more object maps to be composed\n     */\n    public static create(maps: JsonReferenceMap[]): Result<CompositeJsonMap> {\n        return captureResult(() => new CompositeJsonMap(maps));\n    }\n\n    /**\n     * Determine if a key might be valid for this map but does not determine\n     * if key actually exists. Allows key range to be constrained.\n     * @param key key to be tested\n     * @returns true if the key is in the valid range, false otherwise.\n     */\n    public keyIsInRange(key: string): boolean {\n        return this._maps.find((map) => map.keyIsInRange(key)) !== undefined;\n    }\n\n    /**\n     * Determines if an object with the specified key actually exists in the map.\n     * @param key key to be tested\n     * @returns true if an object with the specified key exists, false otherwise.\n     */\n    public has(key: string): boolean {\n        return this._maps.find((map) => map.has(key)) !== undefined;\n    }\n\n    /**\n     * Gets a JSON object specified by key.\n     * @param key key of the object to be retrieved\n     * @param context optional @see JsonContext used to format the object\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    public getJsonObject(key: string, context?: JsonContext): DetailedResult<JsonObject, JsonReferenceMapFailureReason> {\n        return this.getJsonValue(key, context).onSuccess((jv) => {\n            if (!isJsonObject(jv)) {\n                return failWithDetail(`${key}: not an object`, 'error');\n            }\n            return succeedWithDetail(jv);\n        });\n    }\n\n    /**\n     * Gets a JSON value specified by key.\n     * @param key key of the object to be retrieved\n     * @param context Optional @see JsonContext used to format the value\n     * @returns Success with the formatted object if successful. Failure with detail 'unknown'\n     * if no such object exists, or failure with detail 'error' if the object was found but\n     * could not be formatted.\n     */\n    // eslint-disable-next-line no-use-before-define\n    public getJsonValue(key: string, context?: JsonContext): DetailedResult<JsonValue, JsonReferenceMapFailureReason> {\n        for (const map of this._maps) {\n            if (map.keyIsInRange(key)) {\n                const result = map.getJsonValue(key, context);\n                if (result.isSuccess() || (result.detail === 'error')) {\n                    return result;\n                }\n            }\n        }\n        return failWithDetail(`${key}: value not found`, 'unknown');\n    }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@fgv/ts-json",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.8.1",
|
|
4
4
|
"description": "Typescript utilities for working with JSON",
|
|
5
5
|
"main": "index.js",
|
|
6
6
|
"scripts": {
|
|
@@ -24,25 +24,25 @@
|
|
|
24
24
|
"author": "Erik Fortune",
|
|
25
25
|
"license": "MIT",
|
|
26
26
|
"devDependencies": {
|
|
27
|
-
"@fgv/ts-utils-jest": "^1.
|
|
28
|
-
"@types/jest": "^
|
|
27
|
+
"@fgv/ts-utils-jest": "^1.8.1",
|
|
28
|
+
"@types/jest": "^29.2.5",
|
|
29
29
|
"@types/mustache": "^4.1.0",
|
|
30
|
-
"@types/node": "^
|
|
31
|
-
"@typescript-eslint/eslint-plugin": "^
|
|
32
|
-
"@typescript-eslint/parser": "^
|
|
33
|
-
"eslint": "^
|
|
34
|
-
"eslint-config-standard": "^
|
|
30
|
+
"@types/node": "^18.11.18",
|
|
31
|
+
"@typescript-eslint/eslint-plugin": "^5.48.0",
|
|
32
|
+
"@typescript-eslint/parser": "^5.48.0",
|
|
33
|
+
"eslint": "^8.31.0",
|
|
34
|
+
"eslint-config-standard": "^17.0.0",
|
|
35
35
|
"eslint-plugin-import": "^2.22.1",
|
|
36
36
|
"eslint-plugin-node": "^11.1.0",
|
|
37
|
-
"eslint-plugin-promise": "^
|
|
38
|
-
"jest": "^
|
|
39
|
-
"jest-extended": "^
|
|
37
|
+
"eslint-plugin-promise": "^6.1.1",
|
|
38
|
+
"jest": "^29.3.1",
|
|
39
|
+
"jest-extended": "^3.2.1",
|
|
40
40
|
"rimraf": "^3.0.2",
|
|
41
|
-
"ts-jest": "^
|
|
42
|
-
"ts-node": "^9.1
|
|
41
|
+
"ts-jest": "^29.0.3",
|
|
42
|
+
"ts-node": "^10.9.1",
|
|
43
43
|
"typescript": "^4.1.3"
|
|
44
44
|
},
|
|
45
45
|
"dependencies": {
|
|
46
|
-
"@fgv/ts-utils": "^1.
|
|
46
|
+
"@fgv/ts-utils": "^1.8.1"
|
|
47
47
|
}
|
|
48
48
|
}
|