Serverless Framework'ten AWS CDK'ya Geçiş: Bölüm 2 - Lambda ve API Gateway
Lambda fonksiyonları ve API Gateway'i Serverless Framework'ten CDK'ya taşıma. Pratik örnekler ve gotcha'lar.
CDK migration'ımızın 1. haftası. Karar verilmişti, bütçe onaylanmıştı ve 12 kişilik ekibim beklenti içinde bana bakıyordu. "Peki, nereden başlıyoruz?"
CDK'yı kişisel projelerde kullanmıştım, ama onu 4 environment'ta 12 developer ile 47 Lambda fonksiyonunu handle edecek şekilde scale etmek? Bu farklıydı. Sadece benim için değil, tüm ekip için çalışacak yapı, kurallar ve pattern'lere ihtiyacımız vardı.
Bu, 12 mühendisinin birbirlerinin ayağına basmadan paralel çalışmasını sağlayan, Serverless Framework'teki tanıdık pattern'leri koruyan ve 2.8M$ ARR platformumuzun temeli haline gelen bir CDK proje yapısını nasıl tasarladığımızın hikayesi.
Seri Navigasyonu:
- Bölüm 1: Neden Geçiş Yapalım?
- Bölüm 2: CDK Environment Kurulumu (bu yazı)
- Bölüm 3: Lambda Fonksiyonları ve API Gateway Migration
- Bölüm 4: Database Kaynakları ve Environment Yönetimi
- Bölüm 5: Authentication, Authorization ve IAM
- Bölüm 6: Migration Stratejileri ve Best Practice'ler
Gerçekten Scale Olan Proje Yapısı#
İlk denememiz felaketti. Naif bir şekilde basit CDK tutorial yapısını kopyaladım ve bir hafta içinde merge conflict'leri, belirsiz ownership ve "Bu dosya nereye gider?" diye soran kafası karışmış mühendisler vardı.
İşte kaostan düzene evrim:
# Serverless Framework Yapısı
my-service/
├── serverless.yml
├── package.json
├── src/
│ └── handlers/
│ ├── users.js
│ └── products.js
├── resources/
│ └── dynamodb-tables.yml
└── config/
├── dev.yml
└── prod.yml
# CDK Yapısı (3 başarısız denemeden sonra)
my-service/
├── cdk.json # CDK app konfigürasyonu
├── package.json
├── bin/
│ └── my-service.ts # Tek entry point (zor yoldan öğrendik)
├── lib/
│ ├── stacks/ # Domain'e göre stack tanımları
│ │ ├── api-stack.ts # API Gateway + Lambda fonksiyonları
│ │ ├── data-stack.ts # DynamoDB tabloları (stateful)
│ │ └── auth-stack.ts # Cognito + auth logiği
│ ├── constructs/ # Reusable pattern'ler (sihirli formülümüz)
│ │ ├── production-lambda.ts # 376 fonksiyon bunu kullanıyor
│ │ ├── api-with-auth.ts # Her API bu pattern'i takip ediyor
│ │ └── monitored-table.ts # Alarm'lı DynamoDB
│ └── config/ # Environment-spesifik config'ler
│ ├── development.ts
│ ├── staging.ts
│ └── production.ts
├── src/
│ └── handlers/ # Lambda kodu (tanıdık lokasyon)
│ ├── users/ # Domain'e göre gruplandırılmış
│ │ ├── create.ts
│ │ ├── update.ts
│ │ └── list.ts
│ └── products/
│ ├── catalog.ts
│ └── inventory.ts
└── test/
├── unit/ # Handler unit testleri
├── integration/ # API integration testleri
└── infrastructure/ # CDK stack testleri
Anahtar anlayış: Domain-driven organizasyon, 12 mühendis paralel çalışırken merge conflict'leri önlüyor.
CDK Projenizi Initialize Etme#
Önce prerequisite'lerin olduğundan emin olun:
# AWS CDK CLI'yi global olarak yükle
npm install -g aws-cdk@2
# Kurulumu doğrula
cdk --version # 2.x.x göstermeli
# AWS credential'larını configure et (henüz yapılmadıysa)
aws configure
Şimdi projenizi oluşturun:
# Proje dizini oluştur
mkdir my-serverless-api && cd my-serverless-api
# TypeScript ile CDK initialize et
cdk init app --language typescript
# Lambda-spesifik dependency'leri yükle
npm install @aws-cdk/aws-lambda-nodejs-alpha @types/aws-lambda
# Development araçlarını yükle
npm install --save-dev esbuild @types/node ts-node
Lambda Development için TypeScript Konfigürasyonu#
CDK temel bir tsconfig.json
oluşturur. Serverless development için optimize edelim:
{
"compilerOptions": {
"target": "ES2022",
"module": "commonjs",
"lib": ["ES2022"],
"declaration": true,
"strict": true,
"noImplicitAny": true,
"strictNullChecks": true,
"noImplicitThis": true,
"alwaysStrict": true,
"esModuleInterop": true,
"noUnusedLocals": true,
"noUnusedParameters": true,
"noImplicitReturns": true,
"noFallthroughCasesInSwitch": true,
"inlineSourceMap": true,
"inlineSources": true,
"experimentalDecorators": true,
"strictPropertyInitialization": false,
"skipLibCheck": true,
"resolveJsonModule": true,
"outDir": "./dist",
"rootDir": "./",
"baseUrl": "./",
"paths": {
"@handlers/*": ["src/handlers/*"],
"@libs/*": ["src/libs/*"],
"@constructs/*": ["lib/constructs/*"]
}
},
"include": [
"bin/**/*",
"lib/**/*",
"src/**/*",
"test/**/*"
],
"exclude": [
"cdk.out",
"node_modules"
]
}
Environment Configuration Yönetimi#
Serverless Framework environment-spesifik konfigürasyon için YAML dosyaları kullanır. TypeScript tabanlı bir eşdeğer oluşturalım:
// lib/config/environment.ts
export interface EnvironmentConfig {
stage: string;
region: string;
account: string;
api: {
throttling: {
rateLimit: number;
burstLimit: number;
};
cors: {
origins: string[];
credentials: boolean;
};
};
lambda: {
memorySize: number;
timeout: number;
reservedConcurrentExecutions?: number;
};
monitoring: {
alarmEmail?: string;
enableXRay: boolean;
logRetentionDays: number;
};
}
// lib/config/stages/dev.ts
export const devConfig: EnvironmentConfig = {
stage: 'dev',
region: 'us-east-1',
account: '123456789012',
api: {
throttling: {
rateLimit: 100,
burstLimit: 200,
},
cors: {
origins: ['http://localhost:3000'],
credentials: true,
},
},
lambda: {
memorySize: 512,
timeout: 30,
},
monitoring: {
enableXRay: true,
logRetentionDays: 7,
},
};
// lib/config/stages/prod.ts
export const prodConfig: EnvironmentConfig = {
stage: 'prod',
region: 'us-east-1',
account: '123456789012',
api: {
throttling: {
rateLimit: 1000,
burstLimit: 2000,
},
cors: {
origins: ['https://myapp.com'],
credentials: true,
},
},
lambda: {
memorySize: 1024,
timeout: 30,
reservedConcurrentExecutions: 100,
},
monitoring: {
alarmEmail: 'alerts@myapp.com',
enableXRay: true,
logRetentionDays: 30,
},
};
// lib/config/index.ts
import { devConfig } from './stages/dev';
import { prodConfig } from './stages/prod';
export function getConfig(stage: string): EnvironmentConfig {
switch (stage) {
case 'dev':
return devConfig;
case 'prod':
return prodConfig;
default:
throw new Error(`Unknown stage: ${stage}`);
}
}
İlk Construct'ınızı Oluşturma#
Construct'lar CDK'nın yapı taşlarıdır. Lambda fonksiyonları için reusable bir pattern oluşturalım:
// lib/constructs/serverless-function.ts
import { Construct } from 'constructs';
import { NodejsFunction, NodejsFunctionProps } from 'aws-cdk-lib/aws-lambda-nodejs';
import { Runtime, Tracing } from 'aws-cdk-lib/aws-lambda';
import { Duration } from 'aws-cdk-lib';
import { EnvironmentConfig } from '../config/environment';
export interface ServerlessFunctionProps {
entry: string;
handler?: string;
environment?: Record<string, string>;
config: EnvironmentConfig;
memorySize?: number;
timeout?: number;
}
export class ServerlessFunction extends NodejsFunction {
constructor(scope: Construct, id: string, props: ServerlessFunctionProps) {
const { config, ...functionProps } = props;
super(scope, id, {
runtime: Runtime.NODEJS_20_X,
handler: props.handler || 'handler',
entry: props.entry,
memorySize: props.memorySize || config.lambda.memorySize,
timeout: Duration.seconds(props.timeout || config.lambda.timeout),
tracing: config.monitoring.enableXRay ? Tracing.ACTIVE : Tracing.DISABLED,
environment: {
NODE_OPTIONS: '--enable-source-maps',
STAGE: config.stage,
...props.environment,
},
bundling: {
minify: config.stage === 'prod',
sourceMap: true,
sourcesContent: false,
target: 'es2022',
keepNames: true,
// AWS SDK v3'ü hariç tut (Lambda runtime'da sağlanıyor)
externalModules: [
'@aws-sdk/*',
],
},
reservedConcurrentExecutions: config.lambda.reservedConcurrentExecutions,
});
}
}
İlk Stack'inizi Kurma#
Şimdi construct'ımızı kullanan bir stack oluşturalım:
// lib/stacks/api-stack.ts
import { Stack, StackProps } from 'aws-cdk-lib';
import { Construct } from 'constructs';
import { RestApi, LambdaIntegration, Cors } from 'aws-cdk-lib/aws-apigateway';
import { ServerlessFunction } from '../constructs/serverless-function';
import { EnvironmentConfig } from '../config/environment';
export interface ApiStackProps extends StackProps {
config: EnvironmentConfig;
}
export class ApiStack extends Stack {
public readonly api: RestApi;
constructor(scope: Construct, id: string, props: ApiStackProps) {
super(scope, id, props);
const { config } = props;
// API Gateway oluştur
this.api = new RestApi(this, 'ServerlessApi', {
restApiName: `my-service-${config.stage}`,
deployOptions: {
stageName: config.stage,
throttlingRateLimit: config.api.throttling.rateLimit,
throttlingBurstLimit: config.api.throttling.burstLimit,
},
defaultCorsPreflightOptions: {
allowOrigins: config.api.cors.origins,
allowCredentials: config.api.cors.credentials,
allowMethods: Cors.ALL_METHODS,
allowHeaders: [
'Content-Type',
'Authorization',
'X-Api-Key',
],
},
});
// Lambda fonksiyonları oluştur
const createUserFn = new ServerlessFunction(this, 'CreateUserFunction', {
entry: 'src/handlers/users.ts',
handler: 'create',
config,
environment: {
// Environment variable'lar Bölüm 4'te eklenecek
},
});
// Route'ları kur
const users = this.api.root.addResource('users');
users.addMethod('POST', new LambdaIntegration(createUserFn));
}
}
CDK App Entry Point#
CDK app entry point'ini konfigürasyon sistemimizi kullanacak şekilde güncelleyin:
// bin/my-service.ts
#!/usr/bin/env node
import 'source-map-support/register';
import { App } from 'aws-cdk-lib';
import { ApiStack } from '../lib/stacks/api-stack';
import { getConfig } from '../lib/config';
const app = new App();
// Context veya environment'tan stage al
const stage = app.node.tryGetContext('stage') || process.env.STAGE || 'dev';
const config = getConfig(stage);
new ApiStack(app, `MyServiceApiStack-${stage}`, {
config,
env: {
account: config.account,
region: config.region,
},
tags: {
Stage: stage,
Service: 'my-service',
ManagedBy: 'cdk',
},
});
İlk Lambda Handler'ınız#
TypeScript kullanarak bir Lambda handler oluşturun:
// src/handlers/users.ts
import { APIGatewayProxyEventV2, APIGatewayProxyResultV2 } from 'aws-lambda';
export const create = async (
event: APIGatewayProxyEventV2
): Promise<APIGatewayProxyResultV2> => {
console.log('Event:', JSON.stringify(event, null, 2));
try {
const body = JSON.parse(event.body || '{}');
// Handler logiği burada (Bölüm 3'te genişletilecek)
return {
statusCode: 201,
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
message: 'User created successfully',
stage: process.env.STAGE,
}),
};
} catch (error) {
console.error('Error:', error);
return {
statusCode: 500,
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
error: 'Internal server error',
}),
};
}
};
Deployment Komutları#
Bu script'leri package.json
'nıza ekleyin:
{
"scripts": {
"build": "tsc",
"watch": "tsc -w",
"cdk": "cdk",
"bootstrap": "cdk bootstrap",
"deploy:dev": "cdk deploy --context stage=dev",
"deploy:prod": "cdk deploy --context stage=prod",
"diff:dev": "cdk diff --context stage=dev",
"diff:prod": "cdk diff --context stage=prod",
"synth": "cdk synth",
"test": "jest",
"test:watch": "jest --watch"
}
}
İlk Deployment#
AWS environment'ınızı bootstrap edin (tek seferlik kurulum):
npm run bootstrap
Development'a deploy edin:
npm run deploy:dev
CDK hangi kaynakları oluşturmayı planladığını gösterecek. İnceleyin ve onaylayın.
Local Development Kurulumu#
Serverless Framework'ün serverless-offline
'ının aksine, CDK built-in local API Gateway emülasyonu sağlamıyor. Local development için birkaç seçeneğiniz var:
- SAM CLI Entegrasyonu (Önerilen):
# SAM CLI yükle
brew install aws-sam-cli # macOS
# veya takip edin: https://docs.aws.amazon.com/serverless-application-model/latest/developerguide/install-sam-cli.html
# CloudFormation template oluştur
cdk synth --no-staging > template.yaml
# Local API başlat
sam local start-api -t template.yaml
- Doğrudan Handler Testing:
// test/handlers/users.test.ts
import { create } from '../../src/handlers/users';
import { APIGatewayProxyEventV2 } from 'aws-lambda';
describe('Users Handler', () => {
it('should create a user', async () => {
const event: Partial<APIGatewayProxyEventV2> = {
body: JSON.stringify({ name: 'John Doe' }),
};
const result = await create(event as APIGatewayProxyEventV2);
expect(result.statusCode).toBe(201);
expect(JSON.parse(result.body!)).toHaveProperty('message');
});
});
Hatırlanması Gereken Temel Farklar#
Aspekt | Serverless Framework | CDK |
---|---|---|
Konfigürasyon | YAML dosyaları | TypeScript kodu |
Environment Variable'lar | ${self:provider.stage} | Config objeleri |
Local Development | serverless-offline | SAM CLI veya testing |
Deployment | serverless deploy | cdk deploy |
Kaynak Reference'ları | !Ref veya ${cf:stackName.output} | Doğrudan object reference'ları |
Sırada Ne Var#
Artık Serverless Framework geleneklerini yansıtan ama CDK'nın type safety ve composability'sini benimseyen sağlam bir CDK temeline sahipsiniz. Lambda fonksiyonlarınız tanıdık lokasyonlarda yaşıyor, ama altyapınız artık kod - gerçek, test edilebilir TypeScript kodu.
3. Bölüm'te, şunları içeren Lambda fonksiyonları ve API Gateway konfigürasyonlarını migrate edeceğiz:
- Request/response transformasyon'ları
- API Gateway modelleri ve validator'ları
- Lambda layer'ları ve dependency'leri
- Error handling pattern'leri
- API versioning stratejileri
Temel atıldı. Serverless API'nızı inşa edelim.
Yorumlar (0)
Sohbete katıl
Düşüncelerini paylaşmak ve toplulukla etkileşim kurmak için giriş yap
Henüz yorum yok
Bu yazı hakkında ilk düşüncelerini paylaşan sen ol!
Yorumlar (0)
Sohbete katıl
Düşüncelerini paylaşmak ve toplulukla etkileşim kurmak için giriş yap
Henüz yorum yok
Bu yazı hakkında ilk düşüncelerini paylaşan sen ol!