-
Notifications
You must be signed in to change notification settings - Fork 92
/
Copy pathcomposed-schema-restrictions.js
89 lines (82 loc) · 3.21 KB
/
composed-schema-restrictions.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
const {
validateSubschemas,
getPropertySchemasByName,
getSchemaType,
isObjectSchema,
isArraySchema,
isEnumerationSchema,
isObject,
SchemaType
} = require('../utils');
module.exports = function(schema, options, { path }) {
return validateSubschemas(schema, path, composedSchemaRestrictions);
};
/**
* Checks to make sure properties across a composed object schema have types that are deemed
* compatible for modeling that schema.
*
* @param {*} schema the schema to check
* @param {*} path the array of path segments indicating the location of "schema" within the API definition
* @returns an array containing the violations found or [] if no violations
*/
function composedSchemaRestrictions(schema, path) {
const errors = [];
// Only object schemas have properties
if (isObjectSchema(schema)) {
// Collects all composed property schemas indexed by the name of the property they define
const schemasByName = getPropertySchemasByName(schema);
for (const propertyName in schemasByName) {
// The reducer will result in a `false` sentinel if two schemas for the same property
// are not deemed compatible
if (
schemasByName[propertyName].reduce(schemaCompatibilityReducer) === false
) {
errors.push({
message: `SDK generation may fail due to incompatible types for property across composite object schema: ${propertyName}`,
path
});
}
}
}
return errors;
}
/**
* Reducer for an array of schemas; for each pair of schemas returns one of them if they're deemed
* compatible and a `false` sentinel otherwise. The `false` sentinel is guaranteed to propagate.
*
* @param {*} left the "left" schema in the comparison
* @param {*} right the "right" schema in the comparison
* @returns a schema for the next comparison if the two are compatible, `false` otherwise
*/
function schemaCompatibilityReducer(left, right) {
return getComparandum(left) === getComparandum(right) ? left : false;
}
/**
* Returns the value appropriate to compare a schema to another schema. This is dependent on type.
* - For indeterminate schemas, deem incomparable and return a unique value
* - For object schemas, deem compatible only for an exact match and return the schema itself
* - For array schemas, deem compatible if their items are compatible (recursive)
* - For enumeration schemas, deem them compatible with strings
* - For all other schemas, deem them compatible with schemas of the same derived type and return it
*
* @param {*} schema the schema from which to derive a "comparandum" to compare with other schemas
* @returns a "comparandum" value to compare with other schemas
*/
function getComparandum(schema) {
if (!isObject(schema) || getSchemaType(schema) === SchemaType.UNKNOWN) {
// not compatible with any other schema
return Symbol();
}
if (isObjectSchema(schema)) {
// only compatible with same exact schema
return schema;
} else if (isArraySchema(schema)) {
// compatible with other arrays whose values are compatible
return getComparandum(schema.items);
} else if (isEnumerationSchema(schema)) {
// compatible with all strings
return SchemaType.STRING;
} else {
return getSchemaType(schema);
}
}