@@ -8,21 +8,35 @@ import {
88 ParsedAppSyncModelConfig ,
99 RawAppSyncModelConfig ,
1010 CodeGenEnum ,
11+ CodeGenUnion ,
12+ CodeGenInterface ,
1113} from './appsync-visitor' ;
1214import { METADATA_SCALAR_MAP } from '../scalars' ;
1315export type JSONSchema = {
1416 models : JSONSchemaModels ;
1517 enums : JSONSchemaEnums ;
1618 nonModels : JSONSchemaTypes ;
19+ interfaces : JSONSchemaInterfaces ;
20+ unions : JSONSchemaUnions ;
1721 version : string ;
1822 codegenVersion : string ;
1923} ;
2024export type JSONSchemaModels = Record < string , JSONSchemaModel > ;
2125export type JSONSchemaTypes = Record < string , JSONSchemaNonModel > ;
26+ export type JSONSchemaInterfaces = Record < string , JSONSchemaInterface > ;
27+ export type JSONSchemaUnions = Record < string , JSONSchemaUnion > ;
2228export type JSONSchemaNonModel = {
2329 name : string ;
2430 fields : JSONModelFields ;
2531} ;
32+ export type JSONSchemaInterface = {
33+ name : string ;
34+ fields : JSONModelFields ;
35+ } ;
36+ export type JSONSchemaUnion = {
37+ name : string ;
38+ types : JSONModelFieldType [ ] ;
39+ } ;
2640type JSONSchemaModel = {
2741 name : string ;
2842 attributes ?: JSONModelAttributes ;
@@ -58,7 +72,7 @@ type AssociationBelongsTo = AssociationBaseType & {
5872
5973type AssociationType = AssociationHasMany | AssociationHasOne | AssociationBelongsTo ;
6074
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 } ;
6276type JSONModelField = {
6377 name : string ;
6478 type : JSONModelFieldType ;
@@ -147,7 +161,27 @@ export class AppSyncJSONVisitor<
147161 }
148162
149163 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+ }>;` ;
151185 }
152186
153187 protected generateJSONMetadata ( ) : string {
@@ -160,6 +194,8 @@ export class AppSyncJSONVisitor<
160194 models : { } ,
161195 enums : { } ,
162196 nonModels : { } ,
197+ interfaces : { } ,
198+ unions : { } ,
163199 // This is hard-coded for the schema version purpose instead of codegen version
164200 // To avoid the failure of validation method checkCodegenSchema in JS Datastore
165201 // The hard code is starting from amplify codegen major version 4
@@ -175,11 +211,19 @@ export class AppSyncJSONVisitor<
175211 return { ...acc , [ nonModel . name ] : this . generateNonModelMetadata ( nonModel ) } ;
176212 } , { } ) ;
177213
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+
178222 const enums = Object . values ( this . enumMap ) . reduce ( ( acc , enumObj ) => {
179223 const enumV = this . generateEnumMetadata ( enumObj ) ;
180224 return { ...acc , [ this . getEnumName ( enumObj ) ] : enumV } ;
181225 } , { } ) ;
182- return { ...result , models, nonModels : nonModels , enums } ;
226+ return { ...result , models, nonModels : nonModels , enums, interfaces , unions } ;
183227 }
184228
185229 private getFieldAssociation ( field : CodeGenField ) : AssociationType | void {
@@ -229,39 +273,58 @@ export class AppSyncJSONVisitor<
229273 private generateNonModelMetadata ( nonModel : CodeGenModel ) : JSONSchemaNonModel {
230274 return {
231275 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+ }
244279
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+ }
248286
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 ) )
256291 } ;
257292 }
293+
258294 private generateEnumMetadata ( enumObj : CodeGenEnum ) : JSONSchemaEnum {
259295 return {
260296 name : enumObj . name ,
261297 values : Object . values ( enumObj . values ) ,
262298 } ;
263299 }
264300
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+
265328 private getType ( gqlType : string ) : JSONModelFieldType {
266329 // Todo: Handle unlisted scalars
267330 if ( gqlType in METADATA_SCALAR_MAP ) {
@@ -273,6 +336,12 @@ export class AppSyncJSONVisitor<
273336 if ( gqlType in this . nonModelMap ) {
274337 return { nonModel : gqlType } ;
275338 }
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+ }
276345 if ( gqlType in this . modelMap ) {
277346 return { model : gqlType } ;
278347 }
0 commit comments