@strapi/typescript-utils 0.0.0-next.e3b4cdeebf6e9b0cd5575ff80b8a86715d44ce98 → 0.0.0-next.e6eaa3d0563c85f80fd88b258df70a55c057096e
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/lib/__tests__/generators/schemas/attributes.test.js +179 -95
- package/lib/__tests__/generators/schemas/imports.test.js +18 -16
- package/lib/__tests__/generators/schemas/utils.test.js +5 -57
- package/lib/admin/create-tsconfig-file.js +2 -16
- package/lib/compile.js +2 -6
- package/lib/compilers/basic.js +12 -4
- package/lib/compilers/index.js +0 -2
- package/lib/generators/{schemas → common}/imports.js +8 -6
- package/lib/generators/common/index.js +9 -0
- package/lib/generators/{schemas → common/models}/attributes.js +62 -36
- package/lib/generators/common/models/index.js +15 -0
- package/lib/generators/{schemas → common/models}/mappers.js +27 -24
- package/lib/generators/{schemas → common/models}/schema.js +10 -5
- package/lib/generators/{schemas → common/models}/utils.js +28 -9
- package/lib/generators/components/index.js +58 -0
- package/lib/generators/constants.js +6 -0
- package/lib/generators/content-types/index.js +58 -0
- package/lib/generators/index.js +118 -3
- package/lib/generators/utils.js +211 -0
- package/package.json +10 -5
- package/tsconfigs/admin.json +18 -19
- package/tsconfigs/server.json +17 -16
- package/lib/__tests__/generators/schemas/global.test.js +0 -108
- package/lib/compilers/watch.js +0 -37
- package/lib/generators/schemas/global.js +0 -67
- package/lib/generators/schemas/index.js +0 -185
|
@@ -2,25 +2,11 @@
|
|
|
2
2
|
|
|
3
3
|
const path = require('path');
|
|
4
4
|
const fs = require('fs-extra');
|
|
5
|
+
const adminTsConfig = require('../../tsconfigs/admin.json');
|
|
5
6
|
|
|
6
7
|
module.exports = async (dest) => {
|
|
7
8
|
const tsConfig = {
|
|
8
|
-
|
|
9
|
-
lib: ['es2019', 'es2020.promise', 'es2020.bigint', 'es2020.string', 'DOM'],
|
|
10
|
-
noImplicitAny: false,
|
|
11
|
-
module: 'es2020',
|
|
12
|
-
target: 'es5',
|
|
13
|
-
jsx: 'react',
|
|
14
|
-
allowJs: true,
|
|
15
|
-
strict: true,
|
|
16
|
-
moduleResolution: 'node',
|
|
17
|
-
skipLibCheck: true,
|
|
18
|
-
esModuleInterop: true,
|
|
19
|
-
allowSyntheticDefaultImports: true,
|
|
20
|
-
resolveJsonModule: true,
|
|
21
|
-
noEmit: false,
|
|
22
|
-
incremental: true,
|
|
23
|
-
},
|
|
9
|
+
...adminTsConfig,
|
|
24
10
|
include: ['../../../src/admin/*', '../../../src/**/**/admin/src/*'],
|
|
25
11
|
exclude: ['node_modules', '**/*.test.js', '*.js'],
|
|
26
12
|
};
|
package/lib/compile.js
CHANGED
|
@@ -3,12 +3,8 @@
|
|
|
3
3
|
const compilers = require('./compilers');
|
|
4
4
|
const getConfigPath = require('./utils/get-config-path');
|
|
5
5
|
|
|
6
|
-
module.exports = async (srcDir, {
|
|
7
|
-
// TODO: Use the Strapi debug logger instead or don't log at all
|
|
8
|
-
console.log(`Starting the compilation for TypeScript files in ${srcDir}`);
|
|
9
|
-
|
|
10
|
-
const compiler = watch ? compilers.watch : compilers.basic;
|
|
6
|
+
module.exports = async (srcDir, { configOptions = {} } = {}) => {
|
|
11
7
|
const configPath = getConfigPath(srcDir);
|
|
12
8
|
|
|
13
|
-
|
|
9
|
+
compilers.basic.run(configPath, configOptions);
|
|
14
10
|
};
|
package/lib/compilers/basic.js
CHANGED
|
@@ -13,15 +13,23 @@ module.exports = {
|
|
|
13
13
|
* @param {Object} configOptions
|
|
14
14
|
* @param {Array.<string>} configOptions.fileNames
|
|
15
15
|
* @param {Object} configOptions.options
|
|
16
|
+
* @param {boolean} configOptions.ignoreDiagnostics
|
|
16
17
|
*/
|
|
17
18
|
run(tsConfigPath, configOptions = {}) {
|
|
19
|
+
const { ignoreDiagnostics = false } = configOptions;
|
|
18
20
|
// Parse the tsconfig.json file & resolve the configuration options
|
|
19
21
|
const { fileNames, options, projectReferences } = resolveConfigOptions(tsConfigPath);
|
|
20
22
|
|
|
23
|
+
const compilerOptions = merge(options, configOptions.options);
|
|
24
|
+
|
|
25
|
+
if (ignoreDiagnostics) {
|
|
26
|
+
Object.assign(compilerOptions, { noEmit: false, noEmitOnError: false });
|
|
27
|
+
}
|
|
28
|
+
|
|
21
29
|
const program = ts.createProgram({
|
|
22
30
|
rootNames: configOptions.fileNames ? configOptions.fileNames : fileNames,
|
|
23
31
|
projectReferences,
|
|
24
|
-
options:
|
|
32
|
+
options: compilerOptions,
|
|
25
33
|
});
|
|
26
34
|
|
|
27
35
|
const emitResults = program.emit();
|
|
@@ -30,12 +38,12 @@ module.exports = {
|
|
|
30
38
|
ts.getPreEmitDiagnostics(program).concat(emitResults.diagnostics)
|
|
31
39
|
);
|
|
32
40
|
|
|
33
|
-
if (diagnostics.length > 0) {
|
|
41
|
+
if (!ignoreDiagnostics && diagnostics.length > 0) {
|
|
34
42
|
reportDiagnostics(diagnostics);
|
|
35
43
|
}
|
|
36
44
|
|
|
37
|
-
// If the compilation failed, exit early
|
|
38
|
-
if (emitResults.emitSkipped) {
|
|
45
|
+
// If the compilation failed and diagnostics are not ignored, exit early
|
|
46
|
+
if (!ignoreDiagnostics && emitResults.emitSkipped) {
|
|
39
47
|
process.exit(1);
|
|
40
48
|
}
|
|
41
49
|
},
|
package/lib/compilers/index.js
CHANGED
|
@@ -22,11 +22,13 @@ module.exports = {
|
|
|
22
22
|
factory.createImportSpecifier(false, undefined, factory.createIdentifier(key))
|
|
23
23
|
);
|
|
24
24
|
|
|
25
|
-
return
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
25
|
+
return [
|
|
26
|
+
factory.createImportDeclaration(
|
|
27
|
+
undefined,
|
|
28
|
+
factory.createImportClause(true, undefined, factory.createNamedImports(formattedImports)),
|
|
29
|
+
factory.createStringLiteral('@strapi/strapi'),
|
|
30
|
+
undefined
|
|
31
|
+
),
|
|
32
|
+
];
|
|
31
33
|
},
|
|
32
34
|
};
|
|
@@ -1,12 +1,14 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
-
const
|
|
3
|
+
const ts = require('typescript');
|
|
4
4
|
const _ = require('lodash/fp');
|
|
5
5
|
|
|
6
|
-
const { addImport } = require('
|
|
7
|
-
const { getTypeNode, toTypeLiteral } = require('./utils');
|
|
6
|
+
const { addImport } = require('../imports');
|
|
7
|
+
const { getTypeNode, toTypeLiteral, withAttributeNamespace, NAMESPACES } = require('./utils');
|
|
8
8
|
const mappers = require('./mappers');
|
|
9
9
|
|
|
10
|
+
const { factory } = ts;
|
|
11
|
+
|
|
10
12
|
/**
|
|
11
13
|
* Create the base type node for a given attribute
|
|
12
14
|
*
|
|
@@ -26,7 +28,8 @@ const getAttributeType = (attributeName, attribute, uid) => {
|
|
|
26
28
|
|
|
27
29
|
const [attributeType, typeParams] = mappers[attribute.type]({ uid, attribute, attributeName });
|
|
28
30
|
|
|
29
|
-
|
|
31
|
+
// Make sure the attribute namespace is imported
|
|
32
|
+
addImport(NAMESPACES.attribute);
|
|
30
33
|
|
|
31
34
|
return getTypeNode(attributeType, typeParams);
|
|
32
35
|
};
|
|
@@ -42,38 +45,36 @@ const getAttributeModifiers = (attribute) => {
|
|
|
42
45
|
|
|
43
46
|
// Required
|
|
44
47
|
if (attribute.required) {
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
+
modifiers.push(
|
|
49
|
+
factory.createTypeReferenceNode(factory.createIdentifier(withAttributeNamespace('Required')))
|
|
50
|
+
);
|
|
48
51
|
}
|
|
49
52
|
|
|
50
53
|
// Private
|
|
51
54
|
if (attribute.private) {
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
+
modifiers.push(
|
|
56
|
+
factory.createTypeReferenceNode(factory.createIdentifier(withAttributeNamespace('Private')))
|
|
57
|
+
);
|
|
55
58
|
}
|
|
56
59
|
|
|
57
60
|
// Unique
|
|
58
61
|
if (attribute.unique) {
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
+
modifiers.push(
|
|
63
|
+
factory.createTypeReferenceNode(factory.createIdentifier(withAttributeNamespace('Unique')))
|
|
64
|
+
);
|
|
62
65
|
}
|
|
63
66
|
|
|
64
67
|
// Configurable
|
|
65
68
|
if (attribute.configurable) {
|
|
66
|
-
addImport('ConfigurableAttribute');
|
|
67
|
-
|
|
68
69
|
modifiers.push(
|
|
69
|
-
factory.createTypeReferenceNode(
|
|
70
|
+
factory.createTypeReferenceNode(
|
|
71
|
+
factory.createIdentifier(withAttributeNamespace('Configurable'))
|
|
72
|
+
)
|
|
70
73
|
);
|
|
71
74
|
}
|
|
72
75
|
|
|
73
76
|
// Custom field
|
|
74
77
|
if (attribute.customField) {
|
|
75
|
-
addImport('CustomField');
|
|
76
|
-
|
|
77
78
|
const customFieldUid = factory.createStringLiteral(attribute.customField);
|
|
78
79
|
const typeArguments = [customFieldUid];
|
|
79
80
|
|
|
@@ -82,17 +83,18 @@ const getAttributeModifiers = (attribute) => {
|
|
|
82
83
|
}
|
|
83
84
|
|
|
84
85
|
modifiers.push(
|
|
85
|
-
factory.createTypeReferenceNode(
|
|
86
|
+
factory.createTypeReferenceNode(
|
|
87
|
+
factory.createIdentifier(withAttributeNamespace('CustomField')),
|
|
88
|
+
typeArguments
|
|
89
|
+
)
|
|
86
90
|
);
|
|
87
91
|
}
|
|
88
92
|
|
|
89
93
|
// Plugin Options
|
|
90
94
|
if (!_.isEmpty(attribute.pluginOptions)) {
|
|
91
|
-
addImport('SetPluginOptions');
|
|
92
|
-
|
|
93
95
|
modifiers.push(
|
|
94
96
|
factory.createTypeReferenceNode(
|
|
95
|
-
factory.createIdentifier('SetPluginOptions'),
|
|
97
|
+
factory.createIdentifier(withAttributeNamespace('SetPluginOptions')),
|
|
96
98
|
// Transform the pluginOptions object into an object literal expression
|
|
97
99
|
[toTypeLiteral(attribute.pluginOptions)]
|
|
98
100
|
)
|
|
@@ -100,40 +102,64 @@ const getAttributeModifiers = (attribute) => {
|
|
|
100
102
|
}
|
|
101
103
|
|
|
102
104
|
// Min / Max
|
|
103
|
-
// TODO: Always provide a second type argument for min/max (ie: resolve the attribute scalar type with a `GetAttributeType<${mappers[attribute][0]}>` (useful for biginter (string values)))
|
|
104
105
|
if (!_.isNil(attribute.min) || !_.isNil(attribute.max)) {
|
|
105
|
-
addImport('SetMinMax');
|
|
106
|
-
|
|
107
106
|
const minMaxProperties = _.pick(['min', 'max'], attribute);
|
|
107
|
+
const { min, max } = minMaxProperties;
|
|
108
|
+
|
|
109
|
+
const typeofMin = typeof min;
|
|
110
|
+
const typeofMax = typeof max;
|
|
111
|
+
|
|
112
|
+
// Throws error if min/max exist but have different types to prevent unexpected behavior
|
|
113
|
+
if (min !== undefined && max !== undefined && typeofMin !== typeofMax) {
|
|
114
|
+
throw new Error('typeof min/max values mismatch');
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
let typeKeyword;
|
|
118
|
+
|
|
119
|
+
// use 'string'
|
|
120
|
+
if (typeofMin === 'string' || typeofMax === 'string') {
|
|
121
|
+
typeKeyword = ts.SyntaxKind.StringKeyword;
|
|
122
|
+
}
|
|
123
|
+
// use 'number'
|
|
124
|
+
else if (typeofMin === 'number' || typeofMax === 'number') {
|
|
125
|
+
typeKeyword = ts.SyntaxKind.NumberKeyword;
|
|
126
|
+
}
|
|
127
|
+
// invalid type
|
|
128
|
+
else {
|
|
129
|
+
throw new Error(
|
|
130
|
+
`Invalid data type for min/max options. Must be string, number or undefined, but found { min: ${min} (${typeofMin}), max: ${max} (${typeofMax}) }`
|
|
131
|
+
);
|
|
132
|
+
}
|
|
108
133
|
|
|
109
134
|
modifiers.push(
|
|
110
|
-
factory.createTypeReferenceNode(
|
|
111
|
-
|
|
112
|
-
|
|
135
|
+
factory.createTypeReferenceNode(
|
|
136
|
+
factory.createIdentifier(withAttributeNamespace('SetMinMax')),
|
|
137
|
+
[toTypeLiteral(minMaxProperties), factory.createKeywordTypeNode(typeKeyword)]
|
|
138
|
+
)
|
|
113
139
|
);
|
|
114
140
|
}
|
|
115
141
|
|
|
116
142
|
// Min length / Max length
|
|
117
143
|
if (!_.isNil(attribute.minLength) || !_.isNil(attribute.maxLength)) {
|
|
118
|
-
addImport('SetMinMaxLength');
|
|
119
|
-
|
|
120
144
|
const minMaxProperties = _.pick(['minLength', 'maxLength'], attribute);
|
|
121
145
|
|
|
122
146
|
modifiers.push(
|
|
123
|
-
factory.createTypeReferenceNode(
|
|
124
|
-
|
|
125
|
-
|
|
147
|
+
factory.createTypeReferenceNode(
|
|
148
|
+
factory.createIdentifier(withAttributeNamespace('SetMinMaxLength')),
|
|
149
|
+
[toTypeLiteral(minMaxProperties)]
|
|
150
|
+
)
|
|
126
151
|
);
|
|
127
152
|
}
|
|
128
153
|
|
|
129
154
|
// Default
|
|
130
155
|
if (!_.isNil(attribute.default)) {
|
|
131
|
-
addImport('DefaultTo');
|
|
132
|
-
|
|
133
156
|
const defaultLiteral = toTypeLiteral(attribute.default);
|
|
134
157
|
|
|
135
158
|
modifiers.push(
|
|
136
|
-
factory.createTypeReferenceNode(
|
|
159
|
+
factory.createTypeReferenceNode(
|
|
160
|
+
factory.createIdentifier(withAttributeNamespace('DefaultTo')),
|
|
161
|
+
[defaultLiteral]
|
|
162
|
+
)
|
|
137
163
|
);
|
|
138
164
|
}
|
|
139
165
|
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const schema = require('./schema');
|
|
4
|
+
const attributes = require('./attributes');
|
|
5
|
+
const mappers = require('./mappers');
|
|
6
|
+
const utils = require('./utils');
|
|
7
|
+
const imports = require('../imports');
|
|
8
|
+
|
|
9
|
+
module.exports = {
|
|
10
|
+
schema,
|
|
11
|
+
attributes,
|
|
12
|
+
mappers,
|
|
13
|
+
utils,
|
|
14
|
+
imports,
|
|
15
|
+
};
|
|
@@ -3,56 +3,56 @@
|
|
|
3
3
|
const ts = require('typescript');
|
|
4
4
|
const _ = require('lodash/fp');
|
|
5
5
|
|
|
6
|
-
const { toTypeLiteral } = require('./utils');
|
|
6
|
+
const { toTypeLiteral, withAttributeNamespace } = require('./utils');
|
|
7
7
|
|
|
8
8
|
const { factory } = ts;
|
|
9
9
|
|
|
10
10
|
module.exports = {
|
|
11
11
|
string() {
|
|
12
|
-
return ['
|
|
12
|
+
return [withAttributeNamespace('String')];
|
|
13
13
|
},
|
|
14
14
|
text() {
|
|
15
|
-
return ['
|
|
15
|
+
return [withAttributeNamespace('Text')];
|
|
16
16
|
},
|
|
17
17
|
richtext() {
|
|
18
|
-
return ['
|
|
18
|
+
return [withAttributeNamespace('RichText')];
|
|
19
19
|
},
|
|
20
20
|
password() {
|
|
21
|
-
return ['
|
|
21
|
+
return [withAttributeNamespace('Password')];
|
|
22
22
|
},
|
|
23
23
|
email() {
|
|
24
|
-
return ['
|
|
24
|
+
return [withAttributeNamespace('Email')];
|
|
25
25
|
},
|
|
26
26
|
date() {
|
|
27
|
-
return ['
|
|
27
|
+
return [withAttributeNamespace('Date')];
|
|
28
28
|
},
|
|
29
29
|
time() {
|
|
30
|
-
return ['
|
|
30
|
+
return [withAttributeNamespace('Time')];
|
|
31
31
|
},
|
|
32
32
|
datetime() {
|
|
33
|
-
return ['
|
|
33
|
+
return [withAttributeNamespace('DateTime')];
|
|
34
34
|
},
|
|
35
35
|
timestamp() {
|
|
36
|
-
return ['
|
|
36
|
+
return [withAttributeNamespace('Timestamp')];
|
|
37
37
|
},
|
|
38
38
|
integer() {
|
|
39
|
-
return ['
|
|
39
|
+
return [withAttributeNamespace('Integer')];
|
|
40
40
|
},
|
|
41
41
|
biginteger() {
|
|
42
|
-
return ['
|
|
42
|
+
return [withAttributeNamespace('BigInteger')];
|
|
43
43
|
},
|
|
44
44
|
float() {
|
|
45
|
-
return ['
|
|
45
|
+
return [withAttributeNamespace('Float')];
|
|
46
46
|
},
|
|
47
47
|
decimal() {
|
|
48
|
-
return ['
|
|
48
|
+
return [withAttributeNamespace('Decimal')];
|
|
49
49
|
},
|
|
50
50
|
uid({ attribute, uid }) {
|
|
51
51
|
const { targetField, options } = attribute;
|
|
52
52
|
|
|
53
53
|
// If there are no params to compute, then return the attribute type alone
|
|
54
54
|
if (targetField === undefined && options === undefined) {
|
|
55
|
-
return ['
|
|
55
|
+
return [withAttributeNamespace('UID')];
|
|
56
56
|
}
|
|
57
57
|
|
|
58
58
|
const params = [];
|
|
@@ -74,21 +74,24 @@ module.exports = {
|
|
|
74
74
|
params.push(toTypeLiteral(options));
|
|
75
75
|
}
|
|
76
76
|
|
|
77
|
-
return ['
|
|
77
|
+
return [withAttributeNamespace('UID'), params];
|
|
78
78
|
},
|
|
79
79
|
enumeration({ attribute }) {
|
|
80
80
|
const { enum: enumValues } = attribute;
|
|
81
81
|
|
|
82
|
-
return ['
|
|
82
|
+
return [withAttributeNamespace('Enumeration'), [toTypeLiteral(enumValues)]];
|
|
83
83
|
},
|
|
84
84
|
boolean() {
|
|
85
|
-
return ['
|
|
85
|
+
return [withAttributeNamespace('Boolean')];
|
|
86
86
|
},
|
|
87
87
|
json() {
|
|
88
|
-
return ['
|
|
88
|
+
return [withAttributeNamespace('JSON')];
|
|
89
|
+
},
|
|
90
|
+
blocks() {
|
|
91
|
+
return [withAttributeNamespace('Blocks')];
|
|
89
92
|
},
|
|
90
93
|
media() {
|
|
91
|
-
return ['
|
|
94
|
+
return [withAttributeNamespace('Media')];
|
|
92
95
|
},
|
|
93
96
|
relation({ uid, attribute }) {
|
|
94
97
|
const { relation, target } = attribute;
|
|
@@ -97,13 +100,13 @@ module.exports = {
|
|
|
97
100
|
|
|
98
101
|
if (isMorphRelation) {
|
|
99
102
|
return [
|
|
100
|
-
'
|
|
103
|
+
withAttributeNamespace('Relation'),
|
|
101
104
|
[factory.createStringLiteral(uid, true), factory.createStringLiteral(relation, true)],
|
|
102
105
|
];
|
|
103
106
|
}
|
|
104
107
|
|
|
105
108
|
return [
|
|
106
|
-
'
|
|
109
|
+
withAttributeNamespace('Relation'),
|
|
107
110
|
[
|
|
108
111
|
factory.createStringLiteral(uid, true),
|
|
109
112
|
factory.createStringLiteral(relation, true),
|
|
@@ -119,13 +122,13 @@ module.exports = {
|
|
|
119
122
|
params.push(factory.createTrue());
|
|
120
123
|
}
|
|
121
124
|
|
|
122
|
-
return ['
|
|
125
|
+
return [withAttributeNamespace('Component'), params];
|
|
123
126
|
},
|
|
124
127
|
dynamiczone({ attribute }) {
|
|
125
128
|
const componentsParam = factory.createTupleTypeNode(
|
|
126
129
|
attribute.components.map((component) => factory.createStringLiteral(component))
|
|
127
130
|
);
|
|
128
131
|
|
|
129
|
-
return ['
|
|
132
|
+
return [withAttributeNamespace('DynamicZone'), [componentsParam]];
|
|
130
133
|
},
|
|
131
134
|
};
|
|
@@ -4,9 +4,14 @@ const ts = require('typescript');
|
|
|
4
4
|
const { factory } = require('typescript');
|
|
5
5
|
const { isEmpty } = require('lodash/fp');
|
|
6
6
|
|
|
7
|
-
const {
|
|
7
|
+
const {
|
|
8
|
+
getSchemaExtendsTypeName,
|
|
9
|
+
getSchemaInterfaceName,
|
|
10
|
+
toTypeLiteral,
|
|
11
|
+
NAMESPACES,
|
|
12
|
+
} = require('./utils');
|
|
8
13
|
const attributeToPropertySignature = require('./attributes');
|
|
9
|
-
const { addImport } = require('
|
|
14
|
+
const { addImport } = require('../imports');
|
|
10
15
|
|
|
11
16
|
/**
|
|
12
17
|
* Generate a property signature for the schema's `attributes` field
|
|
@@ -51,11 +56,11 @@ const generateSchemaDefinition = (schema) => {
|
|
|
51
56
|
const interfaceName = getSchemaInterfaceName(uid);
|
|
52
57
|
const parentType = getSchemaExtendsTypeName(schema);
|
|
53
58
|
|
|
54
|
-
// Make sure the
|
|
55
|
-
addImport(
|
|
59
|
+
// Make sure the Schema namespace is imported
|
|
60
|
+
addImport(NAMESPACES.schema);
|
|
56
61
|
|
|
57
62
|
// Properties whose values can be mapped to a literal type expression
|
|
58
|
-
const literalPropertiesDefinitions = ['info', 'options', 'pluginOptions']
|
|
63
|
+
const literalPropertiesDefinitions = ['collectionName', 'info', 'options', 'pluginOptions']
|
|
59
64
|
// Ignore non-existent or empty declarations
|
|
60
65
|
.filter((key) => !isEmpty(schema[key]))
|
|
61
66
|
// Generate literal definition for each property
|
|
@@ -17,13 +17,10 @@ const {
|
|
|
17
17
|
propEq,
|
|
18
18
|
} = require('lodash/fp');
|
|
19
19
|
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
* @returns {object}
|
|
25
|
-
*/
|
|
26
|
-
const getAllStrapiSchemas = (strapi) => ({ ...strapi.contentTypes, ...strapi.components });
|
|
20
|
+
const NAMESPACES = {
|
|
21
|
+
schema: 'Schema',
|
|
22
|
+
attribute: 'Attribute',
|
|
23
|
+
};
|
|
27
24
|
|
|
28
25
|
/**
|
|
29
26
|
* Extract a valid interface name from a schema uid
|
|
@@ -58,7 +55,11 @@ const getSchemaModelType = (schema) => {
|
|
|
58
55
|
const getSchemaExtendsTypeName = (schema) => {
|
|
59
56
|
const base = getSchemaModelType(schema);
|
|
60
57
|
|
|
61
|
-
|
|
58
|
+
if (base === null) {
|
|
59
|
+
return null;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
return `${NAMESPACES.schema}.${upperFirst(base)}`;
|
|
62
63
|
};
|
|
63
64
|
|
|
64
65
|
/**
|
|
@@ -143,8 +144,26 @@ const getDefinitionAttributesCount = (definition) => {
|
|
|
143
144
|
return attributesNode.type.members.length;
|
|
144
145
|
};
|
|
145
146
|
|
|
147
|
+
/**
|
|
148
|
+
* Add the attribute namespace before the typename
|
|
149
|
+
*
|
|
150
|
+
* @param {string} typeName
|
|
151
|
+
* @returns {string}
|
|
152
|
+
*/
|
|
153
|
+
const withAttributeNamespace = (typeName) => `${NAMESPACES.attribute}.${typeName}`;
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Add the schema namespace before the typename
|
|
157
|
+
*
|
|
158
|
+
* @param {string} typeName
|
|
159
|
+
* @returns {string}
|
|
160
|
+
*/
|
|
161
|
+
const withSchemaNamespace = (typeName) => `${NAMESPACES.schema}.${typeName}`;
|
|
162
|
+
|
|
146
163
|
module.exports = {
|
|
147
|
-
|
|
164
|
+
NAMESPACES,
|
|
165
|
+
withAttributeNamespace,
|
|
166
|
+
withSchemaNamespace,
|
|
148
167
|
getSchemaInterfaceName,
|
|
149
168
|
getSchemaExtendsTypeName,
|
|
150
169
|
getSchemaModelType,
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const { factory } = require('typescript');
|
|
4
|
+
|
|
5
|
+
const { models } = require('../common');
|
|
6
|
+
const { emitDefinitions, format, generateSharedExtensionDefinition } = require('../utils');
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Generate type definitions for Strapi Components
|
|
10
|
+
*
|
|
11
|
+
* @param {object} [options]
|
|
12
|
+
* @param {object} options.strapi
|
|
13
|
+
* @param {object} options.logger
|
|
14
|
+
* @param {string} options.pwd
|
|
15
|
+
*/
|
|
16
|
+
const generateComponentsDefinitions = async (options = {}) => {
|
|
17
|
+
const { strapi } = options;
|
|
18
|
+
|
|
19
|
+
const { components } = strapi;
|
|
20
|
+
|
|
21
|
+
const componentsDefinitions = Object.values(components).map((contentType) => ({
|
|
22
|
+
uid: contentType.uid,
|
|
23
|
+
definition: models.schema.generateSchemaDefinition(contentType),
|
|
24
|
+
}));
|
|
25
|
+
|
|
26
|
+
const formattedSchemasDefinitions = componentsDefinitions.reduce((acc, def) => {
|
|
27
|
+
acc.push(
|
|
28
|
+
// Definition
|
|
29
|
+
def.definition,
|
|
30
|
+
|
|
31
|
+
// Add a newline between each interface declaration
|
|
32
|
+
factory.createIdentifier('\n')
|
|
33
|
+
);
|
|
34
|
+
|
|
35
|
+
return acc;
|
|
36
|
+
}, []);
|
|
37
|
+
|
|
38
|
+
const allDefinitions = [
|
|
39
|
+
// Imports
|
|
40
|
+
...models.imports.generateImportDefinition(),
|
|
41
|
+
|
|
42
|
+
// Add a newline after the import statement
|
|
43
|
+
factory.createIdentifier('\n'),
|
|
44
|
+
|
|
45
|
+
// Schemas
|
|
46
|
+
...formattedSchemasDefinitions,
|
|
47
|
+
|
|
48
|
+
// Global
|
|
49
|
+
generateSharedExtensionDefinition('Components', componentsDefinitions),
|
|
50
|
+
];
|
|
51
|
+
|
|
52
|
+
const output = emitDefinitions(allDefinitions);
|
|
53
|
+
const formattedOutput = await format(output);
|
|
54
|
+
|
|
55
|
+
return { output: formattedOutput, stats: {} };
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
module.exports = generateComponentsDefinitions;
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const { factory } = require('typescript');
|
|
4
|
+
|
|
5
|
+
const { models } = require('../common');
|
|
6
|
+
const { emitDefinitions, format, generateSharedExtensionDefinition } = require('../utils');
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Generate type definitions for Strapi Content-Types
|
|
10
|
+
*
|
|
11
|
+
* @param {object} [options]
|
|
12
|
+
* @param {object} options.strapi
|
|
13
|
+
* @param {object} options.logger
|
|
14
|
+
* @param {string} options.pwd
|
|
15
|
+
*/
|
|
16
|
+
const generateContentTypesDefinitions = async (options = {}) => {
|
|
17
|
+
const { strapi } = options;
|
|
18
|
+
|
|
19
|
+
const { contentTypes } = strapi;
|
|
20
|
+
|
|
21
|
+
const contentTypesDefinitions = Object.values(contentTypes).map((contentType) => ({
|
|
22
|
+
uid: contentType.uid,
|
|
23
|
+
definition: models.schema.generateSchemaDefinition(contentType),
|
|
24
|
+
}));
|
|
25
|
+
|
|
26
|
+
const formattedSchemasDefinitions = contentTypesDefinitions.reduce((acc, def) => {
|
|
27
|
+
acc.push(
|
|
28
|
+
// Definition
|
|
29
|
+
def.definition,
|
|
30
|
+
|
|
31
|
+
// Add a newline between each interface declaration
|
|
32
|
+
factory.createIdentifier('\n')
|
|
33
|
+
);
|
|
34
|
+
|
|
35
|
+
return acc;
|
|
36
|
+
}, []);
|
|
37
|
+
|
|
38
|
+
const allDefinitions = [
|
|
39
|
+
// Imports
|
|
40
|
+
...models.imports.generateImportDefinition(),
|
|
41
|
+
|
|
42
|
+
// Add a newline after the import statement
|
|
43
|
+
factory.createIdentifier('\n'),
|
|
44
|
+
|
|
45
|
+
// Schemas
|
|
46
|
+
...formattedSchemasDefinitions,
|
|
47
|
+
|
|
48
|
+
// Global
|
|
49
|
+
generateSharedExtensionDefinition('ContentTypes', contentTypesDefinitions),
|
|
50
|
+
];
|
|
51
|
+
|
|
52
|
+
const output = emitDefinitions(allDefinitions);
|
|
53
|
+
const formattedOutput = await format(output);
|
|
54
|
+
|
|
55
|
+
return { output: formattedOutput, stats: {} };
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
module.exports = generateContentTypesDefinitions;
|