使用 AWS Lambda 和 CDK 為 Node.js 專案建立 WebSocket API 無伺服器應用程式

建立專案

建立專案。

1
2
mkdir cdk-node-example
cd cdk-node-example

初始化專案。

1
cdk init app --language typescript

安裝依賴套件。

1
2
npm i aws-sdk @aws-sdk/client-dynamodb @aws-sdk/lib-dynamodb
npm i -D @types/aws-sdk @types/aws-lambda

建立函式

建立 lambda 資料夾。

1
mkdir lambda

建立 lambda/connect-handler.ts 檔,處理建立連線的行為。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import { DynamoDBClient } from '@aws-sdk/client-dynamodb';
import { DynamoDBDocumentClient, PutCommand } from '@aws-sdk/lib-dynamodb';
import { APIGatewayProxyWebsocketHandlerV2 } from 'aws-lambda';

const client = new DynamoDBClient();
const docClient = DynamoDBDocumentClient.from(client);

export const handler: APIGatewayProxyWebsocketHandlerV2 = async (event) => {
const command = new PutCommand({
TableName: process.env.TABLE_NAME as string,
Item: {
connectionId: event.requestContext.connectionId,
},
});
await docClient.send(command);
return {
statusCode: 200,
};
};

建立 lambda/disconnect-handler.ts 檔,處理斷開連線的行為。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import { DynamoDBClient } from '@aws-sdk/client-dynamodb';
import { DeleteCommand, DynamoDBDocumentClient } from '@aws-sdk/lib-dynamodb';
import { APIGatewayProxyWebsocketHandlerV2 } from 'aws-lambda';

const client = new DynamoDBClient();
const docClient = DynamoDBDocumentClient.from(client);

export const handler: APIGatewayProxyWebsocketHandlerV2 = async (event) => {
const command = new DeleteCommand({
TableName: process.env.TABLE_NAME as string,
Key: {
connectionId: event.requestContext.connectionId,
},
});
await docClient.send(command);
return {
statusCode: 200,
};
};

建立 lambda/default-handler.ts 檔,處理預設路由的行為。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
import { APIGatewayProxyWebsocketHandlerV2 } from 'aws-lambda';
import * as AWS from 'aws-sdk';

export const handler: APIGatewayProxyWebsocketHandlerV2 = async (event) => {
const { connectionId } = event.requestContext;

const callbackAPI = new AWS.ApiGatewayManagementApi({
apiVersion: '2018-11-29',
endpoint: `${event.requestContext.domainName}/${event.requestContext.stage}`,
});

let connectionInfo: AWS.ApiGatewayManagementApi.GetConnectionResponse;
try {
connectionInfo = await callbackAPI
.getConnection({
ConnectionId: event.requestContext.connectionId,
})
.promise();
} catch (e) {
console.log(e);
}

const info = {
...connectionInfo!,
connectionId,
};

await callbackAPI.postToConnection({
ConnectionId: event.requestContext.connectionId,
Data: `Use the send-message route to send a message. Your info: ${JSON.stringify(info)}`,
}).promise();

return {
statusCode: 200,
};
};

建立 lambda/send-handler.ts 檔,處理傳送訊息的行為。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
import { DynamoDBClient } from '@aws-sdk/client-dynamodb';
import { DynamoDBDocumentClient, ScanCommand } from '@aws-sdk/lib-dynamodb';
import { APIGatewayProxyWebsocketHandlerV2 } from 'aws-lambda';
import * as AWS from 'aws-sdk';

const client = new DynamoDBClient();
const docClient = DynamoDBDocumentClient.from(client);

export const handler: APIGatewayProxyWebsocketHandlerV2 = async (event) => {
const command = new ScanCommand({
TableName: process.env.TABLE_NAME as string,
});
const response = await docClient.send(command);
const connections = response.Items ?? [];

const callbackAPI = new AWS.ApiGatewayManagementApi({
apiVersion: '2018-11-29',
endpoint: `${event.requestContext.domainName}/${event.requestContext.stage}`,
});

const message = JSON.parse(event.body ?? '{}').message;

await Promise.all(
connections
.filter(({ connectionId }) => connectionId !== event.requestContext.connectionId)
.map(({ connectionId }) => (
callbackAPI
.postToConnection({
ConnectionId: connectionId,
Data: message,
})
.promise()
))
);
return {
statusCode: 200,
};
};

建立堆疊

修改 lib/cdk-node-example-stack.ts 檔。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
import * as cdk from 'aws-cdk-lib';
import { WebSocketApi, WebSocketStage } from 'aws-cdk-lib/aws-apigatewayv2';
import { WebSocketLambdaIntegration } from 'aws-cdk-lib/aws-apigatewayv2-integrations';
import { BillingMode, Table } from 'aws-cdk-lib/aws-dynamodb';
import { Architecture, Runtime } from 'aws-cdk-lib/aws-lambda';
import { Construct } from 'constructs';

export class CdkNodeExampleStack extends cdk.Stack {
constructor(scope: Construct, id: string, props?: cdk.StackProps) {
super(scope, id, props);

// 創建一個 DynamoDB 表格來存儲 WebSocket 連線的資訊
const table = new cdk.aws_dynamodb.Table(this, 'ConnectionTable', {
billingMode: BillingMode.PAY_PER_REQUEST,
removalPolicy: cdk.RemovalPolicy.DESTROY,
partitionKey: {
name: 'connectionId',
type: cdk.aws_dynamodb.AttributeType.STRING,
},
});

// 使用自定義方法建立處理 WebSocket 建立連線、斷開連線、發送訊息和預設行為的 Lambda 函數
const connectHandler = this.connectHandlerBuilder(table);
const disconnectHandler = this.disconnectHandlerBuilder(table);
const sendMessageHandler = this.sendMessageHandlerBuilder(table);
const defaultHandler = this.defaultHandlerBuilder();

// 創建 WebSocket API 並設定路由選擇表達式和各路由的 Lambda 集成
const webSocketApi = new WebSocketApi(this, 'CdkNodeExampleWebSocketApi', {
// 路由選擇表達式
routeSelectionExpression: '$request.body.action',
// 建立連線時的路由選項
connectRouteOptions: {
integration: new WebSocketLambdaIntegration('ConnectIntegration', connectHandler),
},
// 斷開連線時的路由選項
disconnectRouteOptions: {
integration: new WebSocketLambdaIntegration('DisconnectIntegration', disconnectHandler),
},
// 預設行為的路由選項
defaultRouteOptions: {
integration: new WebSocketLambdaIntegration('DefaultIntegration', defaultHandler),
},
});

// 為發送訊息添加專用路由
webSocketApi.addRoute('send-message', {
integration: new WebSocketLambdaIntegration('SendMessageIntegration', sendMessageHandler),
});
// 賦予發送訊息和預設行為的處理器管理 WebSocket 連線的權限
webSocketApi.grantManageConnections(sendMessageHandler);
webSocketApi.grantManageConnections(defaultHandler);

// 創建 WebSocket API 的一個階段並自動部署
new WebSocketStage(this, 'CdkNodeExampleProductionStage', {
webSocketApi,
stageName: 'production',
autoDeploy: true,
});
}

// 定義一個方法來構建處理 WebSocket 建立連線的 Lambda 函數
connectHandlerBuilder(table: Table) {
const handler = new cdk.aws_lambda_nodejs.NodejsFunction(this, 'ConnectHandler', {
environment: {
TABLE_NAME: table.tableName,
},
architecture: Architecture.ARM_64,
runtime: Runtime.NODEJS_20_X,
entry: 'lambda/connect-handler.ts',
});

// 賦予 Lambda 函數寫入 DynamoDB 表的權限
table.grantWriteData(handler);

return handler;
}

// 定義一個方法來構建處理 WebSocket 斷開連線的 Lambda 函數
disconnectHandlerBuilder(table: Table) {
const handler = new cdk.aws_lambda_nodejs.NodejsFunction(this, 'DisconnectHandler', {
environment: {
TABLE_NAME: table.tableName,
},
architecture: Architecture.ARM_64,
runtime: Runtime.NODEJS_20_X,
entry: 'lambda/disconnect-handler.ts',
});

// 賦予 Lambda 函數寫入 DynamoDB 表的權限
table.grantWriteData(handler);

return handler;
}

// 定義一個方法來構建處理發送 WebSocket 訊息的 Lambda 函數
sendMessageHandlerBuilder(table: Table) {
const handler = new cdk.aws_lambda_nodejs.NodejsFunction(this, 'SendMessageHandler', {
environment: {
TABLE_NAME: table.tableName,
},
architecture: Architecture.ARM_64,
runtime: Runtime.NODEJS_20_X,
entry: 'lambda/send-handler.ts',
});

// 賦予 Lambda 函數讀寫 DynamoDB 表的權限
table.grantReadWriteData(handler);

return handler;
}

// 定義一個方法來構建處理 WebSocket 預設行為的 Lambda 函數
defaultHandlerBuilder() {
return new cdk.aws_lambda_nodejs.NodejsFunction(this, 'DefaultHandler', {
architecture: Architecture.ARM_64,
runtime: Runtime.NODEJS_20_X,
entry: 'lambda/default-handler.ts',
});
}
}

列出所有堆疊。

1
cdk ls

查看堆疊變化。

1
aws-vault exec your-profile -- cdk diff

部署

啟動初始化程序。

1
aws-vault exec your-profile -- cdk bootstrap

部署應用程式。

1
aws-vault exec your-profile -- cdk deploy

如果要清理的話,移除應用程式。

1
aws-vault exec your-profile -- cdk destroy

測試

安裝 wscat 指令。

1
npm i -g wscat

開啟終端機,建立連線。

1
wscat -c wss://xxx.execute-api.ap-northeast-1.amazonaws.com/production

開啟另一個終端機,建立連線。

1
wscat -c wss://xxx.execute-api.ap-northeast-1.amazonaws.com/production

發送訊息:

1
> {"action": "send-message", "message": "hello, everyone!"}

接收訊息:

1
< hello, everyone!

程式碼

參考資料