@@ -8,21 +8,35 @@ import {
8
8
ParsedAppSyncModelConfig ,
9
9
RawAppSyncModelConfig ,
10
10
CodeGenEnum ,
11
+ CodeGenUnion ,
12
+ CodeGenInterface ,
11
13
} from './appsync-visitor' ;
12
14
import { METADATA_SCALAR_MAP } from '../scalars' ;
13
15
export type JSONSchema = {
14
16
models : JSONSchemaModels ;
15
17
enums : JSONSchemaEnums ;
16
18
nonModels : JSONSchemaTypes ;
19
+ interfaces : JSONSchemaInterfaces ;
20
+ unions : JSONSchemaUnions ;
17
21
version : string ;
18
22
codegenVersion : string ;
19
23
} ;
20
24
export type JSONSchemaModels = Record < string , JSONSchemaModel > ;
21
25
export type JSONSchemaTypes = Record < string , JSONSchemaNonModel > ;
26
+ export type JSONSchemaInterfaces = Record < string , JSONSchemaInterface > ;
27
+ export type JSONSchemaUnions = Record < string , JSONSchemaUnion > ;
22
28
export type JSONSchemaNonModel = {
23
29
name : string ;
24
30
fields : JSONModelFields ;
25
31
} ;
32
+ export type JSONSchemaInterface = {
33
+ name : string ;
34
+ fields : JSONModelFields ;
35
+ } ;
36
+ export type JSONSchemaUnion = {
37
+ name : string ;
38
+ types : JSONModelFieldType [ ] ;
39
+ } ;
26
40
type JSONSchemaModel = {
27
41
name : string ;
28
42
attributes ?: JSONModelAttributes ;
@@ -58,7 +72,7 @@ type AssociationBelongsTo = AssociationBaseType & {
58
72
59
73
type AssociationType = AssociationHasMany | AssociationHasOne | AssociationBelongsTo ;
60
74
61
- type JSONModelFieldType = keyof typeof METADATA_SCALAR_MAP | { model : string } | { enum : string } | { nonModel : string } ;
75
+ type JSONModelFieldType = keyof typeof METADATA_SCALAR_MAP | { model : string } | { enum : string } | { nonModel : string } | { interface : string } | { union : string } ;
62
76
type JSONModelField = {
63
77
name : string ;
64
78
type : JSONModelFieldType ;
@@ -147,7 +161,27 @@ export class AppSyncJSONVisitor<
147
161
}
148
162
149
163
protected generateTypeDeclaration ( ) {
150
- return [ "import { Schema } from '@aws-amplify/datastore';" , '' , 'export declare const schema: Schema;' ] . join ( '\n' ) ;
164
+ return `import type { Schema, SchemaNonModel, ModelField, ModelFieldType } from '@aws-amplify/datastore';
165
+
166
+ type Replace<T, R> = Omit<T, keyof R> & R;
167
+ type WithFields = { fields: Record<string, ModelField> };
168
+ type SchemaTypes = Record<string, WithFields>;
169
+
170
+ export type ExtendModelFieldType = ModelField['type'] | { interface: string } | { union: string };
171
+ export type ExtendModelField = Replace<ModelField, { type: ExtendModelFieldType }>;
172
+ export type ExtendType<T extends WithFields> = Replace<T, { fields: Record<string, ExtendModelField> }>
173
+ export type ExtendFields<Types extends SchemaTypes | undefined> = {
174
+ [TypeName in keyof Types]: ExtendType<Types[TypeName]>
175
+ }
176
+
177
+ type ExtendFieldsAll<T> = {
178
+ [K in keyof T]: T[K] extends SchemaTypes | undefined ? ExtendFields<T[K]> : T[K];
179
+ };
180
+
181
+ export declare const schema: ExtendFieldsAll<Schema & {
182
+ interfaces: Schema['nonModels'];
183
+ unions?: Record<string, {name: string, types: ExtendModelFieldType[]}>;
184
+ }>;` ;
151
185
}
152
186
153
187
protected generateJSONMetadata ( ) : string {
@@ -160,6 +194,8 @@ export class AppSyncJSONVisitor<
160
194
models : { } ,
161
195
enums : { } ,
162
196
nonModels : { } ,
197
+ interfaces : { } ,
198
+ unions : { } ,
163
199
// This is hard-coded for the schema version purpose instead of codegen version
164
200
// To avoid the failure of validation method checkCodegenSchema in JS Datastore
165
201
// The hard code is starting from amplify codegen major version 4
@@ -175,11 +211,19 @@ export class AppSyncJSONVisitor<
175
211
return { ...acc , [ nonModel . name ] : this . generateNonModelMetadata ( nonModel ) } ;
176
212
} , { } ) ;
177
213
214
+ const interfaces = Object . values ( this . getSelectedInterfaces ( ) ) . reduce ( ( acc , codegenInterface : CodeGenInterface ) => {
215
+ return { ...acc , [ codegenInterface . name ] : this . generateInterfaceMetadata ( codegenInterface ) } ;
216
+ } , { } ) ;
217
+
218
+ const unions = Object . values ( this . getSelectedUnions ( ) ) . reduce ( ( acc , union : CodeGenUnion ) => {
219
+ return { ...acc , [ union . name ] : this . generateUnionMetadata ( union ) } ;
220
+ } , { } ) ;
221
+
178
222
const enums = Object . values ( this . enumMap ) . reduce ( ( acc , enumObj ) => {
179
223
const enumV = this . generateEnumMetadata ( enumObj ) ;
180
224
return { ...acc , [ this . getEnumName ( enumObj ) ] : enumV } ;
181
225
} , { } ) ;
182
- return { ...result , models, nonModels : nonModels , enums } ;
226
+ return { ...result , models, nonModels : nonModels , enums, interfaces , unions } ;
183
227
}
184
228
185
229
private getFieldAssociation ( field : CodeGenField ) : AssociationType | void {
@@ -229,39 +273,58 @@ export class AppSyncJSONVisitor<
229
273
private generateNonModelMetadata ( nonModel : CodeGenModel ) : JSONSchemaNonModel {
230
274
return {
231
275
name : this . getModelName ( nonModel ) ,
232
- fields : nonModel . fields . reduce ( ( acc : JSONModelFields , field : CodeGenField ) => {
233
- const fieldMeta : JSONModelField = {
234
- name : this . getFieldName ( field ) ,
235
- isArray : field . isList ,
236
- type : this . getType ( field . type ) ,
237
- isRequired : ! field . isNullable ,
238
- attributes : [ ] ,
239
- } ;
240
-
241
- if ( field . isListNullable !== undefined ) {
242
- fieldMeta . isArrayNullable = field . isListNullable ;
243
- }
276
+ fields : this . generateFieldsMetadata ( nonModel . fields )
277
+ } ;
278
+ }
244
279
245
- if ( field . isReadOnly !== undefined ) {
246
- fieldMeta . isReadOnly = field . isReadOnly ;
247
- }
280
+ private generateInterfaceMetadata ( codeGenInterface : CodeGenInterface ) : JSONSchemaInterface {
281
+ return {
282
+ name : codeGenInterface . name ,
283
+ fields : this . generateFieldsMetadata ( codeGenInterface . fields ) ,
284
+ } ;
285
+ }
248
286
249
- const association : AssociationType | void = this . getFieldAssociation ( field ) ;
250
- if ( association ) {
251
- fieldMeta . association = association ;
252
- }
253
- acc [ fieldMeta . name ] = fieldMeta ;
254
- return acc ;
255
- } , { } ) ,
287
+ private generateUnionMetadata ( codeGenUnion : CodeGenUnion ) : JSONSchemaUnion {
288
+ return {
289
+ name : codeGenUnion . name ,
290
+ types : codeGenUnion . typeNames . map ( t => this . getType ( t ) )
256
291
} ;
257
292
}
293
+
258
294
private generateEnumMetadata ( enumObj : CodeGenEnum ) : JSONSchemaEnum {
259
295
return {
260
296
name : enumObj . name ,
261
297
values : Object . values ( enumObj . values ) ,
262
298
} ;
263
299
}
264
300
301
+ private generateFieldsMetadata ( fields : CodeGenField [ ] ) : JSONModelFields {
302
+ return fields . reduce ( ( acc : JSONModelFields , field : CodeGenField ) => {
303
+ const fieldMeta : JSONModelField = {
304
+ name : this . getFieldName ( field ) ,
305
+ isArray : field . isList ,
306
+ type : this . getType ( field . type ) ,
307
+ isRequired : ! field . isNullable ,
308
+ attributes : [ ] ,
309
+ } ;
310
+
311
+ if ( field . isListNullable !== undefined ) {
312
+ fieldMeta . isArrayNullable = field . isListNullable ;
313
+ }
314
+
315
+ if ( field . isReadOnly !== undefined ) {
316
+ fieldMeta . isReadOnly = field . isReadOnly ;
317
+ }
318
+
319
+ const association : AssociationType | void = this . getFieldAssociation ( field ) ;
320
+ if ( association ) {
321
+ fieldMeta . association = association ;
322
+ }
323
+ acc [ fieldMeta . name ] = fieldMeta ;
324
+ return acc ;
325
+ } , { } )
326
+ }
327
+
265
328
private getType ( gqlType : string ) : JSONModelFieldType {
266
329
// Todo: Handle unlisted scalars
267
330
if ( gqlType in METADATA_SCALAR_MAP ) {
@@ -273,6 +336,12 @@ export class AppSyncJSONVisitor<
273
336
if ( gqlType in this . nonModelMap ) {
274
337
return { nonModel : gqlType } ;
275
338
}
339
+ if ( gqlType in this . interfaceMap ) {
340
+ return { interface : this . interfaceMap [ gqlType ] . name } ;
341
+ }
342
+ if ( gqlType in this . unionMap ) {
343
+ return { union : this . unionMap [ gqlType ] . name } ;
344
+ }
276
345
if ( gqlType in this . modelMap ) {
277
346
return { model : gqlType } ;
278
347
}
0 commit comments