@strapi/typescript-utils 0.0.0-next.e41415e8ff5f565ff959667d5c5ba4f20bee013c → 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.
@@ -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
- compilerOptions: {
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, { watch = false, configOptions = {} } = {}) => {
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
- compiler.run(configPath, configOptions);
9
+ compilers.basic.run(configPath, configOptions);
14
10
  };
@@ -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: merge(options, configOptions.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
  },
@@ -1,9 +1,7 @@
1
1
  'use strict';
2
2
 
3
3
  const basic = require('./basic');
4
- const watch = require('./watch');
5
4
 
6
5
  module.exports = {
7
6
  basic,
8
- watch,
9
7
  };
@@ -22,11 +22,13 @@ module.exports = {
22
22
  factory.createImportSpecifier(false, undefined, factory.createIdentifier(key))
23
23
  );
24
24
 
25
- return factory.createImportDeclaration(
26
- undefined,
27
- factory.createImportClause(false, undefined, factory.createNamedImports(formattedImports)),
28
- factory.createStringLiteral('@strapi/strapi'),
29
- undefined
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
  };
@@ -0,0 +1,9 @@
1
+ 'use strict';
2
+
3
+ const models = require('./models');
4
+ const imports = require('./imports');
5
+
6
+ module.exports = {
7
+ models,
8
+ imports,
9
+ };
@@ -1,12 +1,14 @@
1
1
  'use strict';
2
2
 
3
- const { factory } = require('typescript');
3
+ const ts = require('typescript');
4
4
  const _ = require('lodash/fp');
5
5
 
6
- const { addImport } = require('./imports');
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
- addImport(attributeType);
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
- addImport('RequiredAttribute');
46
-
47
- modifiers.push(factory.createTypeReferenceNode(factory.createIdentifier('RequiredAttribute')));
48
+ modifiers.push(
49
+ factory.createTypeReferenceNode(factory.createIdentifier(withAttributeNamespace('Required')))
50
+ );
48
51
  }
49
52
 
50
53
  // Private
51
54
  if (attribute.private) {
52
- addImport('PrivateAttribute');
53
-
54
- modifiers.push(factory.createTypeReferenceNode(factory.createIdentifier('PrivateAttribute')));
55
+ modifiers.push(
56
+ factory.createTypeReferenceNode(factory.createIdentifier(withAttributeNamespace('Private')))
57
+ );
55
58
  }
56
59
 
57
60
  // Unique
58
61
  if (attribute.unique) {
59
- addImport('UniqueAttribute');
60
-
61
- modifiers.push(factory.createTypeReferenceNode(factory.createIdentifier('UniqueAttribute')));
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(factory.createIdentifier('ConfigurableAttribute'))
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(factory.createIdentifier('CustomField'), typeArguments)
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(factory.createIdentifier('SetMinMax'), [
111
- toTypeLiteral(minMaxProperties),
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(factory.createIdentifier('SetMinMaxLength'), [
124
- toTypeLiteral(minMaxProperties),
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(factory.createIdentifier('DefaultTo'), [defaultLiteral])
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 ['StringAttribute'];
12
+ return [withAttributeNamespace('String')];
13
13
  },
14
14
  text() {
15
- return ['TextAttribute'];
15
+ return [withAttributeNamespace('Text')];
16
16
  },
17
17
  richtext() {
18
- return ['RichTextAttribute'];
18
+ return [withAttributeNamespace('RichText')];
19
19
  },
20
20
  password() {
21
- return ['PasswordAttribute'];
21
+ return [withAttributeNamespace('Password')];
22
22
  },
23
23
  email() {
24
- return ['EmailAttribute'];
24
+ return [withAttributeNamespace('Email')];
25
25
  },
26
26
  date() {
27
- return ['DateAttribute'];
27
+ return [withAttributeNamespace('Date')];
28
28
  },
29
29
  time() {
30
- return ['TimeAttribute'];
30
+ return [withAttributeNamespace('Time')];
31
31
  },
32
32
  datetime() {
33
- return ['DateTimeAttribute'];
33
+ return [withAttributeNamespace('DateTime')];
34
34
  },
35
35
  timestamp() {
36
- return ['TimestampAttribute'];
36
+ return [withAttributeNamespace('Timestamp')];
37
37
  },
38
38
  integer() {
39
- return ['IntegerAttribute'];
39
+ return [withAttributeNamespace('Integer')];
40
40
  },
41
41
  biginteger() {
42
- return ['BigIntegerAttribute'];
42
+ return [withAttributeNamespace('BigInteger')];
43
43
  },
44
44
  float() {
45
- return ['FloatAttribute'];
45
+ return [withAttributeNamespace('Float')];
46
46
  },
47
47
  decimal() {
48
- return ['DecimalAttribute'];
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 ['UIDAttribute'];
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 ['UIDAttribute', params];
77
+ return [withAttributeNamespace('UID'), params];
78
78
  },
79
79
  enumeration({ attribute }) {
80
80
  const { enum: enumValues } = attribute;
81
81
 
82
- return ['EnumerationAttribute', [toTypeLiteral(enumValues)]];
82
+ return [withAttributeNamespace('Enumeration'), [toTypeLiteral(enumValues)]];
83
83
  },
84
84
  boolean() {
85
- return ['BooleanAttribute'];
85
+ return [withAttributeNamespace('Boolean')];
86
86
  },
87
87
  json() {
88
- return ['JSONAttribute'];
88
+ return [withAttributeNamespace('JSON')];
89
+ },
90
+ blocks() {
91
+ return [withAttributeNamespace('Blocks')];
89
92
  },
90
93
  media() {
91
- return ['MediaAttribute'];
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
- 'RelationAttribute',
103
+ withAttributeNamespace('Relation'),
101
104
  [factory.createStringLiteral(uid, true), factory.createStringLiteral(relation, true)],
102
105
  ];
103
106
  }
104
107
 
105
108
  return [
106
- 'RelationAttribute',
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 ['ComponentAttribute', params];
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 ['DynamicZoneAttribute', [componentsParam]];
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 { getSchemaExtendsTypeName, getSchemaInterfaceName, toTypeLiteral } = require('./utils');
7
+ const {
8
+ getSchemaExtendsTypeName,
9
+ getSchemaInterfaceName,
10
+ toTypeLiteral,
11
+ NAMESPACES,
12
+ } = require('./utils');
8
13
  const attributeToPropertySignature = require('./attributes');
9
- const { addImport } = require('./imports');
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 extended interface are imported
55
- addImport(parentType);
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
- * Get all components and content-types in a Strapi application
22
- *
23
- * @param {Strapi} strapi
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
- return `${upperFirst(base)}Schema`;
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
- getAllStrapiSchemas,
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,6 @@
1
+ 'use strict';
2
+
3
+ const TYPES_ROOT_DIR = 'types';
4
+ const GENERATED_OUT_DIR = 'generated';
5
+
6
+ module.exports = { GENERATED_OUT_DIR, TYPES_ROOT_DIR };
@@ -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;