UNPKG

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

219 lines (218 loc) 22 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ 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 {?} */ const mesg = formatMessage(message, path); throw new Error(mesg); } /** * @param {?} message * @param {?} path * @return {?} */ function schemaWarning(message, path) { /** @type {?} */ const mesg = formatMessage(message, path); throw new Error(mesg); } export class SchemaPreprocessor { /** * @param {?} jsonSchema * @param {?=} path * @return {?} */ static preprocess(jsonSchema, 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 {?} */ static checkProperties(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 {?} */ static checkAndCreateFieldsets(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 {?} */ static checkFieldsUsage(jsonSchema, path) { /** @type {?} */ const fieldsId = Object.keys(jsonSchema.properties); /** @type {?} */ const usedFields = {}; for (const fieldset of jsonSchema.fieldsets) { for (const fieldId of fieldset.fields) { if (usedFields[fieldId] === undefined) { usedFields[fieldId] = []; } usedFields[fieldId].push(fieldset.id); } } for (const fieldId of fieldsId) { 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); } } for (const remainingfieldsId in usedFields) { if (usedFields.hasOwnProperty(remainingfieldsId)) { schemaWarning(`Referencing non-existent field ${remainingfieldsId} in one or more fieldsets`, path); } } } /** * @param {?} jsonSchema * @return {?} */ static createFieldsets(jsonSchema) { jsonSchema.order = Object.keys(jsonSchema.properties); SchemaPreprocessor.replaceOrderByFieldsets(jsonSchema); } /** * @param {?} jsonSchema * @return {?} */ static replaceOrderByFieldsets(jsonSchema) { jsonSchema.fieldsets = [{ id: 'fieldset-default', title: jsonSchema.title || '', description: jsonSchema.description || '', name: jsonSchema.name || '', fields: jsonSchema.order }]; delete jsonSchema.order; } /** * @param {?} fieldSchema * @return {?} */ static normalizeWidget(fieldSchema) { /** @type {?} */ let 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 {?} */ static checkItems(jsonSchema, path) { if (jsonSchema.items === undefined) { schemaError('No \'items\' property in array', path); } } /** * @param {?} jsonSchema * @param {?} path * @return {?} */ static recursiveCheck(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 (const fieldId in jsonSchema.definitions) { if (jsonSchema.definitions.hasOwnProperty(fieldId)) { /** @type {?} */ const 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 {?} */ static removeRecursiveRefProperties(jsonSchema, definitionPath) { // to avoid infinite loop if (jsonSchema.type === 'object') { for (const 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); } } } } } } //# 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"]}