my-appsync-repo/ ├── backend/ │ ├── schema/ │ │ └── schema.graphql │ ├── resolvers/ │ │ ├── Query/ │ │ │ ├── getItem.js │ │ │ └── listItems.js │ │ ├── Mutation/ │ │ │ ├── createItem.js │ │ │ └── updateItem.js │ │ └── pipelines/ │ ├── datasources/ │ │ └── datasources.json │ └── functions/ │ └── auth.js ├── infrastructure/ │ ├── appsync-stack.ts (CDK) │ └── config/ ├── tests/ │ ├── unit/ │ └── integration/ ├── scripts/ │ └── deploy.sh └── README.md This is the heart of your API. It defines types, queries, mutations, and subscriptions. Keep it in a single file or split it using #import directives. Example:

import * as appsync from 'aws-cdk-lib/aws-appsync'; import * as dynamodb from 'aws-cdk-lib/aws-dynamodb'; const api = new appsync.GraphqlApi(this, 'Api', { name: 'MyAPI', schema: appsync.Schema.fromAsset('backend/schema/schema.graphql'), authorizationConfig: { defaultAuthorization: { authorizationType: appsync.AuthorizationType.API_KEY } }, });

const table = new dynamodb.Table(this, 'ItemsTable', { ... }); const dataSource = api.addDynamoDbDataSource('ItemsDS', table);

// getItem.test.js import { request } from './getItem'; test('request includes user ID from identity', () => { const ctx = { args: { id: '123' }, identity: { claims: { sub: 'user1' } } }; expect(request(ctx).key.userId).toBe('user1'); }); Deploy your API to a test environment and run real queries using aws-appsync or Apollo Client. End-to-End Tests Test the full flow: mutation → subscription → query. CI/CD Pipeline for Your AppSync Repo Your pipeline should automate every step from commit to production. Here is a GitHub Actions workflow for an AppSync repo:

import { util } from '@aws-appsync/utils'; export function request(ctx) { const userId = ctx.identity.claims.sub; return { operation: 'GetItem', key: { id: ctx.args.id, userId } }; } Store subscription resolvers separately. Use @aws_subscribe directives in your schema to link mutations to subscriptions. Your repo should include directives. Testing Your AppSync Repo An AppSync repo without tests is risky. Implement three layers: Unit Tests (Jest + @aws-appsync/utils ) Test resolver logic without AWS infrastructure.

Appsync — Repo

my-appsync-repo/ ├── backend/ │ ├── schema/ │ │ └── schema.graphql │ ├── resolvers/ │ │ ├── Query/ │ │ │ ├── getItem.js │ │ │ └── listItems.js │ │ ├── Mutation/ │ │ │ ├── createItem.js │ │ │ └── updateItem.js │ │ └── pipelines/ │ ├── datasources/ │ │ └── datasources.json │ └── functions/ │ └── auth.js ├── infrastructure/ │ ├── appsync-stack.ts (CDK) │ └── config/ ├── tests/ │ ├── unit/ │ └── integration/ ├── scripts/ │ └── deploy.sh └── README.md This is the heart of your API. It defines types, queries, mutations, and subscriptions. Keep it in a single file or split it using #import directives. Example:

import * as appsync from 'aws-cdk-lib/aws-appsync'; import * as dynamodb from 'aws-cdk-lib/aws-dynamodb'; const api = new appsync.GraphqlApi(this, 'Api', { name: 'MyAPI', schema: appsync.Schema.fromAsset('backend/schema/schema.graphql'), authorizationConfig: { defaultAuthorization: { authorizationType: appsync.AuthorizationType.API_KEY } }, }); appsync repo

const table = new dynamodb.Table(this, 'ItemsTable', { ... }); const dataSource = api.addDynamoDbDataSource('ItemsDS', table); CI/CD Pipeline for Your AppSync Repo Your pipeline

// getItem.test.js import { request } from './getItem'; test('request includes user ID from identity', () => { const ctx = { args: { id: '123' }, identity: { claims: { sub: 'user1' } } }; expect(request(ctx).key.userId).toBe('user1'); }); Deploy your API to a test environment and run real queries using aws-appsync or Apollo Client. End-to-End Tests Test the full flow: mutation → subscription → query. CI/CD Pipeline for Your AppSync Repo Your pipeline should automate every step from commit to production. Here is a GitHub Actions workflow for an AppSync repo: return { operation: 'GetItem'

import { util } from '@aws-appsync/utils'; export function request(ctx) { const userId = ctx.identity.claims.sub; return { operation: 'GetItem', key: { id: ctx.args.id, userId } }; } Store subscription resolvers separately. Use @aws_subscribe directives in your schema to link mutations to subscriptions. Your repo should include directives. Testing Your AppSync Repo An AppSync repo without tests is risky. Implement three layers: Unit Tests (Jest + @aws-appsync/utils ) Test resolver logic without AWS infrastructure.