ngx-schema-forms
Version:
New features: - Ajv schema validator. - Angular forms compatible: Property tree is created using FormGroup, FormArray and FormControl classes. - Array now properly loads initial data from model. - WidgetTyep: WidgetRegistry now supports WidgetType, now wo
303 lines (302 loc) • 24.7 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
import * as tslib_1 from "tslib";
import { isBlank } from './utils';
/**
* @param {?} message
* @param {?} path
* @return {?}
*/
function formatMessage(message, path) {
return "Parsing error on " + path + ": " + message;
}
/**
* @param {?} message
* @param {?} path
* @return {?}
*/
function schemaError(message, path) {
/** @type {?} */
var mesg = formatMessage(message, path);
throw new Error(mesg);
}
/**
* @param {?} message
* @param {?} path
* @return {?}
*/
function schemaWarning(message, path) {
/** @type {?} */
var mesg = formatMessage(message, path);
throw new Error(mesg);
}
var SchemaPreprocessor = /** @class */ (function () {
function SchemaPreprocessor() {
}
/**
* @param {?} jsonSchema
* @param {?=} path
* @return {?}
*/
SchemaPreprocessor.preprocess = /**
* @param {?} jsonSchema
* @param {?=} path
* @return {?}
*/
function (jsonSchema, path) {
if (path === void 0) { path = '/'; }
jsonSchema = jsonSchema || {};
if (jsonSchema.type === 'object') {
SchemaPreprocessor.checkProperties(jsonSchema, path);
SchemaPreprocessor.checkAndCreateFieldsets(jsonSchema, path);
}
else if (jsonSchema.type === 'array') {
SchemaPreprocessor.checkItems(jsonSchema, path);
}
SchemaPreprocessor.normalizeWidget(jsonSchema);
SchemaPreprocessor.recursiveCheck(jsonSchema, path);
};
/**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
SchemaPreprocessor.checkProperties = /**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
function (jsonSchema, path) {
if (isBlank(jsonSchema.properties)) {
jsonSchema.properties = {};
schemaWarning('Provided json schema does not contain a \'properties\' entry. Output schema will be empty', path);
}
};
/**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
SchemaPreprocessor.checkAndCreateFieldsets = /**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
function (jsonSchema, path) {
if (jsonSchema.fieldsets === undefined) {
if (jsonSchema.order !== undefined) {
SchemaPreprocessor.replaceOrderByFieldsets(jsonSchema);
}
else {
SchemaPreprocessor.createFieldsets(jsonSchema);
}
}
SchemaPreprocessor.checkFieldsUsage(jsonSchema, path);
};
/**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
SchemaPreprocessor.checkFieldsUsage = /**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
function (jsonSchema, path) {
/** @type {?} */
var fieldsId = Object.keys(jsonSchema.properties);
/** @type {?} */
var usedFields = {};
try {
for (var _a = tslib_1.__values(jsonSchema.fieldsets), _b = _a.next(); !_b.done; _b = _a.next()) {
var fieldset = _b.value;
try {
for (var _c = tslib_1.__values(fieldset.fields), _d = _c.next(); !_d.done; _d = _c.next()) {
var fieldId = _d.value;
if (usedFields[fieldId] === undefined) {
usedFields[fieldId] = [];
}
usedFields[fieldId].push(fieldset.id);
}
}
catch (e_1_1) { e_1 = { error: e_1_1 }; }
finally {
try {
if (_d && !_d.done && (_e = _c.return)) _e.call(_c);
}
finally { if (e_1) throw e_1.error; }
}
}
}
catch (e_2_1) { e_2 = { error: e_2_1 }; }
finally {
try {
if (_b && !_b.done && (_f = _a.return)) _f.call(_a);
}
finally { if (e_2) throw e_2.error; }
}
try {
for (var fieldsId_1 = tslib_1.__values(fieldsId), fieldsId_1_1 = fieldsId_1.next(); !fieldsId_1_1.done; fieldsId_1_1 = fieldsId_1.next()) {
var fieldId = fieldsId_1_1.value;
if (usedFields.hasOwnProperty(fieldId)) {
if (usedFields[fieldId].length > 1) {
schemaError(fieldId + " is referenced by more than one fieldset: " + usedFields[fieldId], path);
}
delete usedFields[fieldId];
}
else if (jsonSchema.required.indexOf(fieldId) > -1) {
schemaError(fieldId + " is a required field but it is not referenced as part of a 'order' or a 'fieldset' property", path);
}
else {
delete jsonSchema[fieldId];
schemaWarning("Removing unreferenced field " + fieldId, path);
}
}
}
catch (e_3_1) { e_3 = { error: e_3_1 }; }
finally {
try {
if (fieldsId_1_1 && !fieldsId_1_1.done && (_g = fieldsId_1.return)) _g.call(fieldsId_1);
}
finally { if (e_3) throw e_3.error; }
}
for (var remainingfieldsId in usedFields) {
if (usedFields.hasOwnProperty(remainingfieldsId)) {
schemaWarning("Referencing non-existent field " + remainingfieldsId + " in one or more fieldsets", path);
}
}
var e_2, _f, e_1, _e, e_3, _g;
};
/**
* @param {?} jsonSchema
* @return {?}
*/
SchemaPreprocessor.createFieldsets = /**
* @param {?} jsonSchema
* @return {?}
*/
function (jsonSchema) {
jsonSchema.order = Object.keys(jsonSchema.properties);
SchemaPreprocessor.replaceOrderByFieldsets(jsonSchema);
};
/**
* @param {?} jsonSchema
* @return {?}
*/
SchemaPreprocessor.replaceOrderByFieldsets = /**
* @param {?} jsonSchema
* @return {?}
*/
function (jsonSchema) {
jsonSchema.fieldsets = [{
id: 'fieldset-default',
title: jsonSchema.title || '',
description: jsonSchema.description || '',
name: jsonSchema.name || '',
fields: jsonSchema.order
}];
delete jsonSchema.order;
};
/**
* @param {?} fieldSchema
* @return {?}
*/
SchemaPreprocessor.normalizeWidget = /**
* @param {?} fieldSchema
* @return {?}
*/
function (fieldSchema) {
/** @type {?} */
var widget = fieldSchema.widget;
if (widget === undefined) {
widget = { 'id': fieldSchema.type };
}
else if (typeof widget === 'string') {
widget = { 'id': widget };
}
fieldSchema.widget = widget;
};
/**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
SchemaPreprocessor.checkItems = /**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
function (jsonSchema, path) {
if (jsonSchema.items === undefined) {
schemaError('No \'items\' property in array', path);
}
};
/**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
SchemaPreprocessor.recursiveCheck = /**
* @param {?} jsonSchema
* @param {?} path
* @return {?}
*/
function (jsonSchema, path) {
if (jsonSchema.type === 'object') {
/*
for (const fieldId in jsonSchema.properties) {
if (jsonSchema.properties.hasOwnProperty(fieldId)) {
const fieldSchema = jsonSchema.properties[fieldId];
SchemaPreprocessor.preprocess(fieldSchema, path + fieldId + '/');
}
}
*/
if (jsonSchema.hasOwnProperty('definitions')) {
for (var fieldId in jsonSchema.definitions) {
if (jsonSchema.definitions.hasOwnProperty(fieldId)) {
/** @type {?} */
var fieldSchema = jsonSchema.definitions[fieldId];
SchemaPreprocessor.removeRecursiveRefProperties(fieldSchema, "#/definitions/" + fieldId);
// formPropertyFactory recursive is used instead
// SchemaPreprocessor.preprocess(fieldSchema, path + fieldId + '/');
}
}
}
} // else if (jsonSchema.type === 'array') {
// formPropertyFactory recursive is used instead
// SchemaPreprocessor.preprocess(jsonSchema.items, path + '*/');
// }
};
/**
* @param {?} jsonSchema
* @param {?} definitionPath
* @return {?}
*/
SchemaPreprocessor.removeRecursiveRefProperties = /**
* @param {?} jsonSchema
* @param {?} definitionPath
* @return {?}
*/
function (jsonSchema, definitionPath) {
// to avoid infinite loop
if (jsonSchema.type === 'object') {
for (var fieldId in jsonSchema.properties) {
if (jsonSchema.properties.hasOwnProperty(fieldId)) {
if (jsonSchema.properties[fieldId].$ref
&& jsonSchema.properties[fieldId].$ref === definitionPath) {
delete jsonSchema.properties[fieldId];
}
else if (jsonSchema.properties[fieldId].type === 'object') {
SchemaPreprocessor.removeRecursiveRefProperties(jsonSchema.properties[fieldId], definitionPath);
}
}
}
}
};
return SchemaPreprocessor;
}());
export { SchemaPreprocessor };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schemapreprocessor.js","sourceRoot":"ng://ngx-schema-forms/","sources":["lib/model/schemapreprocessor.ts"],"names":[],"mappings":"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","sourcesContent":["import {isBlank} from './utils';\n\nfunction formatMessage(message, path) {\n  return `Parsing error on ${path}: ${message}`;\n}\n\nfunction schemaError(message, path): void {\n  const mesg = formatMessage(message, path);\n  throw new Error(mesg);\n}\n\nfunction schemaWarning(message, path): void {\n  const mesg = formatMessage(message, path);\n  throw new Error(mesg);\n}\n\n// TODO create error classes\nexport class SchemaPreprocessor {\n\n  static preprocess(jsonSchema: any, path = '/'): any {\n    jsonSchema = jsonSchema || {};\n\n    if (jsonSchema.type === 'object') {\n      SchemaPreprocessor.checkProperties(jsonSchema, path);\n      SchemaPreprocessor.checkAndCreateFieldsets(jsonSchema, path);\n    } else if (jsonSchema.type === 'array') {\n      SchemaPreprocessor.checkItems(jsonSchema, path);\n    }\n    SchemaPreprocessor.normalizeWidget(jsonSchema);\n    SchemaPreprocessor.recursiveCheck(jsonSchema, path);\n\n  }\n\n  private static checkProperties(jsonSchema, path: string) {\n    if (isBlank(jsonSchema.properties)) {\n      jsonSchema.properties = {};\n      schemaWarning('Provided json schema does not contain a \\'properties\\' entry. Output schema will be empty', path);\n    }\n  }\n\n  private static checkAndCreateFieldsets(jsonSchema: any, path: string) {\n    if (jsonSchema.fieldsets === undefined) {\n      if (jsonSchema.order !== undefined) {\n        SchemaPreprocessor.replaceOrderByFieldsets(jsonSchema);\n      } else {\n        SchemaPreprocessor.createFieldsets(jsonSchema);\n      }\n    }\n    SchemaPreprocessor.checkFieldsUsage(jsonSchema, path);\n  }\n\n  private static checkFieldsUsage(jsonSchema, path: string) {\n    const fieldsId: string[] = Object.keys(jsonSchema.properties);\n    const usedFields = {};\n    for (const fieldset of jsonSchema.fieldsets) {\n      for (const fieldId of fieldset.fields) {\n        if (usedFields[fieldId] === undefined) {\n          usedFields[fieldId] = [];\n        }\n        usedFields[fieldId].push(fieldset.id);\n      }\n    }\n\n    for (const fieldId of fieldsId) {\n      if (usedFields.hasOwnProperty(fieldId)) {\n        if (usedFields[fieldId].length > 1) {\n          schemaError(`${fieldId} is referenced by more than one fieldset: ${usedFields[fieldId]}`, path);\n        }\n        delete usedFields[fieldId];\n      } else if (jsonSchema.required.indexOf(fieldId) > -1) {\n        schemaError(`${fieldId} is a required field but it is not referenced as part of a 'order' or a 'fieldset' property`, path);\n      } else {\n        delete jsonSchema[fieldId];\n        schemaWarning(`Removing unreferenced field ${fieldId}`, path);\n      }\n    }\n\n    for (const remainingfieldsId in usedFields) {\n      if (usedFields.hasOwnProperty(remainingfieldsId)) {\n        schemaWarning(`Referencing non-existent field ${remainingfieldsId} in one or more fieldsets`, path);\n      }\n    }\n  }\n\n  private static createFieldsets(jsonSchema) {\n    jsonSchema.order = Object.keys(jsonSchema.properties);\n    SchemaPreprocessor.replaceOrderByFieldsets(jsonSchema);\n  }\n\n  private static replaceOrderByFieldsets(jsonSchema) {\n    jsonSchema.fieldsets = [{\n      id: 'fieldset-default',\n      title: jsonSchema.title || '',\n      description: jsonSchema.description || '',\n      name: jsonSchema.name || '',\n      fields: jsonSchema.order\n    }];\n    delete jsonSchema.order;\n  }\n\n  private static normalizeWidget(fieldSchema: any) {\n    let widget = fieldSchema.widget;\n    if (widget === undefined) {\n      widget = {'id': fieldSchema.type};\n    } else if (typeof widget === 'string') {\n      widget = {'id': widget};\n    }\n    fieldSchema.widget = widget;\n  }\n\n  private static checkItems(jsonSchema, path) {\n    if (jsonSchema.items === undefined) {\n      schemaError('No \\'items\\' property in array', path);\n    }\n  }\n\n  // TODO rename and remove unnecessary code according to change\n  // TODO test, to make sure removal of recursion checks does not break anything\n  private static recursiveCheck(jsonSchema, path: string) {\n    if (jsonSchema.type === 'object') {\n      /*\n      for (const fieldId in jsonSchema.properties) {\n        if (jsonSchema.properties.hasOwnProperty(fieldId)) {\n          const fieldSchema = jsonSchema.properties[fieldId];\n          SchemaPreprocessor.preprocess(fieldSchema, path + fieldId + '/');\n        }\n      }\n      */\n      if (jsonSchema.hasOwnProperty('definitions')) {\n        for (const fieldId in jsonSchema.definitions) {\n          if (jsonSchema.definitions.hasOwnProperty(fieldId)) {\n            const fieldSchema = jsonSchema.definitions[fieldId];\n            SchemaPreprocessor.removeRecursiveRefProperties(\n              fieldSchema,\n              `#/definitions/${fieldId}`\n            );\n            // formPropertyFactory recursive is used instead\n            // SchemaPreprocessor.preprocess(fieldSchema, path + fieldId + '/');\n          }\n        }\n      }\n    }// else if (jsonSchema.type === 'array') {\n      // formPropertyFactory recursive is used instead\n      // SchemaPreprocessor.preprocess(jsonSchema.items, path + '*/');\n    // }\n  }\n\n  private static removeRecursiveRefProperties(jsonSchema, definitionPath) {\n    // to avoid infinite loop\n    if (jsonSchema.type === 'object') {\n      for (const fieldId in jsonSchema.properties) {\n        if (jsonSchema.properties.hasOwnProperty(fieldId)) {\n          if (jsonSchema.properties[fieldId].$ref\n            && jsonSchema.properties[fieldId].$ref === definitionPath) {\n            delete jsonSchema.properties[fieldId];\n          } else if (jsonSchema.properties[fieldId].type === 'object') {\n            SchemaPreprocessor.removeRecursiveRefProperties(\n              jsonSchema.properties[fieldId],\n              definitionPath\n            );\n          }\n        }\n      }\n    }\n  }\n}\n\n"]}