@@ -15,8 +15,8 @@ import * as targets from "aws-cdk-lib/aws-route53-targets";
15
15
import * as route53 from "aws-cdk-lib/aws-route53" ;
16
16
17
17
export interface DiscordProps extends cdk . StackProps {
18
- readonly domain : [ string , string ] | string ;
19
- readonly leaderboardApi : string ;
18
+ readonly discordDomain : [ string , string ] | string ;
19
+ readonly leaderboardDomain : [ string , string ] | string ;
20
20
readonly discordAppId : string ;
21
21
readonly publicKey : string ;
22
22
}
@@ -25,7 +25,13 @@ const __dirname = path.dirname(fileURLToPath(import.meta.url));
25
25
26
26
export class DiscordStack extends cdk . Stack {
27
27
constructor ( scope : cdk . App , id : string , props : DiscordProps ) {
28
- const { domain, publicKey, discordAppId, leaderboardApi, ...rest } = props ;
28
+ const {
29
+ discordDomain,
30
+ leaderboardDomain,
31
+ publicKey,
32
+ discordAppId,
33
+ ...rest
34
+ } = props ;
29
35
super ( scope , id , rest ) ;
30
36
31
37
// DynamoDB tables
@@ -36,6 +42,13 @@ export class DiscordStack extends cdk.Stack {
36
42
} ,
37
43
tableClass : dynamodb . TableClass . STANDARD ,
38
44
} ) ;
45
+ const currentBoardTable = new dynamodb . Table ( this , "CurrentBoard" , {
46
+ partitionKey : {
47
+ name : "context" ,
48
+ type : dynamodb . AttributeType . STRING ,
49
+ } ,
50
+ tableClass : dynamodb . TableClass . STANDARD ,
51
+ } ) ;
39
52
const rankBoardTable = new dynamodb . Table ( this , "boards" , {
40
53
partitionKey : { name : "Name" , type : dynamodb . AttributeType . STRING } ,
41
54
tableClass : dynamodb . TableClass . STANDARD ,
@@ -102,55 +115,97 @@ export class DiscordStack extends cdk.Stack {
102
115
) ;
103
116
104
117
const discordHandler = new lambda . Function ( this , "discordLambda" , {
105
- runtime : lambda . Runtime . PROVIDED ,
118
+ runtime : lambda . Runtime . NODEJS_14_X ,
106
119
code : lambda . Code . fromAsset (
107
120
path . join ( __dirname , "../../../.layers/discord" )
108
121
) ,
109
122
handler : "index.handler" ,
110
123
timeout : cdk . Duration . seconds ( 10 ) ,
111
124
memorySize : 256 ,
112
- layers : [ node16Layer ] ,
113
125
environment : {
114
126
PUBLIC_KEY : publicKey ,
115
127
STATIC_IMAGE_URL : `https://${ staticAssetBucket . bucketName } .s3.amazonaws.com` ,
116
128
MINIMUM_LOG_LEVEL : "INFO" ,
117
129
TABLE_NAME_MINECRAFT_PLAYER : minecraftPlayerTable . tableName ,
130
+ TABLE_NAME_CURRENT_BOARD : currentBoardTable . tableName ,
118
131
TABLE_NAME_RANKER_BOARDS : rankBoardTable . tableName ,
119
132
TABLE_NAME_RANKER_SCORES : rankScoresTable . tableName ,
120
133
TABLE_NAME_RANKER_NODES : rankNodesTable . tableName ,
121
134
TABLE_NAME_RANKER_LEADERBOARDS : rankLeaderboardsTable . tableName ,
122
- CURRENT_LEADERBOARD : "potato" ,
123
- LEADERBOARD_BASE : leaderboardApi ,
124
135
DISCORD_DEFERRED_MESSAGE_TOPIC_ARN : deferredMessageTopic . topicArn ,
125
136
} ,
126
137
} ) ;
127
138
128
139
minecraftPlayerTable . grantReadWriteData ( discordHandler ) ;
140
+ currentBoardTable . grantReadWriteData ( discordHandler ) ;
129
141
rankBoardTable . grantReadWriteData ( discordHandler ) ;
130
142
rankScoresTable . grantReadWriteData ( discordHandler ) ;
131
143
rankNodesTable . grantReadWriteData ( discordHandler ) ;
132
144
rankLeaderboardsTable . grantReadWriteData ( discordHandler ) ;
133
145
deferredMessageTopic . grantPublish ( discordHandler ) ;
134
146
147
+ const playersHandler = new lambda . Function ( this , "playerHandler" , {
148
+ runtime : lambda . Runtime . NODEJS_14_X ,
149
+ code : lambda . Code . fromAsset (
150
+ path . join ( __dirname , "../../../.layers/players" )
151
+ ) ,
152
+ handler : "index.handler" ,
153
+ timeout : cdk . Duration . seconds ( 10 ) ,
154
+ memorySize : 256 ,
155
+ environment : {
156
+ MINIMUM_LOG_LEVEL : "INFO" ,
157
+ TABLE_NAME_RANKER_BOARDS : rankBoardTable . tableName ,
158
+ TABLE_NAME_RANKER_SCORES : rankScoresTable . tableName ,
159
+ TABLE_NAME_RANKER_NODES : rankNodesTable . tableName ,
160
+ TABLE_NAME_RANKER_LEADERBOARDS : rankLeaderboardsTable . tableName ,
161
+ } ,
162
+ } ) ;
163
+
164
+ rankBoardTable . grantReadData ( playersHandler ) ;
165
+ rankScoresTable . grantReadData ( playersHandler ) ;
166
+ rankNodesTable . grantReadData ( playersHandler ) ;
167
+ rankLeaderboardsTable . grantReadData ( playersHandler ) ;
168
+
169
+ const leaderboardHandler = new lambda . Function ( this , "leaderboardHandler" , {
170
+ runtime : lambda . Runtime . NODEJS_14_X ,
171
+ code : lambda . Code . fromAsset (
172
+ path . join ( __dirname , "../../../.layers/leaderboard" )
173
+ ) ,
174
+ handler : "index.handler" ,
175
+ timeout : cdk . Duration . seconds ( 10 ) ,
176
+ memorySize : 256 ,
177
+ environment : {
178
+ MINIMUM_LOG_LEVEL : "INFO" ,
179
+ TABLE_NAME_RANKER_BOARDS : rankBoardTable . tableName ,
180
+ TABLE_NAME_RANKER_SCORES : rankScoresTable . tableName ,
181
+ TABLE_NAME_RANKER_NODES : rankNodesTable . tableName ,
182
+ TABLE_NAME_RANKER_LEADERBOARDS : rankLeaderboardsTable . tableName ,
183
+ } ,
184
+ } ) ;
185
+
186
+ rankBoardTable . grantReadData ( leaderboardHandler ) ;
187
+ rankScoresTable . grantReadData ( leaderboardHandler ) ;
188
+ rankNodesTable . grantReadData ( leaderboardHandler ) ;
189
+ rankLeaderboardsTable . grantReadData ( leaderboardHandler ) ;
190
+
135
191
const deferredMessageHandler = new lambda . Function (
136
192
this ,
137
193
"deferredMessageHandler" ,
138
194
{
139
- runtime : lambda . Runtime . PROVIDED ,
195
+ runtime : lambda . Runtime . NODEJS_14_X ,
140
196
code : lambda . Code . fromAsset (
141
197
path . join ( __dirname , "../../../.layers/deferred" )
142
198
) ,
143
199
handler : "index.handler" ,
144
200
timeout : cdk . Duration . seconds ( 30 ) ,
145
201
memorySize : 256 ,
146
- layers : [ node16Layer ] ,
147
202
environment : {
148
203
DISCORD_APPLICATION_ID : discordAppId ,
149
204
STATIC_IMAGE_URL : `https://${ staticAssetBucket . bucketName } .s3.amazonaws.com` ,
150
205
MINIMUM_LOG_LEVEL : "INFO" ,
151
206
CURRENT_LEADERBOARD : "potato" ,
152
- LEADERBOARD_BASE : leaderboardApi ,
153
207
TABLE_NAME_MINECRAFT_PLAYER : minecraftPlayerTable . tableName ,
208
+ TABLE_NAME_CURRENT_BOARD : currentBoardTable . tableName ,
154
209
TABLE_NAME_RANKER_BOARDS : rankBoardTable . tableName ,
155
210
TABLE_NAME_RANKER_SCORES : rankScoresTable . tableName ,
156
211
TABLE_NAME_RANKER_NODES : rankNodesTable . tableName ,
@@ -163,21 +218,22 @@ export class DiscordStack extends cdk.Stack {
163
218
] ,
164
219
}
165
220
) ;
221
+
166
222
minecraftPlayerTable . grantReadWriteData ( deferredMessageHandler ) ;
167
- rankBoardTable . grantReadData ( deferredMessageHandler ) ;
223
+ currentBoardTable . grantReadWriteData ( deferredMessageHandler ) ;
224
+ rankBoardTable . grantReadWriteData ( deferredMessageHandler ) ;
168
225
rankScoresTable . grantReadData ( deferredMessageHandler ) ;
169
226
rankNodesTable . grantReadData ( deferredMessageHandler ) ;
170
227
rankLeaderboardsTable . grantReadData ( deferredMessageHandler ) ;
171
228
172
229
const scoreHandler = new lambda . Function ( this , "scoreHandler" , {
173
- runtime : lambda . Runtime . PROVIDED ,
230
+ runtime : lambda . Runtime . NODEJS_14_X ,
174
231
code : lambda . Code . fromAsset (
175
232
path . join ( __dirname , "../../../.layers/ingest" )
176
233
) ,
177
234
handler : "index.handler" ,
178
235
timeout : cdk . Duration . seconds ( 10 ) ,
179
236
memorySize : 256 ,
180
- layers : [ node16Layer ] ,
181
237
environment : {
182
238
TABLE_NAME_RANKER_BOARDS : rankBoardTable . tableName ,
183
239
TABLE_NAME_RANKER_SCORES : rankScoresTable . tableName ,
@@ -193,40 +249,132 @@ export class DiscordStack extends cdk.Stack {
193
249
rankLeaderboardsTable . grantReadWriteData ( scoreHandler ) ;
194
250
195
251
// Domain
196
- const domains = domain instanceof Array ? domain : [ domain ] ;
197
- const domainName = domains . join ( "." ) ;
198
- const hostedZone = route53 . HostedZone . fromLookup ( this , "HostedZone" , {
199
- domainName : domain . length === 2 ? domains [ 1 ] : domains [ 0 ] ,
200
- } ) ;
252
+ const discordDomains =
253
+ discordDomain instanceof Array ? discordDomain : [ discordDomain ] ;
254
+ const discordDomainName = discordDomains . join ( "." ) ;
255
+ const discordHostedZone = route53 . HostedZone . fromLookup (
256
+ this ,
257
+ "HostedZone" ,
258
+ {
259
+ domainName :
260
+ discordDomains . length === 2 ? discordDomains [ 1 ] : discordDomains [ 0 ] ,
261
+ }
262
+ ) ;
201
263
202
- const certificate = new acm . DnsValidatedCertificate ( this , "certificate" , {
203
- domainName,
204
- hostedZone : hostedZone ,
205
- region : props . env ?. region ,
206
- } ) ;
264
+ const leaderboardDomains =
265
+ leaderboardDomain instanceof Array
266
+ ? leaderboardDomain
267
+ : [ leaderboardDomain ] ;
268
+ const leaderboardDomainName = leaderboardDomains . join ( "." ) ;
269
+ const leaderboardHostedZone = route53 . HostedZone . fromLookup (
270
+ this ,
271
+ "LeaderboardHostedZone" ,
272
+ {
273
+ domainName :
274
+ leaderboardDomains . length === 2
275
+ ? leaderboardDomains [ 1 ]
276
+ : leaderboardDomains [ 0 ] ,
277
+ }
278
+ ) ;
207
279
208
- const api = new apigateway . RestApi ( this , "discordApi" , {
280
+ const discordCertificate = new acm . DnsValidatedCertificate (
281
+ this ,
282
+ "certificate" ,
283
+ {
284
+ domainName : discordDomainName ,
285
+ hostedZone : discordHostedZone ,
286
+ region : props . env ?. region ,
287
+ }
288
+ ) ;
289
+
290
+ const discordApi = new apigateway . RestApi ( this , "discordApi" , {
209
291
restApiName : "Discord Service" ,
210
292
description : "Discord callback" ,
211
293
domainName : {
212
- domainName,
213
- certificate,
294
+ domainName : discordDomainName ,
295
+ certificate : discordCertificate ,
296
+ } ,
297
+ } ) ;
298
+
299
+ const leaderboardCertificate = new acm . DnsValidatedCertificate (
300
+ this ,
301
+ "leaderboardCertificate" ,
302
+ {
303
+ domainName : leaderboardDomainName ,
304
+ hostedZone : leaderboardHostedZone ,
305
+ region : props . env ?. region ,
306
+ }
307
+ ) ;
308
+
309
+ const leaderboardApi = new apigateway . RestApi ( this , "leaderboardApi" , {
310
+ restApiName : "Leaderboard Service" ,
311
+ description : "Leaderboard callback" ,
312
+ domainName : {
313
+ domainName : leaderboardDomainName ,
314
+ certificate : leaderboardCertificate ,
214
315
} ,
215
316
} ) ;
317
+ const defaultUsagePlan = leaderboardApi . addUsagePlan ( "defaultUsagePlan" , {
318
+ name : "Default Usage Plan" ,
319
+ description : "Default Usage Plan" ,
320
+ } ) ;
321
+ const defaultApiKey = leaderboardApi . addApiKey ( "defaultKey" , {
322
+ description : "Default leaderboard API key" ,
323
+ } ) ;
324
+ defaultUsagePlan . addApiKey ( defaultApiKey ) ;
325
+ defaultUsagePlan . addApiStage ( {
326
+ stage : leaderboardApi . deploymentStage ,
327
+ } ) ;
216
328
217
329
const discordIntegration = new apigateway . LambdaIntegration ( discordHandler ) ;
218
- const resource = api . root . addResource ( "discord" ) ;
219
- resource . addMethod ( "POST" , discordIntegration ) ;
330
+ const discordResource = discordApi . root . addResource ( "discord" ) ;
331
+ discordResource . addMethod ( "POST" , discordIntegration ) ;
332
+ const experienceResource = leaderboardApi . root . addResource ( "{experience}" ) ;
333
+ const leaderboardResource = experienceResource . addResource ( "leaderboard" ) ;
334
+ leaderboardResource . addMethod (
335
+ "GET" ,
336
+ new apigateway . LambdaIntegration ( leaderboardHandler ) ,
337
+ {
338
+ apiKeyRequired : true ,
339
+ }
340
+ ) ;
341
+
342
+ const playersResource = experienceResource . addResource ( "players" ) ;
343
+ playersResource . addMethod (
344
+ "GET" ,
345
+ new apigateway . LambdaIntegration ( playersHandler ) ,
346
+ {
347
+ apiKeyRequired : true ,
348
+ }
349
+ ) ;
220
350
221
351
new route53 . ARecord ( this , "ipv4-record" , {
222
- zone : hostedZone ,
223
- recordName : domainName ,
224
- target : route53 . RecordTarget . fromAlias ( new targets . ApiGateway ( api ) ) ,
352
+ zone : discordHostedZone ,
353
+ recordName : discordDomainName ,
354
+ target : route53 . RecordTarget . fromAlias (
355
+ new targets . ApiGateway ( discordApi )
356
+ ) ,
225
357
} ) ;
226
358
new route53 . AaaaRecord ( this , "ipv6-record" , {
227
- zone : hostedZone ,
228
- recordName : domainName ,
229
- target : route53 . RecordTarget . fromAlias ( new targets . ApiGateway ( api ) ) ,
359
+ zone : discordHostedZone ,
360
+ recordName : discordDomainName ,
361
+ target : route53 . RecordTarget . fromAlias (
362
+ new targets . ApiGateway ( discordApi )
363
+ ) ,
364
+ } ) ;
365
+ new route53 . ARecord ( this , "ipv4-record-leaderboard" , {
366
+ zone : leaderboardHostedZone ,
367
+ recordName : leaderboardDomainName ,
368
+ target : route53 . RecordTarget . fromAlias (
369
+ new targets . ApiGateway ( leaderboardApi )
370
+ ) ,
371
+ } ) ;
372
+ new route53 . AaaaRecord ( this , "ipv6-record-leaderboard" , {
373
+ zone : leaderboardHostedZone ,
374
+ recordName : leaderboardDomainName ,
375
+ target : route53 . RecordTarget . fromAlias (
376
+ new targets . ApiGateway ( leaderboardApi )
377
+ ) ,
230
378
} ) ;
231
379
232
380
new cdk . CfnOutput ( this , "snsScoreTopicArn" , {
0 commit comments