UNPKG

20.7 kBJavaScriptView Raw
1function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
2
3function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
4
5function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
6
7import objectValues from "../polyfills/objectValues.mjs";
8import keyMap from "../jsutils/keyMap.mjs";
9import inspect from "../jsutils/inspect.mjs";
10import invariant from "../jsutils/invariant.mjs";
11import naturalCompare from "../jsutils/naturalCompare.mjs";
12import { print } from "../language/printer.mjs";
13import { visit } from "../language/visitor.mjs";
14import { isSpecifiedScalarType } from "../type/scalars.mjs";
15import { isScalarType, isObjectType, isInterfaceType, isUnionType, isEnumType, isInputObjectType, isNonNullType, isListType, isNamedType, isRequiredArgument, isRequiredInputField } from "../type/definition.mjs";
16import { astFromValue } from "./astFromValue.mjs";
17export var BreakingChangeType = Object.freeze({
18 TYPE_REMOVED: 'TYPE_REMOVED',
19 TYPE_CHANGED_KIND: 'TYPE_CHANGED_KIND',
20 TYPE_REMOVED_FROM_UNION: 'TYPE_REMOVED_FROM_UNION',
21 VALUE_REMOVED_FROM_ENUM: 'VALUE_REMOVED_FROM_ENUM',
22 REQUIRED_INPUT_FIELD_ADDED: 'REQUIRED_INPUT_FIELD_ADDED',
23 IMPLEMENTED_INTERFACE_REMOVED: 'IMPLEMENTED_INTERFACE_REMOVED',
24 FIELD_REMOVED: 'FIELD_REMOVED',
25 FIELD_CHANGED_KIND: 'FIELD_CHANGED_KIND',
26 REQUIRED_ARG_ADDED: 'REQUIRED_ARG_ADDED',
27 ARG_REMOVED: 'ARG_REMOVED',
28 ARG_CHANGED_KIND: 'ARG_CHANGED_KIND',
29 DIRECTIVE_REMOVED: 'DIRECTIVE_REMOVED',
30 DIRECTIVE_ARG_REMOVED: 'DIRECTIVE_ARG_REMOVED',
31 REQUIRED_DIRECTIVE_ARG_ADDED: 'REQUIRED_DIRECTIVE_ARG_ADDED',
32 DIRECTIVE_REPEATABLE_REMOVED: 'DIRECTIVE_REPEATABLE_REMOVED',
33 DIRECTIVE_LOCATION_REMOVED: 'DIRECTIVE_LOCATION_REMOVED'
34});
35export var DangerousChangeType = Object.freeze({
36 VALUE_ADDED_TO_ENUM: 'VALUE_ADDED_TO_ENUM',
37 TYPE_ADDED_TO_UNION: 'TYPE_ADDED_TO_UNION',
38 OPTIONAL_INPUT_FIELD_ADDED: 'OPTIONAL_INPUT_FIELD_ADDED',
39 OPTIONAL_ARG_ADDED: 'OPTIONAL_ARG_ADDED',
40 IMPLEMENTED_INTERFACE_ADDED: 'IMPLEMENTED_INTERFACE_ADDED',
41 ARG_DEFAULT_VALUE_CHANGE: 'ARG_DEFAULT_VALUE_CHANGE'
42});
43
44/**
45 * Given two schemas, returns an Array containing descriptions of all the types
46 * of breaking changes covered by the other functions down below.
47 */
48export function findBreakingChanges(oldSchema, newSchema) {
49 var breakingChanges = findSchemaChanges(oldSchema, newSchema).filter(function (change) {
50 return change.type in BreakingChangeType;
51 });
52 return breakingChanges;
53}
54/**
55 * Given two schemas, returns an Array containing descriptions of all the types
56 * of potentially dangerous changes covered by the other functions down below.
57 */
58
59export function findDangerousChanges(oldSchema, newSchema) {
60 var dangerousChanges = findSchemaChanges(oldSchema, newSchema).filter(function (change) {
61 return change.type in DangerousChangeType;
62 });
63 return dangerousChanges;
64}
65
66function findSchemaChanges(oldSchema, newSchema) {
67 return [].concat(findTypeChanges(oldSchema, newSchema), findDirectiveChanges(oldSchema, newSchema));
68}
69
70function findDirectiveChanges(oldSchema, newSchema) {
71 var schemaChanges = [];
72 var directivesDiff = diff(oldSchema.getDirectives(), newSchema.getDirectives());
73
74 for (var _i2 = 0, _directivesDiff$remov2 = directivesDiff.removed; _i2 < _directivesDiff$remov2.length; _i2++) {
75 var oldDirective = _directivesDiff$remov2[_i2];
76 schemaChanges.push({
77 type: BreakingChangeType.DIRECTIVE_REMOVED,
78 description: "".concat(oldDirective.name, " was removed.")
79 });
80 }
81
82 for (var _i4 = 0, _directivesDiff$persi2 = directivesDiff.persisted; _i4 < _directivesDiff$persi2.length; _i4++) {
83 var _ref2 = _directivesDiff$persi2[_i4];
84 var _oldDirective = _ref2[0];
85 var newDirective = _ref2[1];
86 var argsDiff = diff(_oldDirective.args, newDirective.args);
87
88 for (var _i6 = 0, _argsDiff$added2 = argsDiff.added; _i6 < _argsDiff$added2.length; _i6++) {
89 var newArg = _argsDiff$added2[_i6];
90
91 if (isRequiredArgument(newArg)) {
92 schemaChanges.push({
93 type: BreakingChangeType.REQUIRED_DIRECTIVE_ARG_ADDED,
94 description: "A required arg ".concat(newArg.name, " on directive ").concat(_oldDirective.name, " was added.")
95 });
96 }
97 }
98
99 for (var _i8 = 0, _argsDiff$removed2 = argsDiff.removed; _i8 < _argsDiff$removed2.length; _i8++) {
100 var oldArg = _argsDiff$removed2[_i8];
101 schemaChanges.push({
102 type: BreakingChangeType.DIRECTIVE_ARG_REMOVED,
103 description: "".concat(oldArg.name, " was removed from ").concat(_oldDirective.name, ".")
104 });
105 }
106
107 if (_oldDirective.isRepeatable && !newDirective.isRepeatable) {
108 schemaChanges.push({
109 type: BreakingChangeType.DIRECTIVE_REPEATABLE_REMOVED,
110 description: "Repeatable flag was removed from ".concat(_oldDirective.name, ".")
111 });
112 }
113
114 for (var _i10 = 0, _oldDirective$locatio2 = _oldDirective.locations; _i10 < _oldDirective$locatio2.length; _i10++) {
115 var location = _oldDirective$locatio2[_i10];
116
117 if (newDirective.locations.indexOf(location) === -1) {
118 schemaChanges.push({
119 type: BreakingChangeType.DIRECTIVE_LOCATION_REMOVED,
120 description: "".concat(location, " was removed from ").concat(_oldDirective.name, ".")
121 });
122 }
123 }
124 }
125
126 return schemaChanges;
127}
128
129function findTypeChanges(oldSchema, newSchema) {
130 var schemaChanges = [];
131 var typesDiff = diff(objectValues(oldSchema.getTypeMap()), objectValues(newSchema.getTypeMap()));
132
133 for (var _i12 = 0, _typesDiff$removed2 = typesDiff.removed; _i12 < _typesDiff$removed2.length; _i12++) {
134 var oldType = _typesDiff$removed2[_i12];
135 schemaChanges.push({
136 type: BreakingChangeType.TYPE_REMOVED,
137 description: isSpecifiedScalarType(oldType) ? "Standard scalar ".concat(oldType.name, " was removed because it is not referenced anymore.") : "".concat(oldType.name, " was removed.")
138 });
139 }
140
141 for (var _i14 = 0, _typesDiff$persisted2 = typesDiff.persisted; _i14 < _typesDiff$persisted2.length; _i14++) {
142 var _ref4 = _typesDiff$persisted2[_i14];
143 var _oldType = _ref4[0];
144 var newType = _ref4[1];
145
146 if (isEnumType(_oldType) && isEnumType(newType)) {
147 schemaChanges.push.apply(schemaChanges, findEnumTypeChanges(_oldType, newType));
148 } else if (isUnionType(_oldType) && isUnionType(newType)) {
149 schemaChanges.push.apply(schemaChanges, findUnionTypeChanges(_oldType, newType));
150 } else if (isInputObjectType(_oldType) && isInputObjectType(newType)) {
151 schemaChanges.push.apply(schemaChanges, findInputObjectTypeChanges(_oldType, newType));
152 } else if (isObjectType(_oldType) && isObjectType(newType)) {
153 schemaChanges.push.apply(schemaChanges, findFieldChanges(_oldType, newType).concat(findImplementedInterfacesChanges(_oldType, newType)));
154 } else if (isInterfaceType(_oldType) && isInterfaceType(newType)) {
155 schemaChanges.push.apply(schemaChanges, findFieldChanges(_oldType, newType).concat(findImplementedInterfacesChanges(_oldType, newType)));
156 } else if (_oldType.constructor !== newType.constructor) {
157 schemaChanges.push({
158 type: BreakingChangeType.TYPE_CHANGED_KIND,
159 description: "".concat(_oldType.name, " changed from ") + "".concat(typeKindName(_oldType), " to ").concat(typeKindName(newType), ".")
160 });
161 }
162 }
163
164 return schemaChanges;
165}
166
167function findInputObjectTypeChanges(oldType, newType) {
168 var schemaChanges = [];
169 var fieldsDiff = diff(objectValues(oldType.getFields()), objectValues(newType.getFields()));
170
171 for (var _i16 = 0, _fieldsDiff$added2 = fieldsDiff.added; _i16 < _fieldsDiff$added2.length; _i16++) {
172 var newField = _fieldsDiff$added2[_i16];
173
174 if (isRequiredInputField(newField)) {
175 schemaChanges.push({
176 type: BreakingChangeType.REQUIRED_INPUT_FIELD_ADDED,
177 description: "A required field ".concat(newField.name, " on input type ").concat(oldType.name, " was added.")
178 });
179 } else {
180 schemaChanges.push({
181 type: DangerousChangeType.OPTIONAL_INPUT_FIELD_ADDED,
182 description: "An optional field ".concat(newField.name, " on input type ").concat(oldType.name, " was added.")
183 });
184 }
185 }
186
187 for (var _i18 = 0, _fieldsDiff$removed2 = fieldsDiff.removed; _i18 < _fieldsDiff$removed2.length; _i18++) {
188 var oldField = _fieldsDiff$removed2[_i18];
189 schemaChanges.push({
190 type: BreakingChangeType.FIELD_REMOVED,
191 description: "".concat(oldType.name, ".").concat(oldField.name, " was removed.")
192 });
193 }
194
195 for (var _i20 = 0, _fieldsDiff$persisted2 = fieldsDiff.persisted; _i20 < _fieldsDiff$persisted2.length; _i20++) {
196 var _ref6 = _fieldsDiff$persisted2[_i20];
197 var _oldField = _ref6[0];
198 var _newField = _ref6[1];
199 var isSafe = isChangeSafeForInputObjectFieldOrFieldArg(_oldField.type, _newField.type);
200
201 if (!isSafe) {
202 schemaChanges.push({
203 type: BreakingChangeType.FIELD_CHANGED_KIND,
204 description: "".concat(oldType.name, ".").concat(_oldField.name, " changed type from ") + "".concat(String(_oldField.type), " to ").concat(String(_newField.type), ".")
205 });
206 }
207 }
208
209 return schemaChanges;
210}
211
212function findUnionTypeChanges(oldType, newType) {
213 var schemaChanges = [];
214 var possibleTypesDiff = diff(oldType.getTypes(), newType.getTypes());
215
216 for (var _i22 = 0, _possibleTypesDiff$ad2 = possibleTypesDiff.added; _i22 < _possibleTypesDiff$ad2.length; _i22++) {
217 var newPossibleType = _possibleTypesDiff$ad2[_i22];
218 schemaChanges.push({
219 type: DangerousChangeType.TYPE_ADDED_TO_UNION,
220 description: "".concat(newPossibleType.name, " was added to union type ").concat(oldType.name, ".")
221 });
222 }
223
224 for (var _i24 = 0, _possibleTypesDiff$re2 = possibleTypesDiff.removed; _i24 < _possibleTypesDiff$re2.length; _i24++) {
225 var oldPossibleType = _possibleTypesDiff$re2[_i24];
226 schemaChanges.push({
227 type: BreakingChangeType.TYPE_REMOVED_FROM_UNION,
228 description: "".concat(oldPossibleType.name, " was removed from union type ").concat(oldType.name, ".")
229 });
230 }
231
232 return schemaChanges;
233}
234
235function findEnumTypeChanges(oldType, newType) {
236 var schemaChanges = [];
237 var valuesDiff = diff(oldType.getValues(), newType.getValues());
238
239 for (var _i26 = 0, _valuesDiff$added2 = valuesDiff.added; _i26 < _valuesDiff$added2.length; _i26++) {
240 var newValue = _valuesDiff$added2[_i26];
241 schemaChanges.push({
242 type: DangerousChangeType.VALUE_ADDED_TO_ENUM,
243 description: "".concat(newValue.name, " was added to enum type ").concat(oldType.name, ".")
244 });
245 }
246
247 for (var _i28 = 0, _valuesDiff$removed2 = valuesDiff.removed; _i28 < _valuesDiff$removed2.length; _i28++) {
248 var oldValue = _valuesDiff$removed2[_i28];
249 schemaChanges.push({
250 type: BreakingChangeType.VALUE_REMOVED_FROM_ENUM,
251 description: "".concat(oldValue.name, " was removed from enum type ").concat(oldType.name, ".")
252 });
253 }
254
255 return schemaChanges;
256}
257
258function findImplementedInterfacesChanges(oldType, newType) {
259 var schemaChanges = [];
260 var interfacesDiff = diff(oldType.getInterfaces(), newType.getInterfaces());
261
262 for (var _i30 = 0, _interfacesDiff$added2 = interfacesDiff.added; _i30 < _interfacesDiff$added2.length; _i30++) {
263 var newInterface = _interfacesDiff$added2[_i30];
264 schemaChanges.push({
265 type: DangerousChangeType.IMPLEMENTED_INTERFACE_ADDED,
266 description: "".concat(newInterface.name, " added to interfaces implemented by ").concat(oldType.name, ".")
267 });
268 }
269
270 for (var _i32 = 0, _interfacesDiff$remov2 = interfacesDiff.removed; _i32 < _interfacesDiff$remov2.length; _i32++) {
271 var oldInterface = _interfacesDiff$remov2[_i32];
272 schemaChanges.push({
273 type: BreakingChangeType.IMPLEMENTED_INTERFACE_REMOVED,
274 description: "".concat(oldType.name, " no longer implements interface ").concat(oldInterface.name, ".")
275 });
276 }
277
278 return schemaChanges;
279}
280
281function findFieldChanges(oldType, newType) {
282 var schemaChanges = [];
283 var fieldsDiff = diff(objectValues(oldType.getFields()), objectValues(newType.getFields()));
284
285 for (var _i34 = 0, _fieldsDiff$removed4 = fieldsDiff.removed; _i34 < _fieldsDiff$removed4.length; _i34++) {
286 var oldField = _fieldsDiff$removed4[_i34];
287 schemaChanges.push({
288 type: BreakingChangeType.FIELD_REMOVED,
289 description: "".concat(oldType.name, ".").concat(oldField.name, " was removed.")
290 });
291 }
292
293 for (var _i36 = 0, _fieldsDiff$persisted4 = fieldsDiff.persisted; _i36 < _fieldsDiff$persisted4.length; _i36++) {
294 var _ref8 = _fieldsDiff$persisted4[_i36];
295 var _oldField2 = _ref8[0];
296 var newField = _ref8[1];
297 schemaChanges.push.apply(schemaChanges, findArgChanges(oldType, _oldField2, newField));
298 var isSafe = isChangeSafeForObjectOrInterfaceField(_oldField2.type, newField.type);
299
300 if (!isSafe) {
301 schemaChanges.push({
302 type: BreakingChangeType.FIELD_CHANGED_KIND,
303 description: "".concat(oldType.name, ".").concat(_oldField2.name, " changed type from ") + "".concat(String(_oldField2.type), " to ").concat(String(newField.type), ".")
304 });
305 }
306 }
307
308 return schemaChanges;
309}
310
311function findArgChanges(oldType, oldField, newField) {
312 var schemaChanges = [];
313 var argsDiff = diff(oldField.args, newField.args);
314
315 for (var _i38 = 0, _argsDiff$removed4 = argsDiff.removed; _i38 < _argsDiff$removed4.length; _i38++) {
316 var oldArg = _argsDiff$removed4[_i38];
317 schemaChanges.push({
318 type: BreakingChangeType.ARG_REMOVED,
319 description: "".concat(oldType.name, ".").concat(oldField.name, " arg ").concat(oldArg.name, " was removed.")
320 });
321 }
322
323 for (var _i40 = 0, _argsDiff$persisted2 = argsDiff.persisted; _i40 < _argsDiff$persisted2.length; _i40++) {
324 var _ref10 = _argsDiff$persisted2[_i40];
325 var _oldArg = _ref10[0];
326 var newArg = _ref10[1];
327 var isSafe = isChangeSafeForInputObjectFieldOrFieldArg(_oldArg.type, newArg.type);
328
329 if (!isSafe) {
330 schemaChanges.push({
331 type: BreakingChangeType.ARG_CHANGED_KIND,
332 description: "".concat(oldType.name, ".").concat(oldField.name, " arg ").concat(_oldArg.name, " has changed type from ") + "".concat(String(_oldArg.type), " to ").concat(String(newArg.type), ".")
333 });
334 } else if (_oldArg.defaultValue !== undefined) {
335 if (newArg.defaultValue === undefined) {
336 schemaChanges.push({
337 type: DangerousChangeType.ARG_DEFAULT_VALUE_CHANGE,
338 description: "".concat(oldType.name, ".").concat(oldField.name, " arg ").concat(_oldArg.name, " defaultValue was removed.")
339 });
340 } else {
341 // Since we looking only for client's observable changes we should
342 // compare default values in the same representation as they are
343 // represented inside introspection.
344 var oldValueStr = stringifyValue(_oldArg.defaultValue, _oldArg.type);
345 var newValueStr = stringifyValue(newArg.defaultValue, newArg.type);
346
347 if (oldValueStr !== newValueStr) {
348 schemaChanges.push({
349 type: DangerousChangeType.ARG_DEFAULT_VALUE_CHANGE,
350 description: "".concat(oldType.name, ".").concat(oldField.name, " arg ").concat(_oldArg.name, " has changed defaultValue from ").concat(oldValueStr, " to ").concat(newValueStr, ".")
351 });
352 }
353 }
354 }
355 }
356
357 for (var _i42 = 0, _argsDiff$added4 = argsDiff.added; _i42 < _argsDiff$added4.length; _i42++) {
358 var _newArg = _argsDiff$added4[_i42];
359
360 if (isRequiredArgument(_newArg)) {
361 schemaChanges.push({
362 type: BreakingChangeType.REQUIRED_ARG_ADDED,
363 description: "A required arg ".concat(_newArg.name, " on ").concat(oldType.name, ".").concat(oldField.name, " was added.")
364 });
365 } else {
366 schemaChanges.push({
367 type: DangerousChangeType.OPTIONAL_ARG_ADDED,
368 description: "An optional arg ".concat(_newArg.name, " on ").concat(oldType.name, ".").concat(oldField.name, " was added.")
369 });
370 }
371 }
372
373 return schemaChanges;
374}
375
376function isChangeSafeForObjectOrInterfaceField(oldType, newType) {
377 if (isListType(oldType)) {
378 return (// if they're both lists, make sure the underlying types are compatible
379 isListType(newType) && isChangeSafeForObjectOrInterfaceField(oldType.ofType, newType.ofType) || // moving from nullable to non-null of the same underlying type is safe
380 isNonNullType(newType) && isChangeSafeForObjectOrInterfaceField(oldType, newType.ofType)
381 );
382 }
383
384 if (isNonNullType(oldType)) {
385 // if they're both non-null, make sure the underlying types are compatible
386 return isNonNullType(newType) && isChangeSafeForObjectOrInterfaceField(oldType.ofType, newType.ofType);
387 }
388
389 return (// if they're both named types, see if their names are equivalent
390 isNamedType(newType) && oldType.name === newType.name || // moving from nullable to non-null of the same underlying type is safe
391 isNonNullType(newType) && isChangeSafeForObjectOrInterfaceField(oldType, newType.ofType)
392 );
393}
394
395function isChangeSafeForInputObjectFieldOrFieldArg(oldType, newType) {
396 if (isListType(oldType)) {
397 // if they're both lists, make sure the underlying types are compatible
398 return isListType(newType) && isChangeSafeForInputObjectFieldOrFieldArg(oldType.ofType, newType.ofType);
399 }
400
401 if (isNonNullType(oldType)) {
402 return (// if they're both non-null, make sure the underlying types are
403 // compatible
404 isNonNullType(newType) && isChangeSafeForInputObjectFieldOrFieldArg(oldType.ofType, newType.ofType) || // moving from non-null to nullable of the same underlying type is safe
405 !isNonNullType(newType) && isChangeSafeForInputObjectFieldOrFieldArg(oldType.ofType, newType)
406 );
407 } // if they're both named types, see if their names are equivalent
408
409
410 return isNamedType(newType) && oldType.name === newType.name;
411}
412
413function typeKindName(type) {
414 if (isScalarType(type)) {
415 return 'a Scalar type';
416 }
417
418 if (isObjectType(type)) {
419 return 'an Object type';
420 }
421
422 if (isInterfaceType(type)) {
423 return 'an Interface type';
424 }
425
426 if (isUnionType(type)) {
427 return 'a Union type';
428 }
429
430 if (isEnumType(type)) {
431 return 'an Enum type';
432 } // istanbul ignore else (See: 'https://github.com/graphql/graphql-js/issues/2618')
433
434
435 if (isInputObjectType(type)) {
436 return 'an Input type';
437 } // istanbul ignore next (Not reachable. All possible named types have been considered)
438
439
440 false || invariant(0, 'Unexpected type: ' + inspect(type));
441}
442
443function stringifyValue(value, type) {
444 var ast = astFromValue(value, type);
445 ast != null || invariant(0);
446 var sortedAST = visit(ast, {
447 ObjectValue: function ObjectValue(objectNode) {
448 // Make a copy since sort mutates array
449 var fields = [].concat(objectNode.fields);
450 fields.sort(function (fieldA, fieldB) {
451 return naturalCompare(fieldA.name.value, fieldB.name.value);
452 });
453 return _objectSpread(_objectSpread({}, objectNode), {}, {
454 fields: fields
455 });
456 }
457 });
458 return print(sortedAST);
459}
460
461function diff(oldArray, newArray) {
462 var added = [];
463 var removed = [];
464 var persisted = [];
465 var oldMap = keyMap(oldArray, function (_ref11) {
466 var name = _ref11.name;
467 return name;
468 });
469 var newMap = keyMap(newArray, function (_ref12) {
470 var name = _ref12.name;
471 return name;
472 });
473
474 for (var _i44 = 0; _i44 < oldArray.length; _i44++) {
475 var oldItem = oldArray[_i44];
476 var newItem = newMap[oldItem.name];
477
478 if (newItem === undefined) {
479 removed.push(oldItem);
480 } else {
481 persisted.push([oldItem, newItem]);
482 }
483 }
484
485 for (var _i46 = 0; _i46 < newArray.length; _i46++) {
486 var _newItem = newArray[_i46];
487
488 if (oldMap[_newItem.name] === undefined) {
489 added.push(_newItem);
490 }
491 }
492
493 return {
494 added: added,
495 persisted: persisted,
496 removed: removed
497 };
498}