Skip to content

Commit a99fc38

Browse files
authored
Support multiple 'generate' configs, allowing projects to have different settings for different directories or files (#51)
## Summary: This will allow CLC to turn on experimentalEnums in a limited fashion, and will also allow us to turn on type generation for the course editor stuff, which uses a different schema. Issue: XXX-XXXX ## Test plan: jest, eslint & flow - [x] in the mobile repo, modify the config to use the new format & try out this version (https://github.com/Khan/mobile/blob/gqf-ok/react-native/tools/graphql-flow.json). See that no generated types are changed - [x] in the webapp repo, do the same, see that no generated types are changed (https://github.com/Khan/webapp/blob/gqf-multi/services/static/dev/graphql-flow/config.js). Also test out the multiple generate configs, enabling experimentalEnums for just services/static/javascript/discussion-package Author: jaredly Reviewers: nedredmond, jaredly, benchristel, kevinbarabash, jeremywiebe Required Reviewers: Approved By: nedredmond, nedredmond Checks: ✅ Lint & Test (ubuntu-latest, 16.x) Pull Request URL: #51
1 parent 1e1de13 commit a99fc38

File tree

8 files changed

+224
-53
lines changed

8 files changed

+224
-53
lines changed

.changeset/wise-kiwis-speak.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
'@khanacademy/graphql-flow': minor
3+
---
4+
5+
Support multiple 'generate' configs, allowing projects to have different settings for different directories or files
6+
7+
`config.generate` can now be an object or an array of objects, with `match` and `exclude` arrays (either a RegExp or a string that will be passed to `new RegExp()`) to fine-tune which files they apply to.

Readme.md

+47-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ This is a tool for generating flow types from graphql queries in javascript fron
44

55
## Using as a CLI tool
66

7-
Write a config file, following the schema defined in [src/cli/schema.json](src/cli/schema.json).
7+
Write a config file, following the schema defined in [src/cli/schema.json](src/cli/schema.json), either as a `.json` file, or a `.js` file that `module.exports` an object adhering to the schema.
88

99
Then run from the CLI, like so:
1010

@@ -14,6 +14,52 @@ $ graphql-flow path/to/config.json
1414

1515
Files will be discovered relative to the `crawl.root`.
1616

17+
### Multiple generate configs
18+
19+
To customize type generation for certain directories or files, you can provide multiple
20+
`generate` configs as an array, using `match` and `exclude` to customize behavior.
21+
22+
For a given file containing operations, the first `generate` config that matches that path
23+
(and doesn't exclude it) will be used to generate types for those operations. If a `generate`
24+
config doesn't have a `match` attribute, it will match all files (but might exclude some via the
25+
`exclude` attribute).
26+
27+
For example:
28+
29+
```js
30+
// dev/graphql-flow/config.js
31+
32+
const options = {
33+
schemaFilePath: "../../gengraphql/composed-schema.graphql",
34+
regenerateCommand: "make gqlflow",
35+
generatedDirectory: "__graphql-types__",
36+
exclude: [
37+
/_test.js$/,
38+
/.fixture.js$/,
39+
/\b__flowtests__\b/,
40+
],
41+
};
42+
43+
module.exports = {
44+
crawl: {
45+
root: "../../",
46+
},
47+
generate: [
48+
{
49+
...options,
50+
schemaFilePath: "../../gengraphql/course-editor-schema.graphql",
51+
match: [/\bcourse-editor-package\b/, /\bcourse-editor\b/],
52+
},
53+
{
54+
...options,
55+
match: [/\bdiscussion-package\b/]
56+
experimentalEnums: true,
57+
},
58+
options,
59+
],
60+
};
61+
```
62+
1763
## Introspecting your backend's graphql schema
1864
Here's how to get your backend's schema in the way that this tool expects, using the builtin 'graphql introspection query':
1965

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
"scripts": {
88
"test": "jest",
99
"publish:ci": "yarn run build && yarn run copy-flow && changeset publish",
10-
"build": "babel src --out-dir dist --source-maps --ignore 'src/**/*.spec.js','src/**/*.test.js'",
10+
"build": "babel src -D --out-dir dist --source-maps --ignore 'src/**/*.spec.js','src/**/*.test.js'",
1111
"copy-flow": "node ./build-copy-source.js"
1212
},
1313
"main": "dist/index.js",

src/cli/__test__/config.test.js

+73-6
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,53 @@
11
// @flow
22
import type {Config} from '../../types';
33

4-
import {validateOrThrow} from '../config';
4+
import {findApplicableConfig, validateOrThrow} from '../config';
55
import configSchema from '../schema.json'; // eslint-disable-line flowtype-errors/uncovered
66

7+
describe('findApplicableConfig', () => {
8+
it('should work with one that matches', () => {
9+
const config = {
10+
schemaFilePath: 'ok.graphql',
11+
};
12+
expect(findApplicableConfig('/hello', config)).toBe(config);
13+
});
14+
15+
it('should be falsy if nothing matches', () => {
16+
const config = {
17+
schemaFilePath: 'ok.graphql',
18+
exclude: [/hello$/],
19+
};
20+
expect(findApplicableConfig('/hello', config)).toBeUndefined();
21+
});
22+
23+
it('should match & exclude with multiple configs', () => {
24+
const configs = [
25+
{schemaFilePath: 'one', match: [/\.jsx$/], exclude: [/^test/]},
26+
{schemaFilePath: 'two', exclude: [/^hello/]},
27+
{schemaFilePath: 'three'},
28+
];
29+
expect(findApplicableConfig('hello.js', configs)).toBe(configs[2]);
30+
expect(findApplicableConfig('goodbye.js', configs)).toBe(configs[1]);
31+
expect(findApplicableConfig('hello.jsx', configs)).toBe(configs[0]);
32+
expect(findApplicableConfig('test.jsx', configs)).toBe(configs[1]);
33+
});
34+
});
35+
736
describe('jsonschema validation', () => {
837
it('should accept valid schema', () => {
938
const config: Config = {
1039
crawl: {
1140
root: '/here/we/crawl',
12-
excludes: [
13-
'_test.js$',
41+
},
42+
generate: {
43+
match: [/\.fixture\.js$/],
44+
exclude: [
45+
'_test\\.js$',
1446
'\\bcourse-editor-package\\b',
15-
'.fixture.js$',
47+
'\\.fixture\\.js$',
1648
'\\b__flowtests__\\b',
1749
'\\bcourse-editor\\b',
1850
],
19-
},
20-
generate: {
2151
readOnlyArray: false,
2252
regenerateCommand: 'make gqlflow',
2353
scalars: {
@@ -37,6 +67,43 @@ describe('jsonschema validation', () => {
3767
);
3868
});
3969

70+
it('should accept a schema with multiple generate configs', () => {
71+
const generate = {
72+
match: [/\.fixture\.js$/],
73+
exclude: [
74+
'_test\\.js$',
75+
'\\bcourse-editor-package\\b',
76+
'\\.fixture\\.js$',
77+
'\\b__flowtests__\\b',
78+
'\\bcourse-editor\\b',
79+
],
80+
readOnlyArray: false,
81+
regenerateCommand: 'make gqlflow',
82+
scalars: {
83+
JSONString: 'string',
84+
KALocale: 'string',
85+
NaiveDateTime: 'string',
86+
},
87+
splitTypes: true,
88+
generatedDirectory: '__graphql-types__',
89+
exportAllObjectTypes: true,
90+
schemaFilePath: './composed_schema.graphql',
91+
};
92+
const config: Config = {
93+
crawl: {
94+
root: '/here/we/crawl',
95+
},
96+
generate: [
97+
{...generate, match: [/^static/], exportAllObjectTypes: false},
98+
generate,
99+
],
100+
};
101+
validateOrThrow(
102+
config,
103+
configSchema, // eslint-disable-line flowtype-errors/uncovered
104+
);
105+
});
106+
40107
it('should reject invalid schema', () => {
41108
expect(() =>
42109
validateOrThrow(

src/cli/config.js

+26-3
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ import {
1313
graphqlSync,
1414
type IntrospectionQuery,
1515
} from 'graphql';
16-
import type {Config} from '../types';
16+
import type {Config, GenerateConfig} from '../types';
1717
import {validate} from 'jsonschema'; // eslint-disable-line flowtype-errors/uncovered
1818

1919
export const validateOrThrow = (value: mixed, jsonSchema: mixed) => {
@@ -28,8 +28,8 @@ export const validateOrThrow = (value: mixed, jsonSchema: mixed) => {
2828
};
2929

3030
export const loadConfigFile = (configFile: string): Config => {
31-
// eslint-disable-next-line flowtype-errors/uncovered
32-
const data: Config = JSON.parse(fs.readFileSync(configFile, 'utf8'));
31+
// $FlowIgnore // eslint-disable-next-line flowtype-errors/uncovered
32+
const data: Config = require(configFile);
3333
// eslint-disable-next-line flowtype-errors/uncovered
3434
validateOrThrow(data, configSchema);
3535
return data;
@@ -60,3 +60,26 @@ export const getSchemas = (schemaFilePath: string): [GraphQLSchema, Schema] => {
6060
return [schemaForValidation, schemaForTypeGeneration];
6161
}
6262
};
63+
64+
/**
65+
* Find the first item of the `config.generate` array where both:
66+
* - no item of `exclude` matches
67+
* - at least one item of `match` matches
68+
*/
69+
export const findApplicableConfig = (
70+
path: string,
71+
configs: Array<GenerateConfig> | GenerateConfig,
72+
): ?GenerateConfig => {
73+
if (!Array.isArray(configs)) {
74+
configs = [configs];
75+
}
76+
return configs.find((config) => {
77+
if (config.exclude?.some((exclude) => new RegExp(exclude).test(path))) {
78+
return false;
79+
}
80+
if (!config.match) {
81+
return true;
82+
}
83+
return config.match.some((matcher) => new RegExp(matcher).test(path));
84+
});
85+
};

src/cli/run.js

+29-17
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
import {generateTypeFiles, processPragmas} from '../generateTypeFiles';
55
import {processFiles} from '../parser/parse';
66
import {resolveDocuments} from '../parser/resolve';
7-
import {getSchemas, loadConfigFile} from './config';
7+
import {findApplicableConfig, getSchemas, loadConfigFile} from './config';
88

99
import {addTypenameToDocument} from 'apollo-utilities'; // eslint-disable-line flowtype-errors/uncovered
1010

@@ -68,10 +68,6 @@ const absConfigPath = makeAbsPath(configFilePath, process.cwd());
6868

6969
const config = loadConfigFile(absConfigPath);
7070

71-
const [schemaForValidation, schemaForTypeGeneration] = getSchemas(
72-
makeAbsPath(config.generate.schemaFilePath, path.dirname(absConfigPath)),
73-
);
74-
7571
const inputFiles = cliFiles.length
7672
? cliFiles
7773
: findGraphqlTagReferences(
@@ -122,23 +118,37 @@ console.log(Object.keys(resolved).length, 'resolved queries');
122118

123119
/** Step (4) */
124120

121+
const schemaCache = {};
122+
const getCachedSchemas = (schemaFilePath: string) => {
123+
if (!schemaCache[schemaFilePath]) {
124+
schemaCache[schemaFilePath] = getSchemas(
125+
makeAbsPath(schemaFilePath, path.dirname(absConfigPath)),
126+
);
127+
}
128+
129+
return schemaCache[schemaFilePath];
130+
};
131+
125132
let validationFailures: number = 0;
126133
const printedOperations: Array<string> = [];
127134

128135
Object.keys(resolved).forEach((filePathAndLine) => {
129136
const {document, raw} = resolved[filePathAndLine];
130137

131-
if (
132-
config.crawl.excludes?.some((rx) => new RegExp(rx).test(raw.loc.path))
133-
) {
134-
return; // skip
135-
}
136-
137138
const hasNonFragments = document.definitions.some(
138139
({kind}) => kind !== 'FragmentDefinition',
139140
);
140141
const rawSource: string = raw.literals[0];
141142

143+
const generateConfig = findApplicableConfig(
144+
// strip off the trailing line number, e.g. `:23`
145+
filePathAndLine.split(':')[0],
146+
config.generate,
147+
);
148+
if (!generateConfig) {
149+
return; // no generate config matches, bail
150+
}
151+
142152
// eslint-disable-next-line flowtype-errors/uncovered
143153
const withTypeNames: DocumentNode = addTypenameToDocument(document);
144154
const printed = print(withTypeNames);
@@ -147,18 +157,20 @@ Object.keys(resolved).forEach((filePathAndLine) => {
147157
}
148158

149159
const pragmaResult = processPragmas(
150-
config.generate,
160+
generateConfig,
151161
config.crawl,
152162
rawSource,
153163
);
154164
if (!pragmaResult.generate) {
155165
return;
156166
}
157-
const generateConfig: GenerateConfig = {
158-
...config.generate,
159-
strictNullability:
160-
pragmaResult.strict ?? config.generate.strictNullability,
161-
};
167+
if (pragmaResult.strict != null) {
168+
generateConfig.strictNullability = pragmaResult.strict;
169+
}
170+
171+
const [schemaForValidation, schemaForTypeGeneration] = getCachedSchemas(
172+
generateConfig.schemaFilePath,
173+
);
162174

163175
if (hasNonFragments) {
164176
/* eslint-disable flowtype-errors/uncovered */

0 commit comments

Comments
 (0)