relay-compiler 10.0.1 → 10.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (89) hide show
  1. package/bin/relay-compiler +1347 -604
  2. package/codegen/NormalizationCodeGenerator.js.flow +12 -4
  3. package/codegen/ReaderCodeGenerator.js.flow +38 -3
  4. package/codegen/RelayFileWriter.js.flow +2 -0
  5. package/codegen/writeRelayGeneratedFile.js.flow +1 -1
  6. package/core/ASTCache.js.flow +1 -0
  7. package/core/CompilerContext.js.flow +1 -0
  8. package/core/IR.js.flow +0 -1
  9. package/core/IRPrinter.js.flow +3 -8
  10. package/core/RelayIRTransforms.js.flow +7 -0
  11. package/core/Schema.js.flow +55 -1
  12. package/index.js +1 -1
  13. package/language/javascript/FindGraphQLTags.js.flow +2 -97
  14. package/language/javascript/RelayFlowBabelFactories.js.flow +11 -15
  15. package/language/javascript/RelayFlowGenerator.js.flow +76 -19
  16. package/language/javascript/RelayFlowTypeTransformers.js.flow +4 -4
  17. package/lib/bin/RelayCompilerMain.js +5 -5
  18. package/lib/codegen/CodegenRunner.js +2 -2
  19. package/lib/codegen/NormalizationCodeGenerator.js +20 -8
  20. package/lib/codegen/ReaderCodeGenerator.js +43 -8
  21. package/lib/codegen/RelayFileWriter.js +2 -2
  22. package/lib/codegen/compileRelayArtifacts.js +2 -2
  23. package/lib/codegen/sortObjectByKey.js +2 -2
  24. package/lib/codegen/writeRelayGeneratedFile.js +2 -2
  25. package/lib/core/ASTCache.js +1 -0
  26. package/lib/core/CompilerContext.js +1 -0
  27. package/lib/core/CompilerError.js +2 -2
  28. package/lib/core/IRPrinter.js +3 -4
  29. package/lib/core/RelayGraphQLEnumsGenerator.js +2 -2
  30. package/lib/core/RelayIRTransforms.js +10 -4
  31. package/lib/core/RelayParser.js +4 -4
  32. package/lib/core/Schema.js +48 -7
  33. package/lib/core/getFieldDefinition.js +2 -2
  34. package/lib/core/inferRootArgumentDefinitions.js +4 -4
  35. package/lib/language/javascript/FindGraphQLTags.js +3 -69
  36. package/lib/language/javascript/RelayFlowBabelFactories.js +5 -5
  37. package/lib/language/javascript/RelayFlowGenerator.js +80 -21
  38. package/lib/runner/Artifacts.js +2 -2
  39. package/lib/runner/BufferedFilesystem.js +2 -2
  40. package/lib/runner/GraphQLASTNodeGroup.js +2 -2
  41. package/lib/runner/GraphQLNodeMap.js +2 -2
  42. package/lib/runner/Sources.js +21 -2
  43. package/lib/runner/StrictMap.js +2 -2
  44. package/lib/runner/getChangedNodeNames.js +2 -2
  45. package/lib/transforms/ApplyFragmentArgumentTransform.js +14 -14
  46. package/lib/transforms/ClientExtensionsTransform.js +5 -3
  47. package/lib/transforms/ConnectionTransform.js +8 -9
  48. package/lib/transforms/DeclarativeConnectionMutationTransform.js +113 -55
  49. package/lib/transforms/DeferStreamTransform.js +2 -2
  50. package/lib/transforms/DisallowTypenameOnRoot.js +2 -2
  51. package/lib/transforms/FieldHandleTransform.js +2 -2
  52. package/lib/transforms/FilterCompilerDirectivesTransform.js +29 -0
  53. package/lib/transforms/FlattenTransform.js +13 -12
  54. package/lib/transforms/GenerateIDFieldTransform.js +2 -2
  55. package/lib/transforms/GenerateTypeNameTransform.js +3 -3
  56. package/lib/transforms/InlineDataFragmentTransform.js +2 -2
  57. package/lib/transforms/MaskTransform.js +3 -3
  58. package/lib/transforms/MatchTransform.js +7 -3
  59. package/lib/transforms/ReactFlightComponentTransform.js +162 -0
  60. package/lib/transforms/RefetchableFragmentTransform.js +4 -4
  61. package/lib/transforms/RelayDirectiveTransform.js +2 -2
  62. package/lib/transforms/RequiredFieldTransform.js +380 -0
  63. package/lib/transforms/SkipHandleFieldTransform.js +1 -1
  64. package/lib/transforms/SkipRedundantNodesTransform.js +3 -1
  65. package/lib/transforms/SkipUnreachableNodeTransform.js +1 -1
  66. package/lib/transforms/SkipUnusedVariablesTransform.js +3 -3
  67. package/lib/transforms/TestOperationTransform.js +2 -2
  68. package/lib/transforms/ValidateGlobalVariablesTransform.js +2 -2
  69. package/lib/transforms/ValidateRequiredArgumentsTransform.js +2 -2
  70. package/lib/transforms/ValidateServerOnlyDirectivesTransform.js +2 -2
  71. package/lib/transforms/ValidateUnusedVariablesTransform.js +2 -2
  72. package/lib/transforms/query-generators/FetchableQueryGenerator.js +1 -1
  73. package/lib/transforms/query-generators/NodeQueryGenerator.js +1 -1
  74. package/lib/transforms/query-generators/index.js +2 -2
  75. package/lib/transforms/query-generators/utils.js +2 -2
  76. package/package.json +3 -3
  77. package/relay-compiler.js +4 -4
  78. package/relay-compiler.min.js +4 -4
  79. package/runner/Sources.js.flow +14 -0
  80. package/transforms/ClientExtensionsTransform.js.flow +2 -0
  81. package/transforms/ConnectionTransform.js.flow +0 -1
  82. package/transforms/DeclarativeConnectionMutationTransform.js.flow +137 -48
  83. package/transforms/FieldHandleTransform.js.flow +0 -1
  84. package/transforms/FilterCompilerDirectivesTransform.js.flow +33 -0
  85. package/transforms/FlattenTransform.js.flow +3 -2
  86. package/transforms/MatchTransform.js.flow +6 -0
  87. package/transforms/ReactFlightComponentTransform.js.flow +195 -0
  88. package/transforms/RequiredFieldTransform.js.flow +415 -0
  89. package/transforms/SkipRedundantNodesTransform.js.flow +3 -0
@@ -0,0 +1,162 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ *
7
+ *
8
+ * @format
9
+ * @emails oncall+relay
10
+ */
11
+ 'use strict';
12
+
13
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
14
+
15
+ var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
16
+
17
+ function 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; }
18
+
19
+ function _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) { (0, _defineProperty2["default"])(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; }
20
+
21
+ var IRTransformer = require('../core/IRTransformer');
22
+
23
+ var _require = require('../core/CompilerError'),
24
+ createUserError = _require.createUserError,
25
+ createCompilerError = _require.createCompilerError;
26
+
27
+ var _require2 = require('relay-runtime'),
28
+ RelayFeatureFlags = _require2.RelayFeatureFlags;
29
+
30
+ var FLIGHT_FIELD_COMPONENT_ARGUMENT_TYPE = 'String';
31
+ var FLIGHT_FIELD_COMPONENT_ARGUMENT_NAME = 'component';
32
+ var FLIGHT_FIELD_PROPS_ARGUMENT_NAME = 'props';
33
+ var FLIGHT_FIELD_PROPS_TYPE = 'ReactFlightProps';
34
+ var FLIGHT_FIELD_RETURN_TYPE = 'ReactFlightComponent';
35
+
36
+ /**
37
+ * Experimental transform for React Flight.
38
+ */
39
+ function reactFlightComponentTransform(context) {
40
+ var schema = context.getSchema();
41
+ var propsType = schema.getTypeFromString(FLIGHT_FIELD_PROPS_TYPE);
42
+ propsType = propsType ? schema.asInputType(propsType) : null;
43
+ var componentType = schema.getTypeFromString(FLIGHT_FIELD_RETURN_TYPE);
44
+ componentType = componentType ? schema.asScalarFieldType(componentType) : null;
45
+
46
+ if (!RelayFeatureFlags.ENABLE_REACT_FLIGHT_COMPONENT_FIELD || propsType == null || componentType == null) {
47
+ return context;
48
+ }
49
+
50
+ var types = {
51
+ propsType: propsType,
52
+ componentType: componentType
53
+ };
54
+ return IRTransformer.transform(context, {
55
+ ScalarField: visitScalarField,
56
+ LinkedField: visitLinkedField,
57
+ InlineFragment: visitInlineFragment
58
+ }, function (node) {
59
+ return {
60
+ parentType: node.type,
61
+ types: types
62
+ };
63
+ });
64
+ }
65
+
66
+ function visitInlineFragment(fragment, state) {
67
+ var _fragment$typeConditi;
68
+
69
+ return this.traverse(fragment, {
70
+ parentType: (_fragment$typeConditi = fragment.typeCondition) !== null && _fragment$typeConditi !== void 0 ? _fragment$typeConditi : state.parentType,
71
+ types: state.types
72
+ });
73
+ }
74
+
75
+ function visitLinkedField(field, state) {
76
+ return this.traverse(field, {
77
+ parentType: field.type,
78
+ types: state.types
79
+ });
80
+ }
81
+
82
+ function visitScalarField(field, state) {
83
+ // use the return type to quickly determine if this is a flight field
84
+ var schema = this.getContext().getSchema();
85
+
86
+ if (schema.getRawType(field.type) !== state.types.componentType) {
87
+ return field;
88
+ } // get the name of the component that provides this field
89
+
90
+
91
+ var clientField = schema.getFieldByName(state.parentType, field.name);
92
+
93
+ if (clientField == null) {
94
+ throw createCompilerError("Definition not found for field '".concat(schema.getTypeString(state.parentType), ".").concat(field.name, "'"), [field.loc]);
95
+ }
96
+
97
+ var componentDirective = clientField.directives.find(function (directive) {
98
+ return directive.name === 'react_flight_component';
99
+ });
100
+ var componentNameArg = componentDirective === null || componentDirective === void 0 ? void 0 : componentDirective.args.find(function (arg) {
101
+ return arg.name === 'name';
102
+ });
103
+
104
+ if (componentNameArg == null || componentNameArg.value.kind !== 'StringValue' || typeof componentNameArg.value.value !== 'string') {
105
+ throw createUserError('Invalid Flight field, expected the schema extension to specify ' + "the component's module name with the '@react_flight_component' directive", [field.loc]);
106
+ }
107
+
108
+ var componentName = componentNameArg.value.value; // validate that the parent type has a `flight(component, props)` field
109
+
110
+ var flightField = schema.getFieldByName(state.parentType, 'flight');
111
+
112
+ if (flightField == null) {
113
+ throw createUserError("Invalid Flight field, expected the parent type '".concat(schema.getTypeString(state.parentType), "' ") + "to define a 'flight(component: String, props: ReactFlightProps): ReactFlightComponent' field", [field.loc]);
114
+ }
115
+
116
+ var componentArg = flightField.args.get(FLIGHT_FIELD_COMPONENT_ARGUMENT_NAME);
117
+ var propsArg = flightField.args.get(FLIGHT_FIELD_PROPS_ARGUMENT_NAME);
118
+
119
+ if (componentArg == null || propsArg == null || schema.getRawType(componentArg.type) !== schema.getTypeFromString(FLIGHT_FIELD_COMPONENT_ARGUMENT_TYPE) || schema.getRawType(propsArg.type) !== state.types.propsType || schema.getRawType(flightField.type) !== state.types.componentType) {
120
+ throw createUserError("Invalid Flight field, expected the parent type '".concat(schema.getTypeString(state.parentType), "' ") + "to define a 'flight(component: String, props: ReactFlightProps): ReactFlightComponent' field", [field.loc]);
121
+ }
122
+
123
+ return _objectSpread(_objectSpread({}, field), {}, {
124
+ name: 'flight',
125
+ args: [{
126
+ kind: 'Argument',
127
+ loc: field.loc,
128
+ name: FLIGHT_FIELD_COMPONENT_ARGUMENT_NAME,
129
+ type: schema.getTypeFromString(FLIGHT_FIELD_COMPONENT_ARGUMENT_TYPE),
130
+ value: {
131
+ kind: 'Literal',
132
+ value: componentName,
133
+ loc: field.loc
134
+ }
135
+ }, {
136
+ kind: 'Argument',
137
+ loc: field.loc,
138
+ name: FLIGHT_FIELD_PROPS_ARGUMENT_NAME,
139
+ type: state.types.propsType,
140
+ value: {
141
+ kind: 'ObjectValue',
142
+ fields: field.args.map(function (arg) {
143
+ return {
144
+ kind: 'ObjectFieldValue',
145
+ loc: arg.loc,
146
+ name: arg.name,
147
+ value: arg.value
148
+ };
149
+ }),
150
+ loc: field.loc
151
+ }
152
+ }],
153
+ metadata: _objectSpread(_objectSpread({}, field.metadata || {}), {}, {
154
+ flight: true
155
+ }),
156
+ type: state.types.componentType
157
+ });
158
+ }
159
+
160
+ module.exports = {
161
+ transform: reactFlightComponentTransform
162
+ };
@@ -64,8 +64,8 @@ function refetchableFragmentTransform(context) {
64
64
  transformedFragment = _buildRefetchOperatio.transformedFragment;
65
65
 
66
66
  var connectionMetadata = extractConnectionMetadata(context.getSchema(), transformedFragment);
67
- nextContext = nextContext.replace(_objectSpread({}, transformedFragment, {
68
- metadata: _objectSpread({}, transformedFragment.metadata || {}, {
67
+ nextContext = nextContext.replace(_objectSpread(_objectSpread({}, transformedFragment), {}, {
68
+ metadata: _objectSpread(_objectSpread({}, transformedFragment.metadata || {}), {}, {
69
69
  refetch: {
70
70
  connection: connectionMetadata !== null && connectionMetadata !== void 0 ? connectionMetadata : null,
71
71
  operation: refetchName,
@@ -74,8 +74,8 @@ function refetchableFragmentTransform(context) {
74
74
  }
75
75
  })
76
76
  }));
77
- nextContext = nextContext.add(_objectSpread({}, node, {
78
- metadata: _objectSpread({}, node.metadata || {}, {
77
+ nextContext = nextContext.add(_objectSpread(_objectSpread({}, node), {}, {
78
+ metadata: _objectSpread(_objectSpread({}, node.metadata || {}), {}, {
79
79
  derivedFrom: transformedFragment.name,
80
80
  isRefetchableQuery: true
81
81
  })
@@ -51,12 +51,12 @@ function visitRelayMetadata(metadataFn) {
51
51
 
52
52
  var argValues = getLiteralArgumentValues(relayDirective.args);
53
53
  var metadata = metadataFn(argValues);
54
- return this.traverse(_objectSpread({}, node, {
54
+ return this.traverse(_objectSpread(_objectSpread({}, node), {}, {
55
55
  directives: node.directives.filter(function (directive) {
56
56
  return directive !== relayDirective;
57
57
  }),
58
58
  // $FlowFixMe[cannot-spread-indexer]
59
- metadata: _objectSpread({}, node.metadata || {}, {}, metadata)
59
+ metadata: _objectSpread(_objectSpread({}, node.metadata || {}), metadata)
60
60
  }));
61
61
  };
62
62
  }
@@ -0,0 +1,380 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ *
7
+ *
8
+ * @format
9
+ */
10
+ // flowlint ambiguous-object-type:error
11
+ 'use strict';
12
+
13
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
14
+
15
+ var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
16
+
17
+ var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray"));
18
+
19
+ function _createForOfIteratorHelper(o, allowArrayLike) { var it; if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = o[Symbol.iterator](); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it["return"] != null) it["return"](); } finally { if (didErr) throw err; } } }; }
20
+
21
+ function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
22
+
23
+ function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
24
+
25
+ function 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; }
26
+
27
+ function _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) { (0, _defineProperty2["default"])(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; }
28
+
29
+ var IRTransformer = require('../core/IRTransformer');
30
+
31
+ var partitionArray = require('../util/partitionArray');
32
+
33
+ var _require = require('../core/CompilerError'),
34
+ createUserError = _require.createUserError,
35
+ createCompilerError = _require.createCompilerError;
36
+
37
+ var _require2 = require('relay-runtime'),
38
+ RelayFeatureFlags = _require2.RelayFeatureFlags;
39
+
40
+ var SCHEMA_EXTENSION = "\n enum RequiredFieldAction {\n NONE\n LOG\n THROW\n }\n directive @required(\n action: RequiredFieldAction!\n ) on FIELD\n";
41
+ /**
42
+ * This transform rewrites ScalarField and LinkedField nodes with a @required
43
+ * directive into fields with the directives stripped and sets the `required`
44
+ * and `path` metadata values.
45
+ */
46
+
47
+ function requiredFieldTransform(context) {
48
+ var schema = context.getSchema();
49
+ return IRTransformer.transform(context, {
50
+ LinkedField: visitLinkedField,
51
+ ScalarField: vistitScalarField,
52
+ InlineFragment: visitInlineFragment,
53
+ Fragment: visitFragment,
54
+ Root: visitRoot
55
+ }, function (node) {
56
+ return {
57
+ schema: schema,
58
+ documentName: node.name,
59
+ path: [],
60
+ pathRequiredMap: new Map(),
61
+ currentNodeRequiredChildren: new Map(),
62
+ requiredChildrenMap: new Map(),
63
+ parentAbstractInlineFragment: null
64
+ };
65
+ });
66
+ }
67
+
68
+ function visitFragment(fragment, state) {
69
+ return addChildrenCanBubbleMetadata(this.traverse(fragment, state), state);
70
+ }
71
+
72
+ function visitRoot(root, state) {
73
+ return addChildrenCanBubbleMetadata(this.traverse(root, state), state);
74
+ }
75
+
76
+ function visitInlineFragment(fragment, state) {
77
+ var _state$parentAbstract;
78
+
79
+ // Ideally we could allow @required when the direct parent inline fragment was
80
+ // on a concrete type, but we would need to solve this bug in our Flow type
81
+ // generation first: T65695438
82
+ var parentAbstractInlineFragment = (_state$parentAbstract = state.parentAbstractInlineFragment) !== null && _state$parentAbstract !== void 0 ? _state$parentAbstract : getAbstractInlineFragment(fragment, state.schema);
83
+ return this.traverse(fragment, _objectSpread(_objectSpread({}, state), {}, {
84
+ parentAbstractInlineFragment: parentAbstractInlineFragment
85
+ }));
86
+ }
87
+
88
+ function getAbstractInlineFragment(fragment, schema) {
89
+ var typeCondition = fragment.typeCondition;
90
+
91
+ if (schema.isAbstractType(typeCondition)) {
92
+ return fragment;
93
+ }
94
+
95
+ return null;
96
+ } // Convert action to a number so that we can numerically compare their severity.
97
+
98
+
99
+ function getActionSeverity(action) {
100
+ switch (action) {
101
+ case 'NONE':
102
+ return 0;
103
+
104
+ case 'LOG':
105
+ return 1;
106
+
107
+ case 'THROW':
108
+ return 2;
109
+
110
+ default:
111
+ action;
112
+ throw createCompilerError("Unhandled action type ".concat(action));
113
+ }
114
+ }
115
+
116
+ function visitLinkedField(field, state) {
117
+ var path = [].concat((0, _toConsumableArray2["default"])(state.path), [field.alias]);
118
+
119
+ var newState = _objectSpread(_objectSpread({}, state), {}, {
120
+ currentNodeRequiredChildren: new Map(),
121
+ path: path,
122
+ parentAbstractInlineFragment: null
123
+ });
124
+
125
+ var newField = this.traverse(field, newState);
126
+ var pathName = path.join('.');
127
+ assertCompatibleRequiredChildren(field, pathName, newState);
128
+ newField = applyDirectives(newField, pathName, state.documentName);
129
+ assertCompatibleNullability(newField, pathName, newState.pathRequiredMap);
130
+ var directiveMetadata = getRequiredDirectiveMetadata(newField);
131
+
132
+ if (directiveMetadata != null) {
133
+ assertParentIsNotInvalidInlineFragmet(state.schema, directiveMetadata, state.parentAbstractInlineFragment);
134
+ state.currentNodeRequiredChildren.set(field.alias, newField);
135
+ var severity = getActionSeverity(directiveMetadata.action); // Assert that all @required children have at least this severity.
136
+
137
+ newState.currentNodeRequiredChildren.forEach(function (childField) {
138
+ var childMetadata = getRequiredDirectiveMetadata(childField);
139
+
140
+ if (childMetadata == null) {
141
+ return;
142
+ }
143
+
144
+ if (getActionSeverity(childMetadata.action) < severity) {
145
+ throw createUserError("The @required field [1] may not have an `action` less severe than that of its @required parent [2]. [1] should probably be `action: ".concat(directiveMetadata.action, "`."), [childMetadata.actionLoc, directiveMetadata.actionLoc]);
146
+ }
147
+ });
148
+ }
149
+
150
+ state.requiredChildrenMap.set(pathName, newState.currentNodeRequiredChildren);
151
+ return addChildrenCanBubbleMetadata(newField, newState);
152
+ }
153
+
154
+ function vistitScalarField(field, state) {
155
+ var pathName = [].concat((0, _toConsumableArray2["default"])(state.path), [field.alias]).join('.');
156
+ var newField = applyDirectives(field, pathName, state.documentName);
157
+ var directiveMetadata = getRequiredDirectiveMetadata(newField);
158
+
159
+ if (directiveMetadata != null) {
160
+ assertParentIsNotInvalidInlineFragmet(state.schema, directiveMetadata, state.parentAbstractInlineFragment);
161
+ state.currentNodeRequiredChildren.set(field.alias, newField);
162
+ }
163
+
164
+ assertCompatibleNullability(newField, pathName, state.pathRequiredMap);
165
+ return newField;
166
+ }
167
+
168
+ function addChildrenCanBubbleMetadata(node, state) {
169
+ var _iterator = _createForOfIteratorHelper(state.currentNodeRequiredChildren.values()),
170
+ _step;
171
+
172
+ try {
173
+ for (_iterator.s(); !(_step = _iterator.n()).done;) {
174
+ var child = _step.value;
175
+ var requiredMetadata = getRequiredDirectiveMetadata(child);
176
+
177
+ if (requiredMetadata != null && requiredMetadata.action !== 'THROW') {
178
+ var metadata = _objectSpread(_objectSpread({}, node.metadata), {}, {
179
+ childrenCanBubbleNull: true
180
+ });
181
+
182
+ return _objectSpread(_objectSpread({}, node), {}, {
183
+ metadata: metadata
184
+ });
185
+ }
186
+ }
187
+ } catch (err) {
188
+ _iterator.e(err);
189
+ } finally {
190
+ _iterator.f();
191
+ }
192
+
193
+ return node;
194
+ }
195
+
196
+ function assertParentIsNotInvalidInlineFragmet(schema, directiveMetadata, parentAbstractInlineFragment) {
197
+ if (parentAbstractInlineFragment == null) {
198
+ return;
199
+ }
200
+
201
+ var typeCondition = parentAbstractInlineFragment.typeCondition;
202
+
203
+ if (schema.isUnion(typeCondition)) {
204
+ throw createUserError('The @required directive [1] may not be used anywhere within an inline fragment on a union type [2].', [directiveMetadata.directiveLoc, parentAbstractInlineFragment.loc]);
205
+ } else if (schema.isInterface(typeCondition)) {
206
+ throw createUserError('The @required directive [1] may not be used anywhere within an inline fragment on an interface type [2].', [directiveMetadata.directiveLoc, parentAbstractInlineFragment.loc]);
207
+ } else {
208
+ throw createCompilerError('Unexpected abstract inline fragment type.', [parentAbstractInlineFragment.loc]);
209
+ }
210
+ } // Check that this field's nullability matches all other instances.
211
+
212
+
213
+ function assertCompatibleNullability(field, pathName, pathRequiredMap) {
214
+ var existingField = pathRequiredMap.get(pathName);
215
+
216
+ if (existingField == null) {
217
+ pathRequiredMap.set(pathName, field);
218
+ return;
219
+ }
220
+
221
+ var requiredMetadata = getRequiredDirectiveMetadata(field);
222
+ var existingRequiredMetadata = getRequiredDirectiveMetadata(existingField);
223
+
224
+ if ((requiredMetadata === null || requiredMetadata === void 0 ? void 0 : requiredMetadata.action) === (existingRequiredMetadata === null || existingRequiredMetadata === void 0 ? void 0 : existingRequiredMetadata.action)) {
225
+ return;
226
+ }
227
+
228
+ if (requiredMetadata == null) {
229
+ throw createUserError("The field \"".concat(field.alias, "\" is @required in [1] but not in [2]."), [existingField.loc, field.loc]);
230
+ }
231
+
232
+ if (existingRequiredMetadata == null) {
233
+ throw createUserError("The field \"".concat(field.alias, "\" is @required in [1] but not in [2]."), [field.loc, existingField.loc]);
234
+ }
235
+
236
+ throw createUserError("The field \"".concat(field.alias, "\" has a different @required action in [1] than in [2]."), [requiredMetadata.actionLoc, existingRequiredMetadata.actionLoc]);
237
+ } // Metadata is untyped, so we use this utility function to do the type coersion.
238
+
239
+
240
+ function getRequiredDirectiveMetadata(field) {
241
+ var _field$metadata;
242
+
243
+ return (_field$metadata = field.metadata) === null || _field$metadata === void 0 ? void 0 : _field$metadata.required;
244
+ } // Check that this field has the same required children as all other instances.
245
+
246
+
247
+ function assertCompatibleRequiredChildren(field, fieldPath, _ref) {
248
+ var currentNodeRequiredChildren = _ref.currentNodeRequiredChildren,
249
+ pathRequiredMap = _ref.pathRequiredMap,
250
+ requiredChildrenMap = _ref.requiredChildrenMap;
251
+ var previouslyRequiredChildren = requiredChildrenMap.get(fieldPath);
252
+
253
+ if (previouslyRequiredChildren == null) {
254
+ return;
255
+ } // Check if this field has a required child field which was previously omitted.
256
+
257
+
258
+ var _iterator2 = _createForOfIteratorHelper(currentNodeRequiredChildren),
259
+ _step2;
260
+
261
+ try {
262
+ for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
263
+ var _step2$value = _step2.value,
264
+ path = _step2$value[0],
265
+ childField = _step2$value[1];
266
+
267
+ if (!previouslyRequiredChildren.has(path)) {
268
+ var otherParent = pathRequiredMap.get(fieldPath);
269
+
270
+ if (otherParent == null) {
271
+ throw createCompilerError("Could not find other parent node at path \"".concat(fieldPath, "\"."), [childField.loc]);
272
+ }
273
+
274
+ throw createMissingRequiredFieldError(childField, otherParent);
275
+ }
276
+ } // Check if a previous reference to this field had a required child field which we are missing.
277
+
278
+ } catch (err) {
279
+ _iterator2.e(err);
280
+ } finally {
281
+ _iterator2.f();
282
+ }
283
+
284
+ var _iterator3 = _createForOfIteratorHelper(previouslyRequiredChildren),
285
+ _step3;
286
+
287
+ try {
288
+ for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) {
289
+ var _step3$value = _step3.value,
290
+ _path = _step3$value[0],
291
+ _childField = _step3$value[1];
292
+
293
+ if (!currentNodeRequiredChildren.has(_path)) {
294
+ throw createMissingRequiredFieldError(_childField, field);
295
+ }
296
+ }
297
+ } catch (err) {
298
+ _iterator3.e(err);
299
+ } finally {
300
+ _iterator3.f();
301
+ }
302
+ }
303
+
304
+ function createMissingRequiredFieldError(requiredChild, missingParent) {
305
+ var alias = requiredChild.alias;
306
+ return createUserError("The field \"".concat(alias, "\" is marked as @required in [1] but is missing in [2]."), [requiredChild.loc, missingParent.loc]);
307
+ } // TODO T74397896: Remove prefix gating once @required is rolled out more broadly.
308
+
309
+
310
+ function featureIsEnabled(documentName) {
311
+ var featureFlag = RelayFeatureFlags.ENABLE_REQUIRED_DIRECTIVES;
312
+
313
+ if (typeof featureFlag === 'boolean') {
314
+ return featureFlag;
315
+ } else if (featureFlag === 'LIMITED') {
316
+ return documentName.startsWith('RelayRequiredTest');
317
+ } else if (typeof featureFlag === 'string') {
318
+ return featureFlag.split('|').some(function (prefix) {
319
+ return documentName.startsWith(prefix);
320
+ });
321
+ }
322
+
323
+ return false;
324
+ } // Strip and validate @required directives, and convert them to metadata.
325
+
326
+
327
+ function applyDirectives(field, pathName, documentName) {
328
+ var _partitionArray = partitionArray(field.directives, function (directive) {
329
+ return directive.name === 'required';
330
+ }),
331
+ requiredDirectives = _partitionArray[0],
332
+ otherDirectives = _partitionArray[1];
333
+
334
+ if (requiredDirectives.length === 0) {
335
+ return field;
336
+ }
337
+
338
+ if (!featureIsEnabled(documentName)) {
339
+ throw new createUserError( // Purposefully don't include details in this error message, since we
340
+ // don't want folks adopting this feature until it's been tested more.
341
+ 'The @required directive is experimental and not yet supported for use in product code', requiredDirectives.map(function (x) {
342
+ return x.loc;
343
+ }));
344
+ }
345
+
346
+ if (requiredDirectives.length > 1) {
347
+ throw new createUserError('Did not expect multiple @required directives.', requiredDirectives.map(function (x) {
348
+ return x.loc;
349
+ }));
350
+ }
351
+
352
+ var requiredDirective = requiredDirectives[0];
353
+ var arg = requiredDirective.args[0]; // I would expect this check to be handled by the schema validation, but...
354
+
355
+ if (arg == null) {
356
+ throw createUserError('The @required directive requires an `action` argument.', [requiredDirective.loc]);
357
+ }
358
+
359
+ if (arg.value.kind !== 'Literal') {
360
+ throw createUserError('Expected @required `action` argument to be a literal.', [arg.value.loc]);
361
+ }
362
+
363
+ return _objectSpread(_objectSpread({}, field), {}, {
364
+ directives: otherDirectives,
365
+ metadata: _objectSpread(_objectSpread({}, field.metadata), {}, {
366
+ required: {
367
+ action: arg.value.value,
368
+ actionLoc: arg.loc,
369
+ directiveLoc: requiredDirective.loc,
370
+ path: pathName
371
+ }
372
+ })
373
+ });
374
+ } // Transform @required directive to metadata
375
+
376
+
377
+ module.exports = {
378
+ SCHEMA_EXTENSION: SCHEMA_EXTENSION,
379
+ transform: requiredFieldTransform
380
+ };
@@ -35,7 +35,7 @@ function visitField(field) {
35
35
  var transformedNode = this.traverse(field);
36
36
 
37
37
  if (transformedNode.handles) {
38
- return _objectSpread({}, transformedNode, {
38
+ return _objectSpread(_objectSpread({}, transformedNode), {}, {
39
39
  handles: null
40
40
  });
41
41
  }
@@ -159,6 +159,7 @@ function transformNode(schema, node, selectionMap) {
159
159
  var result;
160
160
 
161
161
  if (isEmptySelectionMap) {
162
+ // $FlowFixMe[escaped-generic]
162
163
  result = cache.get(node);
163
164
 
164
165
  if (result != null) {
@@ -219,7 +220,7 @@ function transformNode(schema, node, selectionMap) {
219
220
  !false ? process.env.NODE_ENV !== "production" ? invariant(false, 'SkipRedundantNodesTransform: Unexpected node kind `%s`.', selection.kind) : invariant(false) : void 0;
220
221
  }
221
222
  });
222
- var nextNode = selections.length ? _objectSpread({}, node, {
223
+ var nextNode = selections.length ? _objectSpread(_objectSpread({}, node), {}, {
223
224
  selections: selections
224
225
  }) : null;
225
226
  result = {
@@ -228,6 +229,7 @@ function transformNode(schema, node, selectionMap) {
228
229
  };
229
230
 
230
231
  if (isEmptySelectionMap) {
232
+ // $FlowFixMe[escaped-generic]
231
233
  cache.set(node, result);
232
234
  }
233
235
 
@@ -136,7 +136,7 @@ function transformNode(context, fragments, node) {
136
136
  }
137
137
 
138
138
  if (selections) {
139
- return _objectSpread({}, node, {
139
+ return _objectSpread(_objectSpread({}, node), {}, {
140
140
  selections: selections
141
141
  });
142
142
  }
@@ -18,9 +18,9 @@ function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (O
18
18
 
19
19
  function _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) { (0, _defineProperty2["default"])(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; }
20
20
 
21
- function _createForOfIteratorHelper(o) { if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (o = _unsupportedIterableToArray(o))) { var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var it, normalCompletion = true, didErr = false, err; return { s: function s() { it = o[Symbol.iterator](); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it["return"] != null) it["return"](); } finally { if (didErr) throw err; } } }; }
21
+ function _createForOfIteratorHelper(o, allowArrayLike) { var it; if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = o[Symbol.iterator](); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it["return"] != null) it["return"](); } finally { if (didErr) throw err; } } }; }
22
22
 
23
- function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(n); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
23
+ function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
24
24
 
25
25
  function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
26
26
 
@@ -56,7 +56,7 @@ function skipUnusedVariablesTransform(context) {
56
56
  });
57
57
 
58
58
  if (usedArgumentDefinitions.length !== node.argumentDefinitions.length) {
59
- nextContext = nextContext.replace(_objectSpread({}, node, {
59
+ nextContext = nextContext.replace(_objectSpread(_objectSpread({}, node), {}, {
60
60
  argumentDefinitions: usedArgumentDefinitions
61
61
  }));
62
62
  }
@@ -132,11 +132,11 @@ function visitRoot(node) {
132
132
  keys.forEach(function (key) {
133
133
  sortedSelectionsTypeInfo[key] = selectionsTypeInfo[key];
134
134
  });
135
- return _objectSpread({}, node, {
135
+ return _objectSpread(_objectSpread({}, node), {}, {
136
136
  directives: node.directives.filter(function (directive) {
137
137
  return directive !== testDirective;
138
138
  }),
139
- metadata: _objectSpread({}, node.metadata || {}, {
139
+ metadata: _objectSpread(_objectSpread({}, node.metadata || {}), {}, {
140
140
  relayTestingSelectionTypeInfo: sortedSelectionsTypeInfo
141
141
  })
142
142
  });