relay-compiler 0.0.0-main-4d287de9 → 0.0.0-main-9a79039d
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/cli.js +23 -0
- package/index.js +19 -3
- package/linux-x64/relay +0 -0
- package/macos-x64/relay +0 -0
- package/package.json +2 -26
- package/win-x64/relay.exe +0 -0
- package/bin/RelayCompilerBin.js.flow +0 -168
- package/bin/RelayCompilerMain.js.flow +0 -513
- package/bin/__fixtures__/plugin-module.js.flow +0 -17
- package/bin/relay-compiler +0 -19066
- package/codegen/CodegenDirectory.js.flow +0 -372
- package/codegen/CodegenRunner.js.flow +0 -424
- package/codegen/CodegenTypes.js.flow +0 -28
- package/codegen/CodegenWatcher.js.flow +0 -250
- package/codegen/NormalizationCodeGenerator.js.flow +0 -565
- package/codegen/ReaderCodeGenerator.js.flow +0 -510
- package/codegen/RelayCodeGenerator.js.flow +0 -85
- package/codegen/RelayFileWriter.js.flow +0 -365
- package/codegen/SourceControl.js.flow +0 -58
- package/codegen/compileRelayArtifacts.js.flow +0 -181
- package/codegen/createPrintRequireModuleDependency.js.flow +0 -19
- package/codegen/sortObjectByKey.js.flow +0 -25
- package/codegen/writeRelayGeneratedFile.js.flow +0 -235
- package/core/ASTCache.js.flow +0 -73
- package/core/ASTConvert.js.flow +0 -230
- package/core/CompilerContext.js.flow +0 -189
- package/core/CompilerError.js.flow +0 -255
- package/core/DotGraphQLParser.js.flow +0 -37
- package/core/GraphQLCompilerProfiler.js.flow +0 -341
- package/core/GraphQLDerivedFromMetadata.js.flow +0 -36
- package/core/GraphQLWatchmanClient.js.flow +0 -109
- package/core/IR.js.flow +0 -326
- package/core/IRPrinter.js.flow +0 -472
- package/core/IRTransformer.js.flow +0 -376
- package/core/IRValidator.js.flow +0 -259
- package/core/IRVisitor.js.flow +0 -150
- package/core/JSModuleParser.js.flow +0 -24
- package/core/RelayCompilerScope.js.flow +0 -199
- package/core/RelayFindGraphQLTags.js.flow +0 -118
- package/core/RelayGraphQLEnumsGenerator.js.flow +0 -55
- package/core/RelayIRTransforms.js.flow +0 -138
- package/core/RelayParser.js.flow +0 -1731
- package/core/RelaySourceModuleParser.js.flow +0 -133
- package/core/Schema.js.flow +0 -2035
- package/core/SchemaUtils.js.flow +0 -120
- package/core/filterContextForNode.js.flow +0 -49
- package/core/getFieldDefinition.js.flow +0 -156
- package/core/getIdentifierForArgumentValue.js.flow +0 -49
- package/core/getIdentifierForSelection.js.flow +0 -68
- package/core/getLiteralArgumentValues.js.flow +0 -32
- package/core/getNormalizationOperationName.js.flow +0 -19
- package/core/inferRootArgumentDefinitions.js.flow +0 -322
- package/index.js.flow +0 -198
- package/language/RelayLanguagePluginInterface.js.flow +0 -283
- package/language/javascript/FindGraphQLTags.js.flow +0 -136
- package/language/javascript/RelayFlowBabelFactories.js.flow +0 -176
- package/language/javascript/RelayFlowGenerator.js.flow +0 -1096
- package/language/javascript/RelayFlowTypeTransformers.js.flow +0 -181
- package/language/javascript/RelayLanguagePluginJavaScript.js.flow +0 -33
- package/language/javascript/formatGeneratedModule.js.flow +0 -65
- package/lib/bin/RelayCompilerBin.js +0 -143
- package/lib/bin/RelayCompilerMain.js +0 -486
- package/lib/bin/__fixtures__/plugin-module.js +0 -16
- package/lib/codegen/CodegenDirectory.js +0 -335
- package/lib/codegen/CodegenRunner.js +0 -433
- package/lib/codegen/CodegenTypes.js +0 -11
- package/lib/codegen/CodegenWatcher.js +0 -271
- package/lib/codegen/NormalizationCodeGenerator.js +0 -487
- package/lib/codegen/ReaderCodeGenerator.js +0 -473
- package/lib/codegen/RelayCodeGenerator.js +0 -75
- package/lib/codegen/RelayFileWriter.js +0 -270
- package/lib/codegen/SourceControl.js +0 -60
- package/lib/codegen/compileRelayArtifacts.js +0 -157
- package/lib/codegen/createPrintRequireModuleDependency.js +0 -19
- package/lib/codegen/sortObjectByKey.js +0 -41
- package/lib/codegen/writeRelayGeneratedFile.js +0 -206
- package/lib/core/ASTCache.js +0 -70
- package/lib/core/ASTConvert.js +0 -198
- package/lib/core/CompilerContext.js +0 -165
- package/lib/core/CompilerError.js +0 -252
- package/lib/core/DotGraphQLParser.js +0 -40
- package/lib/core/GraphQLCompilerProfiler.js +0 -299
- package/lib/core/GraphQLDerivedFromMetadata.js +0 -31
- package/lib/core/GraphQLWatchmanClient.js +0 -150
- package/lib/core/IR.js +0 -11
- package/lib/core/IRPrinter.js +0 -388
- package/lib/core/IRTransformer.js +0 -345
- package/lib/core/IRValidator.js +0 -226
- package/lib/core/IRVisitor.js +0 -45
- package/lib/core/JSModuleParser.js +0 -18
- package/lib/core/RelayCompilerScope.js +0 -183
- package/lib/core/RelayFindGraphQLTags.js +0 -79
- package/lib/core/RelayGraphQLEnumsGenerator.js +0 -50
- package/lib/core/RelayIRTransforms.js +0 -109
- package/lib/core/RelayParser.js +0 -1381
- package/lib/core/RelaySourceModuleParser.js +0 -104
- package/lib/core/Schema.js +0 -1877
- package/lib/core/SchemaUtils.js +0 -98
- package/lib/core/filterContextForNode.js +0 -50
- package/lib/core/getFieldDefinition.js +0 -145
- package/lib/core/getIdentifierForArgumentValue.js +0 -54
- package/lib/core/getIdentifierForSelection.js +0 -49
- package/lib/core/getLiteralArgumentValues.js +0 -26
- package/lib/core/getNormalizationOperationName.js +0 -17
- package/lib/core/inferRootArgumentDefinitions.js +0 -351
- package/lib/index.js +0 -178
- package/lib/language/RelayLanguagePluginInterface.js +0 -26
- package/lib/language/javascript/FindGraphQLTags.js +0 -126
- package/lib/language/javascript/RelayFlowBabelFactories.js +0 -160
- package/lib/language/javascript/RelayFlowGenerator.js +0 -856
- package/lib/language/javascript/RelayFlowTypeTransformers.js +0 -119
- package/lib/language/javascript/RelayLanguagePluginJavaScript.js +0 -30
- package/lib/language/javascript/formatGeneratedModule.js +0 -36
- package/lib/reporters/ConsoleReporter.js +0 -61
- package/lib/reporters/MultiReporter.js +0 -45
- package/lib/reporters/Reporter.js +0 -11
- package/lib/runner/Artifacts.js +0 -323
- package/lib/runner/BufferedFilesystem.js +0 -262
- package/lib/runner/GraphQLASTNodeGroup.js +0 -256
- package/lib/runner/GraphQLASTUtils.js +0 -23
- package/lib/runner/GraphQLNodeMap.js +0 -81
- package/lib/runner/Sources.js +0 -271
- package/lib/runner/StrictMap.js +0 -134
- package/lib/runner/compileArtifacts.js +0 -39
- package/lib/runner/extractAST.js +0 -77
- package/lib/runner/getChangedNodeNames.js +0 -82
- package/lib/runner/getSchemaInstance.js +0 -30
- package/lib/runner/types.js +0 -12
- package/lib/transforms/ApplyFragmentArgumentTransform.js +0 -393
- package/lib/transforms/ClientExtensionsTransform.js +0 -221
- package/lib/transforms/ConnectionTransform.js +0 -639
- package/lib/transforms/DeclarativeConnectionMutationTransform.js +0 -218
- package/lib/transforms/DeferStreamTransform.js +0 -246
- package/lib/transforms/DisallowIdAsAlias.js +0 -40
- package/lib/transforms/DisallowTypenameOnRoot.js +0 -53
- package/lib/transforms/FieldHandleTransform.js +0 -79
- package/lib/transforms/FilterCompilerDirectivesTransform.js +0 -29
- package/lib/transforms/FilterDirectivesTransform.js +0 -42
- package/lib/transforms/FlattenTransform.js +0 -306
- package/lib/transforms/GenerateIDFieldTransform.js +0 -135
- package/lib/transforms/GenerateTypeNameTransform.js +0 -149
- package/lib/transforms/InlineDataFragmentTransform.js +0 -100
- package/lib/transforms/InlineFragmentsTransform.js +0 -61
- package/lib/transforms/MaskTransform.js +0 -117
- package/lib/transforms/MatchTransform.js +0 -434
- package/lib/transforms/ReactFlightComponentTransform.js +0 -158
- package/lib/transforms/RefetchableFragmentTransform.js +0 -249
- package/lib/transforms/RelayDirectiveTransform.js +0 -83
- package/lib/transforms/RequiredFieldTransform.js +0 -369
- package/lib/transforms/SkipClientExtensionsTransform.js +0 -46
- package/lib/transforms/SkipHandleFieldTransform.js +0 -45
- package/lib/transforms/SkipRedundantNodesTransform.js +0 -261
- package/lib/transforms/SkipSplitOperationTransform.js +0 -32
- package/lib/transforms/SkipUnreachableNodeTransform.js +0 -158
- package/lib/transforms/SkipUnusedVariablesTransform.js +0 -75
- package/lib/transforms/SplitModuleImportTransform.js +0 -82
- package/lib/transforms/TestOperationTransform.js +0 -144
- package/lib/transforms/TransformUtils.js +0 -21
- package/lib/transforms/ValidateGlobalVariablesTransform.js +0 -92
- package/lib/transforms/ValidateRequiredArgumentsTransform.js +0 -114
- package/lib/transforms/ValidateServerOnlyDirectivesTransform.js +0 -108
- package/lib/transforms/ValidateUnusedVariablesTransform.js +0 -96
- package/lib/transforms/query-generators/FetchableQueryGenerator.js +0 -157
- package/lib/transforms/query-generators/NodeQueryGenerator.js +0 -166
- package/lib/transforms/query-generators/QueryQueryGenerator.js +0 -48
- package/lib/transforms/query-generators/ViewerQueryGenerator.js +0 -77
- package/lib/transforms/query-generators/index.js +0 -60
- package/lib/transforms/query-generators/utils.js +0 -92
- package/lib/util/CodeMarker.js +0 -80
- package/lib/util/DefaultHandleKey.js +0 -15
- package/lib/util/RelayCompilerCache.js +0 -97
- package/lib/util/Rollout.js +0 -40
- package/lib/util/TimeReporter.js +0 -83
- package/lib/util/areEqualArgValues.js +0 -135
- package/lib/util/argumentContainsVariables.js +0 -37
- package/lib/util/dedupeJSONStringify.js +0 -160
- package/lib/util/generateAbstractTypeRefinementKey.js +0 -24
- package/lib/util/getDefinitionNodeHash.js +0 -22
- package/lib/util/getModuleName.js +0 -32
- package/lib/util/joinArgumentDefinitions.js +0 -67
- package/lib/util/md5.js +0 -17
- package/lib/util/murmurHash.js +0 -86
- package/lib/util/nullthrowsOSS.js +0 -23
- package/lib/util/orList.js +0 -36
- package/lib/util/partitionArray.js +0 -35
- package/relay-compiler.js +0 -17
- package/relay-compiler.min.js +0 -22
- package/reporters/ConsoleReporter.js.flow +0 -81
- package/reporters/MultiReporter.js.flow +0 -43
- package/reporters/Reporter.js.flow +0 -19
- package/runner/Artifacts.js.flow +0 -215
- package/runner/BufferedFilesystem.js.flow +0 -194
- package/runner/GraphQLASTNodeGroup.js.flow +0 -174
- package/runner/GraphQLASTUtils.js.flow +0 -26
- package/runner/GraphQLNodeMap.js.flow +0 -55
- package/runner/Sources.js.flow +0 -227
- package/runner/StrictMap.js.flow +0 -96
- package/runner/compileArtifacts.js.flow +0 -75
- package/runner/extractAST.js.flow +0 -98
- package/runner/getChangedNodeNames.js.flow +0 -48
- package/runner/getSchemaInstance.js.flow +0 -36
- package/runner/types.js.flow +0 -37
- package/transforms/ApplyFragmentArgumentTransform.js.flow +0 -524
- package/transforms/ClientExtensionsTransform.js.flow +0 -224
- package/transforms/ConnectionTransform.js.flow +0 -850
- package/transforms/DeclarativeConnectionMutationTransform.js.flow +0 -245
- package/transforms/DeferStreamTransform.js.flow +0 -263
- package/transforms/DisallowIdAsAlias.js.flow +0 -46
- package/transforms/DisallowTypenameOnRoot.js.flow +0 -44
- package/transforms/FieldHandleTransform.js.flow +0 -77
- package/transforms/FilterCompilerDirectivesTransform.js.flow +0 -33
- package/transforms/FilterDirectivesTransform.js.flow +0 -45
- package/transforms/FlattenTransform.js.flow +0 -458
- package/transforms/GenerateIDFieldTransform.js.flow +0 -151
- package/transforms/GenerateTypeNameTransform.js.flow +0 -159
- package/transforms/InlineDataFragmentTransform.js.flow +0 -123
- package/transforms/InlineFragmentsTransform.js.flow +0 -70
- package/transforms/MaskTransform.js.flow +0 -124
- package/transforms/MatchTransform.js.flow +0 -587
- package/transforms/ReactFlightComponentTransform.js.flow +0 -194
- package/transforms/RefetchableFragmentTransform.js.flow +0 -266
- package/transforms/RelayDirectiveTransform.js.flow +0 -96
- package/transforms/RequiredFieldTransform.js.flow +0 -413
- package/transforms/SkipClientExtensionsTransform.js.flow +0 -54
- package/transforms/SkipHandleFieldTransform.js.flow +0 -44
- package/transforms/SkipRedundantNodesTransform.js.flow +0 -255
- package/transforms/SkipSplitOperationTransform.js.flow +0 -37
- package/transforms/SkipUnreachableNodeTransform.js.flow +0 -148
- package/transforms/SkipUnusedVariablesTransform.js.flow +0 -59
- package/transforms/SplitModuleImportTransform.js.flow +0 -97
- package/transforms/TestOperationTransform.js.flow +0 -142
- package/transforms/TransformUtils.js.flow +0 -26
- package/transforms/ValidateGlobalVariablesTransform.js.flow +0 -80
- package/transforms/ValidateRequiredArgumentsTransform.js.flow +0 -126
- package/transforms/ValidateServerOnlyDirectivesTransform.js.flow +0 -111
- package/transforms/ValidateUnusedVariablesTransform.js.flow +0 -88
- package/transforms/query-generators/FetchableQueryGenerator.js.flow +0 -188
- package/transforms/query-generators/NodeQueryGenerator.js.flow +0 -217
- package/transforms/query-generators/QueryQueryGenerator.js.flow +0 -57
- package/transforms/query-generators/ViewerQueryGenerator.js.flow +0 -97
- package/transforms/query-generators/index.js.flow +0 -89
- package/transforms/query-generators/utils.js.flow +0 -76
- package/util/CodeMarker.js.flow +0 -79
- package/util/DefaultHandleKey.js.flow +0 -17
- package/util/RelayCompilerCache.js.flow +0 -86
- package/util/Rollout.js.flow +0 -39
- package/util/TimeReporter.js.flow +0 -79
- package/util/areEqualArgValues.js.flow +0 -126
- package/util/argumentContainsVariables.js.flow +0 -38
- package/util/dedupeJSONStringify.js.flow +0 -152
- package/util/generateAbstractTypeRefinementKey.js.flow +0 -29
- package/util/getDefinitionNodeHash.js.flow +0 -24
- package/util/getModuleName.js.flow +0 -39
- package/util/joinArgumentDefinitions.js.flow +0 -105
- package/util/md5.js.flow +0 -19
- package/util/murmurHash.js.flow +0 -94
- package/util/nullthrowsOSS.js.flow +0 -25
- package/util/orList.js.flow +0 -37
- package/util/partitionArray.js.flow +0 -37
@@ -1,255 +0,0 @@
|
|
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
|
-
* @flow
|
8
|
-
* @format
|
9
|
-
*/
|
10
|
-
|
11
|
-
// flowlint ambiguous-object-type:error
|
12
|
-
|
13
|
-
'use strict';
|
14
|
-
|
15
|
-
import type CompilerContext from '../core/CompilerContext';
|
16
|
-
import type {Fragment, Node, Root, Selection, SplitOperation} from '../core/IR';
|
17
|
-
import type {Schema} from '../core/Schema';
|
18
|
-
|
19
|
-
const getIdentifierForSelection = require('../core/getIdentifierForSelection');
|
20
|
-
const IRTransformer = require('../core/IRTransformer');
|
21
|
-
const partitionArray = require('../util/partitionArray');
|
22
|
-
const IMap = require('immutable').Map;
|
23
|
-
const invariant = require('invariant');
|
24
|
-
|
25
|
-
/**
|
26
|
-
* A simplified representation of a document: keys in the map are unique
|
27
|
-
* identifiers for the selections of a node, values are either null (for scalars)
|
28
|
-
* or nested maps for items with subselections (linked fields, inline fragments,
|
29
|
-
* etc).
|
30
|
-
*/
|
31
|
-
// $FlowFixMe[value-as-type]
|
32
|
-
type SelectionMap = IMap<string, ?SelectionMap>;
|
33
|
-
|
34
|
-
/**
|
35
|
-
* A transform that removes redundant fields and fragment spreads. Redundancy is
|
36
|
-
* defined in this context as any selection that is guaranteed to already be
|
37
|
-
* fetched by an ancestor selection. This can occur in two cases:
|
38
|
-
*
|
39
|
-
* 1. Simple duplicates at the same level of the document can always be skipped:
|
40
|
-
*
|
41
|
-
* ```
|
42
|
-
* fragment Foo on FooType {
|
43
|
-
* id
|
44
|
-
* id
|
45
|
-
* ...Bar
|
46
|
-
* ...Bar
|
47
|
-
* }
|
48
|
-
* ```
|
49
|
-
*
|
50
|
-
* Becomes
|
51
|
-
*
|
52
|
-
* ```
|
53
|
-
* fragment Foo on FooType {
|
54
|
-
* id
|
55
|
-
* ...Bar
|
56
|
-
* }
|
57
|
-
* ```
|
58
|
-
*
|
59
|
-
* 2. Inline fragments and conditions introduce the possibility for duplication
|
60
|
-
* at different levels of the tree. Whenever a selection is fetched in a parent,
|
61
|
-
* it is redundant to also fetch it in a child:
|
62
|
-
*
|
63
|
-
* ```
|
64
|
-
* fragment Foo on FooType {
|
65
|
-
* id
|
66
|
-
* ... on OtherType {
|
67
|
-
* id # 1
|
68
|
-
* }
|
69
|
-
* ... on FooType @include(if: $cond) {
|
70
|
-
* id # 2
|
71
|
-
* }
|
72
|
-
* }
|
73
|
-
* ```
|
74
|
-
*
|
75
|
-
* Becomes:
|
76
|
-
*
|
77
|
-
* ```
|
78
|
-
* fragment Foo on FooType {
|
79
|
-
* id
|
80
|
-
* }
|
81
|
-
* ```
|
82
|
-
*
|
83
|
-
* In this example:
|
84
|
-
* - 1 can be skipped because `id` is already fetched by the parent. Even
|
85
|
-
* though the type is different (FooType/OtherType), the inline fragment
|
86
|
-
* cannot match without the outer fragment matching so the outer `id` is
|
87
|
-
* guaranteed to already be fetched.
|
88
|
-
* - 2 can be skipped for similar reasons: it doesn't matter if the condition
|
89
|
-
* holds, `id` is already fetched by the parent regardless.
|
90
|
-
*
|
91
|
-
* This transform also handles more complicated cases in which selections are
|
92
|
-
* nested:
|
93
|
-
*
|
94
|
-
* ```
|
95
|
-
* fragment Foo on FooType {
|
96
|
-
* a {
|
97
|
-
* bb
|
98
|
-
* }
|
99
|
-
* ... on OtherType {
|
100
|
-
* a {
|
101
|
-
* bb # 1
|
102
|
-
* cc
|
103
|
-
* }
|
104
|
-
* }
|
105
|
-
* }
|
106
|
-
* ```
|
107
|
-
*
|
108
|
-
* Becomes
|
109
|
-
*
|
110
|
-
* ```
|
111
|
-
* fragment Foo on FooType {
|
112
|
-
* a {
|
113
|
-
* bb
|
114
|
-
* }
|
115
|
-
* ... on OtherType {
|
116
|
-
* a {
|
117
|
-
* cc
|
118
|
-
* }
|
119
|
-
* }
|
120
|
-
* }
|
121
|
-
* ```
|
122
|
-
*
|
123
|
-
* 1 can be skipped because it is already fetched at the outer level.
|
124
|
-
*/
|
125
|
-
function skipRedundantNodesTransform(
|
126
|
-
context: CompilerContext,
|
127
|
-
): CompilerContext {
|
128
|
-
return IRTransformer.transform(context, {
|
129
|
-
Root: visitNode,
|
130
|
-
SplitOperation: visitNode,
|
131
|
-
Fragment: visitNode,
|
132
|
-
});
|
133
|
-
}
|
134
|
-
|
135
|
-
let cache = new Map();
|
136
|
-
function visitNode<T: Fragment | Root | SplitOperation>(node: T): ?T {
|
137
|
-
cache = new Map();
|
138
|
-
const context: CompilerContext = this.getContext();
|
139
|
-
return transformNode(context.getSchema(), node, new IMap()).node;
|
140
|
-
}
|
141
|
-
|
142
|
-
/**
|
143
|
-
* The most straightforward approach would be two passes: one to record the
|
144
|
-
* structure of the document, one to prune duplicates. This implementation uses
|
145
|
-
* a single pass. Selections are sorted with fields first, "conditionals"
|
146
|
-
* (inline fragments & conditions) last. This means that all fields that are
|
147
|
-
* guaranteed to be fetched are encountered prior to any duplicates that may be
|
148
|
-
* fetched within a conditional.
|
149
|
-
*
|
150
|
-
* Because selections fetched within a conditional are not guaranteed to be
|
151
|
-
* fetched in the parent, a fork of the selection map is created when entering a
|
152
|
-
* conditional. The sort ensures that guaranteed fields have already been seen
|
153
|
-
* prior to the clone.
|
154
|
-
*/
|
155
|
-
function transformNode<T: Node>(
|
156
|
-
schema: Schema,
|
157
|
-
node: T,
|
158
|
-
selectionMap: SelectionMap,
|
159
|
-
): {
|
160
|
-
selectionMap: SelectionMap,
|
161
|
-
node: ?T,
|
162
|
-
...
|
163
|
-
} {
|
164
|
-
// This will optimize a traversal of the same subselections.
|
165
|
-
// If it's the same node, and selectionMap is empty
|
166
|
-
// result of transformNode has to be the same.
|
167
|
-
const isEmptySelectionMap = selectionMap.size === 0;
|
168
|
-
let result;
|
169
|
-
if (isEmptySelectionMap) {
|
170
|
-
// $FlowFixMe[escaped-generic]
|
171
|
-
result = cache.get(node);
|
172
|
-
if (result != null) {
|
173
|
-
return result;
|
174
|
-
}
|
175
|
-
}
|
176
|
-
const selections = [];
|
177
|
-
sortSelections(node.selections).forEach(selection => {
|
178
|
-
const identifier = getIdentifierForSelection(schema, selection);
|
179
|
-
switch (selection.kind) {
|
180
|
-
case 'ScalarField':
|
181
|
-
case 'FragmentSpread': {
|
182
|
-
if (!selectionMap.has(identifier)) {
|
183
|
-
selections.push(selection);
|
184
|
-
selectionMap = selectionMap.set(identifier, null);
|
185
|
-
}
|
186
|
-
break;
|
187
|
-
}
|
188
|
-
case 'Defer':
|
189
|
-
case 'Stream':
|
190
|
-
case 'ModuleImport':
|
191
|
-
case 'ClientExtension':
|
192
|
-
case 'InlineDataFragmentSpread':
|
193
|
-
case 'LinkedField': {
|
194
|
-
const transformed = transformNode(
|
195
|
-
schema,
|
196
|
-
selection,
|
197
|
-
selectionMap.get(identifier) || new IMap(),
|
198
|
-
);
|
199
|
-
if (transformed.node) {
|
200
|
-
selections.push(transformed.node);
|
201
|
-
selectionMap = selectionMap.set(identifier, transformed.selectionMap);
|
202
|
-
}
|
203
|
-
break;
|
204
|
-
}
|
205
|
-
case 'InlineFragment':
|
206
|
-
case 'Condition': {
|
207
|
-
// Fork the selection map to prevent conditional selections from
|
208
|
-
// affecting the outer "guaranteed" selections.
|
209
|
-
const transformed = transformNode(
|
210
|
-
schema,
|
211
|
-
selection,
|
212
|
-
selectionMap.get(identifier) || selectionMap,
|
213
|
-
);
|
214
|
-
if (transformed.node) {
|
215
|
-
selections.push(transformed.node);
|
216
|
-
selectionMap = selectionMap.set(identifier, transformed.selectionMap);
|
217
|
-
}
|
218
|
-
break;
|
219
|
-
}
|
220
|
-
default:
|
221
|
-
(selection: empty);
|
222
|
-
invariant(
|
223
|
-
false,
|
224
|
-
'SkipRedundantNodesTransform: Unexpected node kind `%s`.',
|
225
|
-
selection.kind,
|
226
|
-
);
|
227
|
-
}
|
228
|
-
});
|
229
|
-
const nextNode: any = selections.length ? {...node, selections} : null;
|
230
|
-
result = {selectionMap, node: nextNode};
|
231
|
-
if (isEmptySelectionMap) {
|
232
|
-
// $FlowFixMe[escaped-generic]
|
233
|
-
cache.set(node, result);
|
234
|
-
}
|
235
|
-
return result;
|
236
|
-
}
|
237
|
-
|
238
|
-
/**
|
239
|
-
* Sort inline fragments and conditions after other selections.
|
240
|
-
*/
|
241
|
-
function sortSelections(
|
242
|
-
selections: $ReadOnlyArray<Selection>,
|
243
|
-
): $ReadOnlyArray<Selection> {
|
244
|
-
const isScalarOrLinkedField = selection =>
|
245
|
-
selection.kind === 'ScalarField' || selection.kind === 'LinkedField';
|
246
|
-
const [scalarsAndLinkedFields, rest] = partitionArray(
|
247
|
-
selections,
|
248
|
-
isScalarOrLinkedField,
|
249
|
-
);
|
250
|
-
return [...scalarsAndLinkedFields, ...rest];
|
251
|
-
}
|
252
|
-
|
253
|
-
module.exports = {
|
254
|
-
transform: skipRedundantNodesTransform,
|
255
|
-
};
|
@@ -1,37 +0,0 @@
|
|
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
|
-
* @flow strict-local
|
8
|
-
* @format
|
9
|
-
*/
|
10
|
-
|
11
|
-
// flowlint ambiguous-object-type:error
|
12
|
-
|
13
|
-
'use strict';
|
14
|
-
|
15
|
-
import type CompilerContext from '../core/CompilerContext';
|
16
|
-
|
17
|
-
const IRTransformer = require('../core/IRTransformer');
|
18
|
-
|
19
|
-
function skipNode() {
|
20
|
-
return null;
|
21
|
-
}
|
22
|
-
|
23
|
-
/**
|
24
|
-
* A transform that removes field `splitOperations`. Intended for use when e.g.
|
25
|
-
* printing queries to send to a GraphQL server.
|
26
|
-
*/
|
27
|
-
function skipSplitOperationTransform(
|
28
|
-
context: CompilerContext,
|
29
|
-
): CompilerContext {
|
30
|
-
return IRTransformer.transform(context, {
|
31
|
-
SplitOperation: skipNode,
|
32
|
-
});
|
33
|
-
}
|
34
|
-
|
35
|
-
module.exports = {
|
36
|
-
transform: skipSplitOperationTransform,
|
37
|
-
};
|
@@ -1,148 +0,0 @@
|
|
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
|
-
* @flow strict-local
|
8
|
-
* @format
|
9
|
-
*/
|
10
|
-
|
11
|
-
// flowlint ambiguous-object-type:error
|
12
|
-
|
13
|
-
'use strict';
|
14
|
-
|
15
|
-
import type CompilerContext from '../core/CompilerContext';
|
16
|
-
import type {Condition, Fragment, Node, Selection} from '../core/IR';
|
17
|
-
|
18
|
-
const IRTransformer = require('../core/IRTransformer');
|
19
|
-
const invariant = require('invariant');
|
20
|
-
|
21
|
-
type ConditionResult = 'fail' | 'pass' | 'variable';
|
22
|
-
|
23
|
-
const FAIL = 'fail';
|
24
|
-
const PASS = 'pass';
|
25
|
-
const VARIABLE = 'variable';
|
26
|
-
|
27
|
-
/**
|
28
|
-
* A tranform that removes unreachable IR nodes from all documents in a corpus.
|
29
|
-
* The following nodes are removed:
|
30
|
-
* - Any node with `@include(if: false)`
|
31
|
-
* - Any node with `@skip(if: true)`
|
32
|
-
* - Any node with empty `selections`
|
33
|
-
*/
|
34
|
-
function skipUnreachableNodeTransform(
|
35
|
-
context: CompilerContext,
|
36
|
-
): CompilerContext {
|
37
|
-
const fragments: Map<string, ?Fragment> = new Map();
|
38
|
-
const nextContext = IRTransformer.transform(context, {
|
39
|
-
Root: node => transformNode(context, fragments, node),
|
40
|
-
SplitOperation: node => transformNode(context, fragments, node),
|
41
|
-
// Fragments are included below where referenced.
|
42
|
-
// Unreferenced fragments are not included.
|
43
|
-
Fragment: id => null,
|
44
|
-
});
|
45
|
-
return (Array.from(fragments.values()): Array<?Fragment>).reduce(
|
46
|
-
(ctx: CompilerContext, fragment) => (fragment ? ctx.add(fragment) : ctx),
|
47
|
-
nextContext,
|
48
|
-
);
|
49
|
-
}
|
50
|
-
|
51
|
-
function transformNode<T: Node>(
|
52
|
-
context: CompilerContext,
|
53
|
-
fragments: Map<string, ?Fragment>,
|
54
|
-
node: T,
|
55
|
-
): ?T {
|
56
|
-
const queue: Array<Selection> = [...node.selections];
|
57
|
-
let selections;
|
58
|
-
while (queue.length) {
|
59
|
-
const selection: Selection = queue.shift();
|
60
|
-
let nextSelection;
|
61
|
-
switch (selection.kind) {
|
62
|
-
case 'Condition':
|
63
|
-
const match = testCondition(selection);
|
64
|
-
if (match === PASS) {
|
65
|
-
queue.unshift(...selection.selections);
|
66
|
-
} else if (match === VARIABLE) {
|
67
|
-
nextSelection = transformNode(context, fragments, selection);
|
68
|
-
}
|
69
|
-
break;
|
70
|
-
case 'FragmentSpread': {
|
71
|
-
// Skip fragment spreads if the referenced fragment is empty
|
72
|
-
if (!fragments.has(selection.name)) {
|
73
|
-
const fragment = context.getFragment(selection.name);
|
74
|
-
const nextFragment = transformNode(context, fragments, fragment);
|
75
|
-
fragments.set(selection.name, nextFragment);
|
76
|
-
}
|
77
|
-
if (fragments.get(selection.name)) {
|
78
|
-
nextSelection = selection;
|
79
|
-
}
|
80
|
-
break;
|
81
|
-
}
|
82
|
-
case 'ClientExtension':
|
83
|
-
nextSelection = transformNode(context, fragments, selection);
|
84
|
-
break;
|
85
|
-
case 'ModuleImport':
|
86
|
-
nextSelection = transformNode(context, fragments, selection);
|
87
|
-
break;
|
88
|
-
case 'LinkedField':
|
89
|
-
nextSelection = transformNode(context, fragments, selection);
|
90
|
-
break;
|
91
|
-
case 'InlineFragment':
|
92
|
-
// TODO combine with the LinkedField case when flow supports this
|
93
|
-
nextSelection = transformNode(context, fragments, selection);
|
94
|
-
break;
|
95
|
-
case 'Defer':
|
96
|
-
nextSelection = transformNode(context, fragments, selection);
|
97
|
-
break;
|
98
|
-
case 'Stream':
|
99
|
-
nextSelection = transformNode(context, fragments, selection);
|
100
|
-
break;
|
101
|
-
case 'ScalarField':
|
102
|
-
nextSelection = selection;
|
103
|
-
break;
|
104
|
-
case 'InlineDataFragmentSpread':
|
105
|
-
invariant(
|
106
|
-
false,
|
107
|
-
'SkipUnreachableNodeTransform: Did not expect an ' +
|
108
|
-
'InlineDataFragmentSpread here. Only expecting ' +
|
109
|
-
'InlineDataFragmentSpread in reader ASTs and this transform to ' +
|
110
|
-
'run only on normalization ASTs.',
|
111
|
-
);
|
112
|
-
// fallthrough
|
113
|
-
default:
|
114
|
-
(selection.kind: empty);
|
115
|
-
invariant(
|
116
|
-
false,
|
117
|
-
'SkipUnreachableNodeTransform: Unexpected selection kind `%s`.',
|
118
|
-
selection.kind,
|
119
|
-
);
|
120
|
-
}
|
121
|
-
if (nextSelection) {
|
122
|
-
selections = selections || [];
|
123
|
-
selections.push(nextSelection);
|
124
|
-
}
|
125
|
-
}
|
126
|
-
if (selections) {
|
127
|
-
return ({
|
128
|
-
...node,
|
129
|
-
selections,
|
130
|
-
}: $FlowIssue);
|
131
|
-
}
|
132
|
-
return null;
|
133
|
-
}
|
134
|
-
|
135
|
-
/**
|
136
|
-
* Determines whether a condition statically passes/fails or is unknown
|
137
|
-
* (variable).
|
138
|
-
*/
|
139
|
-
function testCondition(condition: Condition): ConditionResult {
|
140
|
-
if (condition.condition.kind === 'Variable') {
|
141
|
-
return VARIABLE;
|
142
|
-
}
|
143
|
-
return condition.condition.value === condition.passingValue ? PASS : FAIL;
|
144
|
-
}
|
145
|
-
|
146
|
-
module.exports = {
|
147
|
-
transform: skipUnreachableNodeTransform,
|
148
|
-
};
|
@@ -1,59 +0,0 @@
|
|
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
|
-
* @flow strict-local
|
8
|
-
* @format
|
9
|
-
*/
|
10
|
-
|
11
|
-
// flowlint ambiguous-object-type:error
|
12
|
-
|
13
|
-
'use strict';
|
14
|
-
|
15
|
-
import type CompilerContext from '../core/CompilerContext';
|
16
|
-
import type {Root} from '../core/IR';
|
17
|
-
|
18
|
-
const inferRootArgumentDefinitions = require('../core/inferRootArgumentDefinitions');
|
19
|
-
|
20
|
-
/**
|
21
|
-
* Refines the argument definitions for operations to remove unused arguments
|
22
|
-
* due to statically pruned conditional branches (e.g. because of overriding
|
23
|
-
* a variable used in `@include()` to be false).
|
24
|
-
*/
|
25
|
-
function skipUnusedVariablesTransform(
|
26
|
-
context: CompilerContext,
|
27
|
-
): CompilerContext {
|
28
|
-
const contextWithUsedArguments = inferRootArgumentDefinitions(context);
|
29
|
-
return context.withMutations(ctx => {
|
30
|
-
let nextContext = ctx;
|
31
|
-
for (const node of nextContext.documents()) {
|
32
|
-
if (node.kind !== 'Root') {
|
33
|
-
continue;
|
34
|
-
}
|
35
|
-
const usedArguments = new Set(
|
36
|
-
contextWithUsedArguments
|
37
|
-
.getRoot(node.name)
|
38
|
-
.argumentDefinitions.map(argDef => argDef.name),
|
39
|
-
);
|
40
|
-
// Remove unused argument definitions
|
41
|
-
const usedArgumentDefinitions = node.argumentDefinitions.filter(argDef =>
|
42
|
-
usedArguments.has(argDef.name),
|
43
|
-
);
|
44
|
-
if (usedArgumentDefinitions.length !== node.argumentDefinitions.length) {
|
45
|
-
nextContext = nextContext.replace(
|
46
|
-
({
|
47
|
-
...node,
|
48
|
-
argumentDefinitions: usedArgumentDefinitions,
|
49
|
-
}: Root),
|
50
|
-
);
|
51
|
-
}
|
52
|
-
}
|
53
|
-
return nextContext;
|
54
|
-
});
|
55
|
-
}
|
56
|
-
|
57
|
-
module.exports = {
|
58
|
-
transform: skipUnusedVariablesTransform,
|
59
|
-
};
|
@@ -1,97 +0,0 @@
|
|
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
|
-
* @flow strict-local
|
8
|
-
* @format
|
9
|
-
*/
|
10
|
-
|
11
|
-
// flowlint ambiguous-object-type:error
|
12
|
-
|
13
|
-
'use strict';
|
14
|
-
|
15
|
-
import type CompilerContext from '../core/CompilerContext';
|
16
|
-
import type {
|
17
|
-
InlineFragment,
|
18
|
-
LinkedField,
|
19
|
-
ModuleImport,
|
20
|
-
SplitOperation,
|
21
|
-
} from '../core/IR';
|
22
|
-
import type {CompositeTypeID} from '../core/Schema';
|
23
|
-
|
24
|
-
const getNormalizationOperationName = require('../core/getNormalizationOperationName');
|
25
|
-
const IRTransformer = require('../core/IRTransformer');
|
26
|
-
|
27
|
-
type State = {|
|
28
|
-
parentType: CompositeTypeID,
|
29
|
-
splitOperations: Map<string, SplitOperation>,
|
30
|
-
|};
|
31
|
-
|
32
|
-
/**
|
33
|
-
* This transform creates a SplitOperation root for every ModuleImport.
|
34
|
-
*/
|
35
|
-
function splitMatchTransform(context: CompilerContext): CompilerContext {
|
36
|
-
const splitOperations = new Map();
|
37
|
-
const transformedContext = IRTransformer.transform(
|
38
|
-
context,
|
39
|
-
{
|
40
|
-
LinkedField: visitLinkedField,
|
41
|
-
InlineFragment: visitInlineFragment,
|
42
|
-
ModuleImport: visitModuleImport,
|
43
|
-
},
|
44
|
-
node => ({
|
45
|
-
parentType: node.type,
|
46
|
-
splitOperations,
|
47
|
-
}),
|
48
|
-
);
|
49
|
-
return transformedContext.addAll(Array.from(splitOperations.values()));
|
50
|
-
}
|
51
|
-
|
52
|
-
function visitLinkedField(field: LinkedField, state: State): LinkedField {
|
53
|
-
return this.traverse(field, {
|
54
|
-
parentType: field.type,
|
55
|
-
splitOperations: state.splitOperations,
|
56
|
-
});
|
57
|
-
}
|
58
|
-
|
59
|
-
function visitInlineFragment(
|
60
|
-
fragment: InlineFragment,
|
61
|
-
state: State,
|
62
|
-
): InlineFragment {
|
63
|
-
return this.traverse(fragment, {
|
64
|
-
parentType: fragment.typeCondition,
|
65
|
-
splitOperations: state.splitOperations,
|
66
|
-
});
|
67
|
-
}
|
68
|
-
|
69
|
-
function visitModuleImport(node: ModuleImport, state: State): ModuleImport {
|
70
|
-
// It's possible for the same fragment to be selected in multiple usages
|
71
|
-
// of @module: skip processing a node if its SplitOperation has already
|
72
|
-
// been generated
|
73
|
-
const normalizationName = getNormalizationOperationName(node.name);
|
74
|
-
const createdSplitOperation = state.splitOperations.get(normalizationName);
|
75
|
-
if (createdSplitOperation) {
|
76
|
-
createdSplitOperation.parentSources.add(node.sourceDocument);
|
77
|
-
return node;
|
78
|
-
}
|
79
|
-
const transformedNode = this.traverse(node, state);
|
80
|
-
const splitOperation: SplitOperation = {
|
81
|
-
kind: 'SplitOperation',
|
82
|
-
name: normalizationName,
|
83
|
-
selections: transformedNode.selections,
|
84
|
-
loc: {kind: 'Derived', source: node.loc},
|
85
|
-
parentSources: new Set([node.sourceDocument]),
|
86
|
-
metadata: {
|
87
|
-
derivedFrom: transformedNode.name,
|
88
|
-
},
|
89
|
-
type: state.parentType,
|
90
|
-
};
|
91
|
-
state.splitOperations.set(normalizationName, splitOperation);
|
92
|
-
return transformedNode;
|
93
|
-
}
|
94
|
-
|
95
|
-
module.exports = {
|
96
|
-
transform: splitMatchTransform,
|
97
|
-
};
|