diff --git a/lib/codegen/fromcto/jsonschema/jsonschemavisitor.js b/lib/codegen/fromcto/jsonschema/jsonschemavisitor.js index 819766f0..7ec27950 100644 --- a/lib/codegen/fromcto/jsonschema/jsonschemavisitor.js +++ b/lib/codegen/fromcto/jsonschema/jsonschemavisitor.js @@ -60,16 +60,16 @@ class JSONSchemaVisitor { /** * Get the validators for a field or a scalar definition in JSON schema form. - * @param {Object} field - the scalar declaration being visited + * @param {String} type - the field type * @param {bool} [isScalarUUID] - flag to indicate given field type is scalar uuid + * @param {bool} [validator] - the field validator * @return {Object} the result of visiting or null * @private */ - getFieldOrScalarDeclarationValidatorsForSchema(field, isScalarUUID = false) { - const validator = field.getValidator(); + getFieldOrScalarDeclarationValidatorsForSchema(type, isScalarUUID = false, validator) { let jsonSchema = {}; - switch (field.getType()) { + switch (type) { case 'String': jsonSchema.type = 'string'; if (isScalarUUID) { @@ -117,6 +117,8 @@ class JSONSchemaVisitor { case 'Boolean': jsonSchema.type = 'boolean'; break; + default: + jsonSchema.type = type; } return jsonSchema; @@ -354,7 +356,7 @@ class JSONSchemaVisitor { debug('entering visitScalarDeclaration', scalarDeclaration.getName()); return { $id: scalarDeclaration.getFullyQualifiedName(), - schema: this.getFieldOrScalarDeclarationValidatorsForSchema(scalarDeclaration) + schema: this.getFieldOrScalarDeclarationValidatorsForSchema(scalarDeclaration.getType(), false, scalarDeclaration.getValidator()) }; } @@ -381,11 +383,6 @@ class JSONSchemaVisitor { visitField(field, parameters, isScalarUUID = false) { debug('entering visitField', field.getName()); - - if (ModelUtil.isMap?.(field)) { - return; - } - // Is this a primitive typed property? let jsonSchema; if (field.isPrimitive()) { @@ -398,7 +395,7 @@ class JSONSchemaVisitor { jsonSchema = { ...jsonSchema, - ...this.getFieldOrScalarDeclarationValidatorsForSchema(field, isScalarUUID) + ...this.getFieldOrScalarDeclarationValidatorsForSchema(field.getType(), isScalarUUID, field.getValidator()) }; // If this field has a default value, add it. @@ -411,6 +408,42 @@ class JSONSchemaVisitor { jsonSchema.description = 'The instance identifier for this type'; } + } else if (ModelUtil.isMap?.(field)) { + const mapDeclaration = field.getParent().getModelFile().getModelManager().getType(field.getFullyQualifiedTypeName()); + + let mapKey = mapDeclaration.getModelFile().getType(mapDeclaration.getKey().getType()); + let mapValue = mapDeclaration.getModelFile().getType(mapDeclaration.getValue().getType()); + + const jsonSchema = { + $id: field.getFullyQualifiedName(), + schema: { + title: mapDeclaration.getName(), + description : `An instance of ${field.getFullyQualifiedTypeName()}`, + type: 'object', + propertyNames: { + type: 'string' + }, + additionalProperties: { + type: this.getFieldOrScalarDeclarationValidatorsForSchema(mapDeclaration.getValue().getType()).type + } + } + }; + + if (mapKey.isScalarDeclaration?.() && mapKey.getValidator()?.getRegex()) { + jsonSchema.schema.propertyNames.pattern = String(mapKey.getValidator().getRegex()); + + } + + if (mapValue.isScalarDeclaration?.() && mapValue.getValidator()?.getRegex()) { + jsonSchema.schema.additionalProperties.pattern = String(mapValue.getValidator().getRegex()); + } + + // if its a ClassDeclaration, add reference to its schema. + if (mapValue?.isClassDeclaration?.()) { + jsonSchema.schema.additionalProperties = {$ref: `#/definitions/${field.getParent().getModelFile().getNamespace().concat(`.${mapDeclaration.getValue().getType()}`)}`}; + } + + return jsonSchema; // Not primitive, so must be a class or enumeration! } else { // Look up the type of the property. diff --git a/test/codegen/__snapshots__/codegen.js.snap b/test/codegen/__snapshots__/codegen.js.snap index c83d6015..068de440 100644 --- a/test/codegen/__snapshots__/codegen.js.snap +++ b/test/codegen/__snapshots__/codegen.js.snap @@ -376,6 +376,7 @@ using AccordProject.Concerto; public class Company : Concept { [System.Text.Json.Serialization.JsonPropertyName("$class")] public override string _class { get; } = "org.acme.hr.Company"; + [System.ComponentModel.DataAnnotations.RegularExpression(@"abc.*", ErrorMessage = "Invalid characters")] public string name { get; set; } public org.acme.hr.base.Address headquarters { get; set; } public Dictionary companyProperties { get; set; } @@ -1684,10 +1685,84 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "description": "The class identifier for org.acme.hr.Company" }, "name": { - "type": "string" + "type": "string", + "pattern": "abc.*" }, "headquarters": { "$ref": "#/definitions/org.acme.hr.base.Address" + }, + "companyProperties": { + "$id": "org.acme.hr.Company.companyProperties", + "schema": { + "title": "CompanyProperties", + "description": "An instance of org.acme.hr.CompanyProperties", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "string" + } + } + }, + "employeeDirectory": { + "$id": "org.acme.hr.Company.employeeDirectory", + "schema": { + "title": "EmployeeDirectory", + "description": "An instance of org.acme.hr.EmployeeDirectory", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr.Employee" + } + } + }, + "employeeTShirtSizes": { + "$id": "org.acme.hr.Company.employeeTShirtSizes", + "schema": { + "title": "EmployeeTShirtSizes", + "description": "An instance of org.acme.hr.base.EmployeeTShirtSizes", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr.TShirtSizeType" + } + } + }, + "employeeProfiles": { + "$id": "org.acme.hr.Company.employeeProfiles", + "schema": { + "title": "EmployeeProfiles", + "description": "An instance of org.acme.hr.EmployeeProfiles", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr.Concept" + } + } + }, + "employeeSocialSecurityNumbers": { + "$id": "org.acme.hr.Company.employeeSocialSecurityNumbers", + "schema": { + "title": "EmployeeSocialSecurityNumbers", + "description": "An instance of org.acme.hr.EmployeeSocialSecurityNumbers", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "SSN", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + } + } } }, "required": [ @@ -1803,6 +1878,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -1893,6 +1982,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -1959,6 +2062,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -2054,6 +2171,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -2803,10 +2934,84 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "description": "The class identifier for org.acme.hr.Company" }, "name": { - "type": "string" + "type": "string", + "pattern": "abc.*" }, "headquarters": { "$ref": "#/components/schemas/org.acme.hr.base.Address" + }, + "companyProperties": { + "$id": "org.acme.hr.Company.companyProperties", + "schema": { + "title": "CompanyProperties", + "description": "An instance of org.acme.hr.CompanyProperties", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "string" + } + } + }, + "employeeDirectory": { + "$id": "org.acme.hr.Company.employeeDirectory", + "schema": { + "title": "EmployeeDirectory", + "description": "An instance of org.acme.hr.EmployeeDirectory", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr.Employee" + } + } + }, + "employeeTShirtSizes": { + "$id": "org.acme.hr.Company.employeeTShirtSizes", + "schema": { + "title": "EmployeeTShirtSizes", + "description": "An instance of org.acme.hr.base.EmployeeTShirtSizes", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr.TShirtSizeType" + } + } + }, + "employeeProfiles": { + "$id": "org.acme.hr.Company.employeeProfiles", + "schema": { + "title": "EmployeeProfiles", + "description": "An instance of org.acme.hr.EmployeeProfiles", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr.Concept" + } + } + }, + "employeeSocialSecurityNumbers": { + "$id": "org.acme.hr.Company.employeeSocialSecurityNumbers", + "schema": { + "title": "EmployeeSocialSecurityNumbers", + "description": "An instance of org.acme.hr.EmployeeSocialSecurityNumbers", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "SSN", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + } + } } }, "required": [ @@ -2922,6 +3127,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -3012,6 +3231,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -3078,6 +3311,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -3173,6 +3420,20 @@ exports[`codegen #formats check we can convert all formats from namespace unvers "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -5928,6 +6189,7 @@ using AccordProject.Concerto; public class Company : Concept { [System.Text.Json.Serialization.JsonPropertyName("$class")] public override string _class { get; } = "org.acme.hr@1.0.0.Company"; + [System.ComponentModel.DataAnnotations.RegularExpression(@"abc.*", ErrorMessage = "Invalid characters")] public string name { get; set; } public org.acme.hr.base.Address headquarters { get; set; } public Dictionary companyProperties { get; set; } @@ -7236,10 +7498,84 @@ exports[`codegen #formats check we can convert all formats from namespace versio "description": "The class identifier for org.acme.hr@1.0.0.Company" }, "name": { - "type": "string" + "type": "string", + "pattern": "abc.*" }, "headquarters": { "$ref": "#/definitions/org.acme.hr.base@1.0.0.Address" + }, + "companyProperties": { + "$id": "org.acme.hr@1.0.0.Company.companyProperties", + "schema": { + "title": "CompanyProperties", + "description": "An instance of org.acme.hr@1.0.0.CompanyProperties", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "string" + } + } + }, + "employeeDirectory": { + "$id": "org.acme.hr@1.0.0.Company.employeeDirectory", + "schema": { + "title": "EmployeeDirectory", + "description": "An instance of org.acme.hr@1.0.0.EmployeeDirectory", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr@1.0.0.Employee" + } + } + }, + "employeeTShirtSizes": { + "$id": "org.acme.hr@1.0.0.Company.employeeTShirtSizes", + "schema": { + "title": "EmployeeTShirtSizes", + "description": "An instance of org.acme.hr.base@1.0.0.EmployeeTShirtSizes", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr@1.0.0.TShirtSizeType" + } + } + }, + "employeeProfiles": { + "$id": "org.acme.hr@1.0.0.Company.employeeProfiles", + "schema": { + "title": "EmployeeProfiles", + "description": "An instance of org.acme.hr@1.0.0.EmployeeProfiles", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr@1.0.0.Concept" + } + } + }, + "employeeSocialSecurityNumbers": { + "$id": "org.acme.hr@1.0.0.Company.employeeSocialSecurityNumbers", + "schema": { + "title": "EmployeeSocialSecurityNumbers", + "description": "An instance of org.acme.hr@1.0.0.EmployeeSocialSecurityNumbers", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "SSN", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + } + } } }, "required": [ @@ -7355,6 +7691,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -7445,6 +7795,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -7511,6 +7875,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -7606,6 +7984,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -8371,10 +8763,84 @@ exports[`codegen #formats check we can convert all formats from namespace versio "description": "The class identifier for org.acme.hr@1.0.0.Company" }, "name": { - "type": "string" + "type": "string", + "pattern": "abc.*" }, "headquarters": { "$ref": "#/components/schemas/org.acme.hr.base@1.0.0.Address" + }, + "companyProperties": { + "$id": "org.acme.hr@1.0.0.Company.companyProperties", + "schema": { + "title": "CompanyProperties", + "description": "An instance of org.acme.hr@1.0.0.CompanyProperties", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "string" + } + } + }, + "employeeDirectory": { + "$id": "org.acme.hr@1.0.0.Company.employeeDirectory", + "schema": { + "title": "EmployeeDirectory", + "description": "An instance of org.acme.hr@1.0.0.EmployeeDirectory", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr@1.0.0.Employee" + } + } + }, + "employeeTShirtSizes": { + "$id": "org.acme.hr@1.0.0.Company.employeeTShirtSizes", + "schema": { + "title": "EmployeeTShirtSizes", + "description": "An instance of org.acme.hr.base@1.0.0.EmployeeTShirtSizes", + "type": "object", + "propertyNames": { + "type": "string", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr@1.0.0.TShirtSizeType" + } + } + }, + "employeeProfiles": { + "$id": "org.acme.hr@1.0.0.Company.employeeProfiles", + "schema": { + "title": "EmployeeProfiles", + "description": "An instance of org.acme.hr@1.0.0.EmployeeProfiles", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "$ref": "#/definitions/org.acme.hr@1.0.0.Concept" + } + } + }, + "employeeSocialSecurityNumbers": { + "$id": "org.acme.hr@1.0.0.Company.employeeSocialSecurityNumbers", + "schema": { + "title": "EmployeeSocialSecurityNumbers", + "description": "An instance of org.acme.hr@1.0.0.EmployeeSocialSecurityNumbers", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "SSN", + "pattern": "/\\\\d{3}-\\\\d{2}-\\\\{4}+/" + } + } } }, "required": [ @@ -8490,6 +8956,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -8580,6 +9060,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -8646,6 +9140,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ @@ -8741,6 +9249,20 @@ exports[`codegen #formats check we can convert all formats from namespace versio "dob": { "format": "date-time", "type": "string" + }, + "nextOfKin": { + "$id": "org.acme.hr@1.0.0.Person.nextOfKin", + "schema": { + "title": "NextOfKin", + "description": "An instance of org.acme.hr@1.0.0.NextOfKin", + "type": "object", + "propertyNames": { + "type": "string" + }, + "additionalProperties": { + "type": "KinTelephone" + } + } } }, "required": [ diff --git a/test/codegen/fromcto/data/model/hr.cto b/test/codegen/fromcto/data/model/hr.cto index cabf4d71..94e62483 100644 --- a/test/codegen/fromcto/data/model/hr.cto +++ b/test/codegen/fromcto/data/model/hr.cto @@ -33,7 +33,7 @@ map EmployeeDirectory { } concept Company { - o String name + o String name regex=/abc.*/ o Address headquarters o CompanyProperties companyProperties optional o EmployeeDirectory employeeDirectory optional diff --git a/test/codegen/fromcto/jsonschema/jsonschemavisitor.js b/test/codegen/fromcto/jsonschema/jsonschemavisitor.js index c4dd1bef..f0fe8774 100644 --- a/test/codegen/fromcto/jsonschema/jsonschemavisitor.js +++ b/test/codegen/fromcto/jsonschema/jsonschemavisitor.js @@ -178,6 +178,63 @@ concept OtherThing identified by id { o String someId } `; + + +const MODEL_MAP_STRING = ` +namespace test + +map Dictionary { + o String + o String +} + +concept Foo { + o Dictionary dictionary +} +`; + +const MODEL_MAP_NUMBER = ` +namespace test + +map Dictionary { + o String + o Double +} + +concept Foo { + o Dictionary dictionary +} +`; + +const MODEL_MAP_BOOLEAN = ` +namespace test + +map Dictionary { + o String + o Boolean +} + +concept Foo { + o Dictionary dictionary +} +`; + +const MODEL_MAP_COMPLEX = ` +namespace test + +concept Person { + o String name +} +map Dictionary { + o String + o Person +} + +concept Foo { + o Dictionary dictionary +} +`; + const MODEL_RELATIONSHIP_SIMPLE = ` namespace test @@ -287,6 +344,56 @@ describe('JSONSchema (samples)', function () { expect(schema.properties.someId.format).to.be.undefined; }); + it('should generate for Map of type ', () => { + const modelManager = new ModelManager(); + modelManager.addCTOModel( MODEL_MAP_STRING ); + const visitor = new JSONSchemaVisitor(); + const schema = modelManager.accept(visitor, { rootType: 'test.Foo'}); + expect(schema.properties.dictionary.schema.title).equal('Dictionary'); + expect(schema.properties.dictionary.schema.description).equal('An instance of test.Dictionary'); + expect(schema.properties.dictionary.schema.type).equal('object'); + expect(schema.properties.dictionary.schema.propertyNames.type).equal('string'); + expect(schema.properties.dictionary.schema.additionalProperties.type).equal('string'); + }); + + + it('should generate for Map of type ', () => { + const modelManager = new ModelManager(); + modelManager.addCTOModel( MODEL_MAP_COMPLEX ); + const visitor = new JSONSchemaVisitor(); + const schema = modelManager.accept(visitor, { rootType: 'test.Foo'}); + expect(schema.properties.dictionary.schema.title).equal('Dictionary'); + expect(schema.properties.dictionary.schema.description).equal('An instance of test.Dictionary'); + expect(schema.properties.dictionary.schema.type).equal('object'); + expect(schema.properties.dictionary.schema.propertyNames.type).equal('string'); + expect(schema.properties.dictionary.schema.additionalProperties.$ref).equal('#/definitions/test.Person'); + }); + + it('should generate for Map of type ', () => { + const modelManager = new ModelManager(); + modelManager.addCTOModel( MODEL_MAP_NUMBER ); + const visitor = new JSONSchemaVisitor(); + const schema = modelManager.accept(visitor, { rootType: 'test.Foo'}); + expect(schema.properties.dictionary.schema.title).equal('Dictionary'); + expect(schema.properties.dictionary.schema.description).equal('An instance of test.Dictionary'); + expect(schema.properties.dictionary.schema.type).equal('object'); + expect(schema.properties.dictionary.schema.propertyNames.type).equal('string'); + expect(schema.properties.dictionary.schema.additionalProperties.type).equal('number'); + }); + + + it('should generate for Map of type ', () => { + const modelManager = new ModelManager(); + modelManager.addCTOModel( MODEL_MAP_BOOLEAN ); + const visitor = new JSONSchemaVisitor(); + const schema = modelManager.accept(visitor, { rootType: 'test.Foo'}); + expect(schema.properties.dictionary.schema.title).equal('Dictionary'); + expect(schema.properties.dictionary.schema.description).equal('An instance of test.Dictionary'); + expect(schema.properties.dictionary.schema.type).equal('object'); + expect(schema.properties.dictionary.schema.propertyNames.type).equal('string'); + expect(schema.properties.dictionary.schema.additionalProperties.type).equal('boolean'); + }); + it('should use min max length for string fields when length validator is configured', () => { const modelManager = new ModelManager(); modelManager.addCTOModel(fs.readFileSync(path.resolve(__dirname, '../data/model/stringlength.cto'), 'utf8'), 'stringlength.cto'); diff --git a/types/lib/codegen/fromcto/jsonschema/jsonschemavisitor.d.ts b/types/lib/codegen/fromcto/jsonschema/jsonschemavisitor.d.ts index 3c6a6ffe..d7b3b146 100644 --- a/types/lib/codegen/fromcto/jsonschema/jsonschemavisitor.d.ts +++ b/types/lib/codegen/fromcto/jsonschema/jsonschemavisitor.d.ts @@ -30,8 +30,9 @@ declare class JSONSchemaVisitor { getDecorators(decorated: object): object; /** * Get the validators for a field or a scalar definition in JSON schema form. - * @param {Object} field - the scalar declaration being visited + * @param {String} type - the field type * @param {bool} [isScalarUUID] - flag to indicate given field type is scalar uuid + * @param {bool} [validator] - the field validator * @return {Object} the result of visiting or null * @private */