From 45d05e10b49ea3104e92d5f6311be03619039bcf Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Thu, 11 Apr 2024 14:35:11 +0200 Subject: [PATCH 01/10] Main structure of trigger --- .../events/APIGatewayV2WebSocketEvent.scala | 295 ++++++++++++++++++ 1 file changed, 295 insertions(+) create mode 100644 lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala new file mode 100644 index 00000000..67c9a5bb --- /dev/null +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -0,0 +1,295 @@ +/* + * Copyright 2021 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package feral.lambda.events + +import io.circe.Decoder + + +sealed abstract class APIGatewayV2WebSocketEvent: + def resource: String + def path: String + def httpMethod: String + def headers: Map[String, String] + def multiValueHeaders: Map[String, List[String]] + def queryStringParameters: Map[String, String] + def multiValueQueryStringParameters: Map[String, List[String]] + def pathParameters: Map[String, String] + def stageVariables: Map[String, String] + def requestContext: RequestContext + def body: String + def isBase64Encoded: Boolean + +object APIGatewayV2WebSocketEvent: + def apply( + resource: String, + path: String, + httpMethod: String, + headers: Map[String, String], + multiValueHeaders: Map[String, List[String]], + queryStringParameters: Map[String, String], + multiValueQueryStringParameters: Map[String, List[String]], + pathParameters: Map[String, String], + stageVariables: Map[String, String], + requestContext: RequestContext, + body: String, + isBase64Encoded: Boolean + ): APIGatewayV2WebSocketEvent = + new Impl( + resource, + path, + httpMethod, + headers, + multiValueHeaders, + queryStringParameters, + multiValueQueryStringParameters, + pathParameters, + stageVariables, + requestContext, + body, + isBase64Encoded + ) + + implicit val decoder: Decoder[APIGatewayV2WebSocketEvent] = Decoder.forProduct12( + "resource", + "path", + "httpMethod", + "headers", + "multiValueHeaders", + "queryStringParameters", + "multiValueQueryStringParameters", + "pathParameters", + "stageVariables", + "requestContext", + "body", + "isBase64Encoded" + )(APIGatewayV2WebSocketEvent.apply) + + private final case class Impl( + resource: String, + path: String, + httpMethod: String, + headers: Map[String, String], + multiValueHeaders: Map[String, List[String]], + queryStringParameters: Map[String, String], + multiValueQueryStringParameters: Map[String, List[String]], + pathParameters: Map[String, String], + stageVariables: Map[String, String], + requestContext: RequestContext, + body: String, + isBase64Encoded: Boolean + ) extends APIGatewayV2WebSocketEvent: + override def productPrefix = "APIGatewayV2WebSocketEvent" + +sealed abstract class RequestIdentity: + def cognitoIdentityPoolId: String + def accountId: String + def cognitoIdentityId: String + def caller: String + def apiKey: String + def sourceIp: String + def cognitoAuthenticationType: String + def cognitoAuthenticationProvider: String + def userArn: String + def userAgent: String + def user: String + def accessKey: String + +object RequestIdentity: + def apply( + cognitoIdentityPoolId: String, + accountId: String, + cognitoIdentityId: String, + caller: String, + apiKey: String, + sourceIp: String, + cognitoAuthenticationType: String, + cognitoAuthenticationProvider: String, + userArn: String, + userAgent: String, + user: String, + accessKey: String + ): RequestIdentity = + new Impl( + cognitoIdentityPoolId, + accountId, + cognitoIdentityId, + caller, + apiKey, + sourceIp, + cognitoAuthenticationType, + cognitoAuthenticationProvider, + userArn, + userAgent, + user, + accessKey + ) + + private[events] implicit val decoder: Decoder[RequestIdentity] = Decoder.forProduct12( + "cognitoIdentityPoolId", + "accountId", + "cognitoIdentityId", + "caller", + "apiKey", + "sourceIp", + "cognitoAuthenticationType", + "cognitoAuthenticationProvider", + "userArn", + "userAgent", + "user", + "accessKey" + )(RequestIdentity.apply) + + private final case class Impl( + cognitoIdentityPoolId: String, + accountId: String, + cognitoIdentityId: String, + caller: String, + apiKey: String, + sourceIp: String, + cognitoAuthenticationType: String, + cognitoAuthenticationProvider: String, + userArn: String, + userAgent: String, + user: String, + accessKey: String + ) extends RequestIdentity: + override def productPrefix = "RequestIdentity" + +sealed abstract class RequestContext: + def accountId: String + def resourceId: String + def stage: String + def requestId: String + def identity: RequestIdentity + def ResourcePath: String + def authorizer: Map[String, Object] + def httpMethod: String + def apiId: String + def connectedAt: Long + def connectionId: String + def domainName: String + def error: String + def eventType: String + def extendedRequestId: String + def integrationLatency: String + def messageDirection: String + def messageId: String + def requestTime: String + def requestTimeEpoch: Long + def routeKey: String + def status: String + +object RequestContext: + def apply( + accountId: String, + resourceId: String, + stage: String, + requestId: String, + identity: RequestIdentity, + ResourcePath: String, + authorizer: Map[String, Object], + httpMethod: String, + apiId: String, + connectedAt: Long, + connectionId: String, + domainName: String, + error: String, + eventType: String, + extendedRequestId: String, + integrationLatency: String, + messageDirection: String, + messageId: String, + requestTime: String, + requestTimeEpoch: Long, + routeKey: String, + status: String + ): RequestContext = + new Impl( + accountId, + resourceId, + stage, + requestId, + identity, + ResourcePath, + authorizer, + httpMethod, + apiId, + connectedAt, + connectionId, + domainName, + error, + eventType, + extendedRequestId, + integrationLatency, + messageDirection, + messageId, + requestTime, + requestTimeEpoch, + routeKey, + status + ) + + private[events] implicit val decoder: Decoder[RequestContext] = Decoder.forProduct22( + "accountId", + "resourceId", + "stage", + "requestId", + "identity", + "ResourcePath", + "author", + "httpMethod", + "apiId", + "connectedAt", + "connectionId", + "domainName", + "error", + "eventType", + "extendedRequestId", + "integrationLatency", + "messageDirection", + "messageId", + "requestTime", + "requestTimeEpoch", + "routeKey", + "status" + )(RequestContext.apply) + + private final case class Impl( + accountId: String, + resourceId: String, + stage: String, + requestId: String, + identity: RequestIdentity, + ResourcePath: String, + authorizer: Map[String, Object], + httpMethod: String, + apiId: String, + connectedAt: Long, + connectionId: String, + domainName: String, + error: String, + eventType: String, + extendedRequestId: String, + integrationLatency: String, + messageDirection: String, + messageId: String, + requestTime: String, + requestTimeEpoch: Long, + routeKey: String, + status: String + ) extends RequestContext: + override def productPrefix = "RequestContext" \ No newline at end of file From bcfc6cfb5999dab136805be29bcd312158b8a0b1 Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Sun, 14 Apr 2024 00:39:29 +0200 Subject: [PATCH 02/10] Fixed syntax errors (I hope) --- .../events/APIGatewayV2WebSocketEvent.scala | 29 ++++++++++++------- 1 file changed, 19 insertions(+), 10 deletions(-) diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala index 67c9a5bb..1bc248dd 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -19,7 +19,7 @@ package feral.lambda.events import io.circe.Decoder -sealed abstract class APIGatewayV2WebSocketEvent: +sealed abstract class APIGatewayV2WebSocketEvent { def resource: String def path: String def httpMethod: String @@ -32,8 +32,9 @@ sealed abstract class APIGatewayV2WebSocketEvent: def requestContext: RequestContext def body: String def isBase64Encoded: Boolean +} -object APIGatewayV2WebSocketEvent: +object APIGatewayV2WebSocketEvent { def apply( resource: String, path: String, @@ -91,10 +92,12 @@ object APIGatewayV2WebSocketEvent: requestContext: RequestContext, body: String, isBase64Encoded: Boolean - ) extends APIGatewayV2WebSocketEvent: + ) extends APIGatewayV2WebSocketEvent { override def productPrefix = "APIGatewayV2WebSocketEvent" + } +} -sealed abstract class RequestIdentity: +sealed abstract class RequestIdentity { def cognitoIdentityPoolId: String def accountId: String def cognitoIdentityId: String @@ -107,8 +110,9 @@ sealed abstract class RequestIdentity: def userAgent: String def user: String def accessKey: String +} -object RequestIdentity: +object RequestIdentity { def apply( cognitoIdentityPoolId: String, accountId: String, @@ -166,10 +170,12 @@ object RequestIdentity: userAgent: String, user: String, accessKey: String - ) extends RequestIdentity: + ) extends RequestIdentity { override def productPrefix = "RequestIdentity" + } +} -sealed abstract class RequestContext: +sealed abstract class RequestContext { def accountId: String def resourceId: String def stage: String @@ -192,8 +198,9 @@ sealed abstract class RequestContext: def requestTimeEpoch: Long def routeKey: String def status: String +} -object RequestContext: +object RequestContext { def apply( accountId: String, resourceId: String, @@ -291,5 +298,7 @@ object RequestContext: requestTimeEpoch: Long, routeKey: String, status: String - ) extends RequestContext: - override def productPrefix = "RequestContext" \ No newline at end of file + ) extends RequestContext { + override def productPrefix = "RequestContext" + } +} \ No newline at end of file From 2004fa605e8b27693925fccf658a8fc17075c4a5 Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Wed, 17 Apr 2024 16:03:06 +0200 Subject: [PATCH 03/10] Fixed a typo --- .../scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala index 1bc248dd..0ceeabac 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -257,7 +257,7 @@ object RequestContext { "requestId", "identity", "ResourcePath", - "author", + "authorizer", "httpMethod", "apiId", "connectedAt", From 39248ef4688d8e24ba0af446b2c73ad6cbf10395 Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Thu, 18 Apr 2024 01:32:20 +0200 Subject: [PATCH 04/10] Renamed classes to avoid naming collisions --- .../events/APIGatewayV2WebSocketEvent.scala | 40 +++++++++---------- 1 file changed, 20 insertions(+), 20 deletions(-) diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala index 0ceeabac..fee2a84e 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -29,7 +29,7 @@ sealed abstract class APIGatewayV2WebSocketEvent { def multiValueQueryStringParameters: Map[String, List[String]] def pathParameters: Map[String, String] def stageVariables: Map[String, String] - def requestContext: RequestContext + def requestContext: WebSocketRequestContext def body: String def isBase64Encoded: Boolean } @@ -45,7 +45,7 @@ object APIGatewayV2WebSocketEvent { multiValueQueryStringParameters: Map[String, List[String]], pathParameters: Map[String, String], stageVariables: Map[String, String], - requestContext: RequestContext, + requestContext: WebSocketRequestContext, body: String, isBase64Encoded: Boolean ): APIGatewayV2WebSocketEvent = @@ -89,7 +89,7 @@ object APIGatewayV2WebSocketEvent { multiValueQueryStringParameters: Map[String, List[String]], pathParameters: Map[String, String], stageVariables: Map[String, String], - requestContext: RequestContext, + requestContext: WebSocketRequestContext, body: String, isBase64Encoded: Boolean ) extends APIGatewayV2WebSocketEvent { @@ -97,7 +97,7 @@ object APIGatewayV2WebSocketEvent { } } -sealed abstract class RequestIdentity { +sealed abstract class WebSocketRequestIdentity { def cognitoIdentityPoolId: String def accountId: String def cognitoIdentityId: String @@ -112,7 +112,7 @@ sealed abstract class RequestIdentity { def accessKey: String } -object RequestIdentity { +object WebSocketRequestIdentity { def apply( cognitoIdentityPoolId: String, accountId: String, @@ -126,7 +126,7 @@ object RequestIdentity { userAgent: String, user: String, accessKey: String - ): RequestIdentity = + ): WebSocketRequestIdentity = new Impl( cognitoIdentityPoolId, accountId, @@ -142,7 +142,7 @@ object RequestIdentity { accessKey ) - private[events] implicit val decoder: Decoder[RequestIdentity] = Decoder.forProduct12( + private[events] implicit val decoder: Decoder[WebSocketRequestIdentity] = Decoder.forProduct12( "cognitoIdentityPoolId", "accountId", "cognitoIdentityId", @@ -155,7 +155,7 @@ object RequestIdentity { "userAgent", "user", "accessKey" - )(RequestIdentity.apply) + )(WebSocketRequestIdentity.apply) private final case class Impl( cognitoIdentityPoolId: String, @@ -170,17 +170,17 @@ object RequestIdentity { userAgent: String, user: String, accessKey: String - ) extends RequestIdentity { - override def productPrefix = "RequestIdentity" + ) extends WebSocketRequestIdentity { + override def productPrefix = "WebSocketRequestIdentity" } } -sealed abstract class RequestContext { +sealed abstract class WebSocketRequestContext { def accountId: String def resourceId: String def stage: String def requestId: String - def identity: RequestIdentity + def identity: WebSocketRequestIdentity def ResourcePath: String def authorizer: Map[String, Object] def httpMethod: String @@ -200,13 +200,13 @@ sealed abstract class RequestContext { def status: String } -object RequestContext { +object WebSocketRequestContext { def apply( accountId: String, resourceId: String, stage: String, requestId: String, - identity: RequestIdentity, + identity: WebSocketRequestIdentity, ResourcePath: String, authorizer: Map[String, Object], httpMethod: String, @@ -224,7 +224,7 @@ object RequestContext { requestTimeEpoch: Long, routeKey: String, status: String - ): RequestContext = + ): WebSocketRequestContext = new Impl( accountId, resourceId, @@ -250,7 +250,7 @@ object RequestContext { status ) - private[events] implicit val decoder: Decoder[RequestContext] = Decoder.forProduct22( + private[events] implicit val decoder: Decoder[WebSocketRequestContext] = Decoder.forProduct22( "accountId", "resourceId", "stage", @@ -273,14 +273,14 @@ object RequestContext { "requestTimeEpoch", "routeKey", "status" - )(RequestContext.apply) + )(WebSocketRequestContext.apply) private final case class Impl( accountId: String, resourceId: String, stage: String, requestId: String, - identity: RequestIdentity, + identity: WebSocketRequestIdentity, ResourcePath: String, authorizer: Map[String, Object], httpMethod: String, @@ -298,7 +298,7 @@ object RequestContext { requestTimeEpoch: Long, routeKey: String, status: String - ) extends RequestContext { - override def productPrefix = "RequestContext" + ) extends WebSocketRequestContext { + override def productPrefix = "WebSocketRequestContext" } } \ No newline at end of file From a4e620dcb98625a25be093d5288280c95553b4fd Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Mon, 10 Jun 2024 02:39:19 +0800 Subject: [PATCH 05/10] Used DefinitelyTyped implementation instead of AWS Lambda Java's --- .../events/APIGatewayV2WebSocketEvent.scala | 444 ++++++------------ 1 file changed, 140 insertions(+), 304 deletions(-) diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala index fee2a84e..77370e4b 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -1,304 +1,140 @@ -/* - * Copyright 2021 Typelevel - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package feral.lambda.events - -import io.circe.Decoder - - -sealed abstract class APIGatewayV2WebSocketEvent { - def resource: String - def path: String - def httpMethod: String - def headers: Map[String, String] - def multiValueHeaders: Map[String, List[String]] - def queryStringParameters: Map[String, String] - def multiValueQueryStringParameters: Map[String, List[String]] - def pathParameters: Map[String, String] - def stageVariables: Map[String, String] - def requestContext: WebSocketRequestContext - def body: String - def isBase64Encoded: Boolean -} - -object APIGatewayV2WebSocketEvent { - def apply( - resource: String, - path: String, - httpMethod: String, - headers: Map[String, String], - multiValueHeaders: Map[String, List[String]], - queryStringParameters: Map[String, String], - multiValueQueryStringParameters: Map[String, List[String]], - pathParameters: Map[String, String], - stageVariables: Map[String, String], - requestContext: WebSocketRequestContext, - body: String, - isBase64Encoded: Boolean - ): APIGatewayV2WebSocketEvent = - new Impl( - resource, - path, - httpMethod, - headers, - multiValueHeaders, - queryStringParameters, - multiValueQueryStringParameters, - pathParameters, - stageVariables, - requestContext, - body, - isBase64Encoded - ) - - implicit val decoder: Decoder[APIGatewayV2WebSocketEvent] = Decoder.forProduct12( - "resource", - "path", - "httpMethod", - "headers", - "multiValueHeaders", - "queryStringParameters", - "multiValueQueryStringParameters", - "pathParameters", - "stageVariables", - "requestContext", - "body", - "isBase64Encoded" - )(APIGatewayV2WebSocketEvent.apply) - - private final case class Impl( - resource: String, - path: String, - httpMethod: String, - headers: Map[String, String], - multiValueHeaders: Map[String, List[String]], - queryStringParameters: Map[String, String], - multiValueQueryStringParameters: Map[String, List[String]], - pathParameters: Map[String, String], - stageVariables: Map[String, String], - requestContext: WebSocketRequestContext, - body: String, - isBase64Encoded: Boolean - ) extends APIGatewayV2WebSocketEvent { - override def productPrefix = "APIGatewayV2WebSocketEvent" - } -} - -sealed abstract class WebSocketRequestIdentity { - def cognitoIdentityPoolId: String - def accountId: String - def cognitoIdentityId: String - def caller: String - def apiKey: String - def sourceIp: String - def cognitoAuthenticationType: String - def cognitoAuthenticationProvider: String - def userArn: String - def userAgent: String - def user: String - def accessKey: String -} - -object WebSocketRequestIdentity { - def apply( - cognitoIdentityPoolId: String, - accountId: String, - cognitoIdentityId: String, - caller: String, - apiKey: String, - sourceIp: String, - cognitoAuthenticationType: String, - cognitoAuthenticationProvider: String, - userArn: String, - userAgent: String, - user: String, - accessKey: String - ): WebSocketRequestIdentity = - new Impl( - cognitoIdentityPoolId, - accountId, - cognitoIdentityId, - caller, - apiKey, - sourceIp, - cognitoAuthenticationType, - cognitoAuthenticationProvider, - userArn, - userAgent, - user, - accessKey - ) - - private[events] implicit val decoder: Decoder[WebSocketRequestIdentity] = Decoder.forProduct12( - "cognitoIdentityPoolId", - "accountId", - "cognitoIdentityId", - "caller", - "apiKey", - "sourceIp", - "cognitoAuthenticationType", - "cognitoAuthenticationProvider", - "userArn", - "userAgent", - "user", - "accessKey" - )(WebSocketRequestIdentity.apply) - - private final case class Impl( - cognitoIdentityPoolId: String, - accountId: String, - cognitoIdentityId: String, - caller: String, - apiKey: String, - sourceIp: String, - cognitoAuthenticationType: String, - cognitoAuthenticationProvider: String, - userArn: String, - userAgent: String, - user: String, - accessKey: String - ) extends WebSocketRequestIdentity { - override def productPrefix = "WebSocketRequestIdentity" - } -} - -sealed abstract class WebSocketRequestContext { - def accountId: String - def resourceId: String - def stage: String - def requestId: String - def identity: WebSocketRequestIdentity - def ResourcePath: String - def authorizer: Map[String, Object] - def httpMethod: String - def apiId: String - def connectedAt: Long - def connectionId: String - def domainName: String - def error: String - def eventType: String - def extendedRequestId: String - def integrationLatency: String - def messageDirection: String - def messageId: String - def requestTime: String - def requestTimeEpoch: Long - def routeKey: String - def status: String -} - -object WebSocketRequestContext { - def apply( - accountId: String, - resourceId: String, - stage: String, - requestId: String, - identity: WebSocketRequestIdentity, - ResourcePath: String, - authorizer: Map[String, Object], - httpMethod: String, - apiId: String, - connectedAt: Long, - connectionId: String, - domainName: String, - error: String, - eventType: String, - extendedRequestId: String, - integrationLatency: String, - messageDirection: String, - messageId: String, - requestTime: String, - requestTimeEpoch: Long, - routeKey: String, - status: String - ): WebSocketRequestContext = - new Impl( - accountId, - resourceId, - stage, - requestId, - identity, - ResourcePath, - authorizer, - httpMethod, - apiId, - connectedAt, - connectionId, - domainName, - error, - eventType, - extendedRequestId, - integrationLatency, - messageDirection, - messageId, - requestTime, - requestTimeEpoch, - routeKey, - status - ) - - private[events] implicit val decoder: Decoder[WebSocketRequestContext] = Decoder.forProduct22( - "accountId", - "resourceId", - "stage", - "requestId", - "identity", - "ResourcePath", - "authorizer", - "httpMethod", - "apiId", - "connectedAt", - "connectionId", - "domainName", - "error", - "eventType", - "extendedRequestId", - "integrationLatency", - "messageDirection", - "messageId", - "requestTime", - "requestTimeEpoch", - "routeKey", - "status" - )(WebSocketRequestContext.apply) - - private final case class Impl( - accountId: String, - resourceId: String, - stage: String, - requestId: String, - identity: WebSocketRequestIdentity, - ResourcePath: String, - authorizer: Map[String, Object], - httpMethod: String, - apiId: String, - connectedAt: Long, - connectionId: String, - domainName: String, - error: String, - eventType: String, - extendedRequestId: String, - integrationLatency: String, - messageDirection: String, - messageId: String, - requestTime: String, - requestTimeEpoch: Long, - routeKey: String, - status: String - ) extends WebSocketRequestContext { - override def productPrefix = "WebSocketRequestContext" - } -} \ No newline at end of file +/* + * Copyright 2021 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package feral.lambda.events + +import io.circe.Decoder + +sealed abstract class APIGatewayV2WebSocketEvent { + def stageVariables: Map[String, String] + def requestContext: WebSocketRequestContext + def body: String + def isBase64Encoded: Boolean +} + +object APIGatewayV2WebSocketEvent { + def apply( + stageVariables: Map[String, String], + requestContext: WebSocketRequestContext, + body: String, + isBase64Encoded: Boolean + ): APIGatewayV2WebSocketEvent = + new Impl( + stageVariables, + requestContext, + body, + isBase64Encoded + ) + + implicit val decoder: Decoder[APIGatewayV2WebSocketEvent] = Decoder.forProduct4( + "stageVariables", + "requestContext", + "body", + "isBase64Encoded" + )(APIGatewayV2WebSocketEvent.apply) + + private final case class Impl( + stageVariables: Map[String, String], + requestContext: WebSocketRequestContext, + body: String, + isBase64Encoded: Boolean + ) extends APIGatewayV2WebSocketEvent { + override def productPrefix = "APIGatewayV2WebSocketEvent" + } +} + +sealed abstract class WebSocketRequestContext { + def stage: String + def requestId: String + def apiId: String + def connectedAt: Long + def connectionId: String + def domainName: String + def eventType: String + def extendedRequestId: String + def messageDirection: String + def messageId: String + def requestTime: String + def requestTimeEpoch: Long + def routeKey: String +} + +object WebSocketRequestContext { + def apply( + stage: String, + requestId: String, + apiId: String, + connectedAt: Long, + connectionId: String, + domainName: String, + eventType: String, + extendedRequestId: String, + messageDirection: String, + messageId: String, + requestTime: String, + requestTimeEpoch: Long, + routeKey: String + ): WebSocketRequestContext = + new Impl( + stage, + requestId, + apiId, + connectedAt, + connectionId, + domainName, + eventType, + extendedRequestId, + messageDirection, + messageId, + requestTime, + requestTimeEpoch, + routeKey + ) + + private[events] implicit val decoder: Decoder[WebSocketRequestContext] = Decoder.forProduct13( + "stage", + "requestId", + "apiId", + "connectedAt", + "connectionId", + "domainName", + "eventType", + "extendedRequestId", + "messageDirection", + "messageId", + "requestTime", + "requestTimeEpoch", + "routeKey" + )(WebSocketRequestContext.apply) + + private final case class Impl( + stage: String, + requestId: String, + apiId: String, + connectedAt: Long, + connectionId: String, + domainName: String, + eventType: String, + extendedRequestId: String, + messageDirection: String, + messageId: String, + requestTime: String, + requestTimeEpoch: Long, + routeKey: String + ) extends WebSocketRequestContext { + override def productPrefix = "WebSocketRequestContext" + } +} From 1f997d5b77aa3296c2a109b43d788035da837010 Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Tue, 18 Jun 2024 19:48:08 +0800 Subject: [PATCH 06/10] Update lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala Co-authored-by: Arman Bilge --- .../scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala index 77370e4b..110adf9a 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -18,7 +18,7 @@ package feral.lambda.events import io.circe.Decoder -sealed abstract class APIGatewayV2WebSocketEvent { +sealed abstract class ApiGatewayV2WebSocketEvent { def stageVariables: Map[String, String] def requestContext: WebSocketRequestContext def body: String From be00d2113b19df709858a112027681b701682ca6 Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Tue, 18 Jun 2024 23:22:32 +0800 Subject: [PATCH 07/10] Modified types to be more specific, added tests --- .../events/APIGatewayV2WebSocketEvent.scala | 87 ++++++----- .../scala/feral/lambda/events/codecs.scala | 4 + .../ApiGatewayV2WebSocketEventSuite.scala | 143 ++++++++++++++++++ 3 files changed, 196 insertions(+), 38 deletions(-) create mode 100644 lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala index 110adf9a..f56ea019 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala @@ -16,22 +16,28 @@ package feral.lambda.events +import com.comcast.ip4s.Hostname import io.circe.Decoder +import java.time.Instant + +import codecs.decodeInstant +import codecs.decodeHostname + sealed abstract class ApiGatewayV2WebSocketEvent { - def stageVariables: Map[String, String] + def stageVariables: Option[Map[String, String]] def requestContext: WebSocketRequestContext - def body: String + def body: Option[String] def isBase64Encoded: Boolean } -object APIGatewayV2WebSocketEvent { +object ApiGatewayV2WebSocketEvent { def apply( - stageVariables: Map[String, String], + stageVariables: Option[Map[String, String]], requestContext: WebSocketRequestContext, - body: String, + body: Option[String], isBase64Encoded: Boolean - ): APIGatewayV2WebSocketEvent = + ): ApiGatewayV2WebSocketEvent = new Impl( stageVariables, requestContext, @@ -39,36 +45,49 @@ object APIGatewayV2WebSocketEvent { isBase64Encoded ) - implicit val decoder: Decoder[APIGatewayV2WebSocketEvent] = Decoder.forProduct4( + implicit val decoder: Decoder[ApiGatewayV2WebSocketEvent] = Decoder.forProduct4( "stageVariables", "requestContext", "body", "isBase64Encoded" - )(APIGatewayV2WebSocketEvent.apply) + )(ApiGatewayV2WebSocketEvent.apply) private final case class Impl( - stageVariables: Map[String, String], + stageVariables: Option[Map[String, String]], requestContext: WebSocketRequestContext, - body: String, + body: Option[String], isBase64Encoded: Boolean - ) extends APIGatewayV2WebSocketEvent { - override def productPrefix = "APIGatewayV2WebSocketEvent" + ) extends ApiGatewayV2WebSocketEvent { + override def productPrefix = "ApiGatewayV2WebSocketEvent" } } +sealed abstract class WebSocketEventType + +object WebSocketEventType { + case object Connect extends WebSocketEventType + case object Message extends WebSocketEventType + case object Disconnect extends WebSocketEventType + + private[events] implicit val decoder: Decoder[WebSocketEventType] = + Decoder.decodeString.map { + case "CONNECT" => Connect + case "MESSAGE" => Message + case "DISCONNECT" => Disconnect + } +} + sealed abstract class WebSocketRequestContext { def stage: String def requestId: String def apiId: String - def connectedAt: Long + def connectedAt: Instant def connectionId: String - def domainName: String - def eventType: String + def domainName: Hostname + def eventType: WebSocketEventType def extendedRequestId: String - def messageDirection: String - def messageId: String - def requestTime: String - def requestTimeEpoch: Long + def messageId: Option[String] + def requestTime: Instant def routeKey: String } @@ -77,15 +96,13 @@ object WebSocketRequestContext { stage: String, requestId: String, apiId: String, - connectedAt: Long, + connectedAt: Instant, connectionId: String, - domainName: String, - eventType: String, + domainName: Hostname, + eventType: WebSocketEventType, extendedRequestId: String, - messageDirection: String, - messageId: String, - requestTime: String, - requestTimeEpoch: Long, + messageId: Option[String], + requestTime: Instant, routeKey: String ): WebSocketRequestContext = new Impl( @@ -97,14 +114,12 @@ object WebSocketRequestContext { domainName, eventType, extendedRequestId, - messageDirection, messageId, requestTime, - requestTimeEpoch, routeKey ) - private[events] implicit val decoder: Decoder[WebSocketRequestContext] = Decoder.forProduct13( + private[events] implicit val decoder: Decoder[WebSocketRequestContext] = Decoder.forProduct11( "stage", "requestId", "apiId", @@ -113,9 +128,7 @@ object WebSocketRequestContext { "domainName", "eventType", "extendedRequestId", - "messageDirection", "messageId", - "requestTime", "requestTimeEpoch", "routeKey" )(WebSocketRequestContext.apply) @@ -124,15 +137,13 @@ object WebSocketRequestContext { stage: String, requestId: String, apiId: String, - connectedAt: Long, + connectedAt: Instant, connectionId: String, - domainName: String, - eventType: String, + domainName: Hostname, + eventType: WebSocketEventType, extendedRequestId: String, - messageDirection: String, - messageId: String, - requestTime: String, - requestTimeEpoch: Long, + messageId: Option[String], + requestTime: Instant, routeKey: String ) extends WebSocketRequestContext { override def productPrefix = "WebSocketRequestContext" diff --git a/lambda/shared/src/main/scala/feral/lambda/events/codecs.scala b/lambda/shared/src/main/scala/feral/lambda/events/codecs.scala index cee02ea1..44713caf 100644 --- a/lambda/shared/src/main/scala/feral/lambda/events/codecs.scala +++ b/lambda/shared/src/main/scala/feral/lambda/events/codecs.scala @@ -16,6 +16,7 @@ package feral.lambda.events +import com.comcast.ip4s.Hostname import com.comcast.ip4s.IpAddress import io.circe.Decoder import io.circe.KeyDecoder @@ -40,6 +41,9 @@ private object codecs { implicit def decodeIpAddress: Decoder[IpAddress] = Decoder.decodeString.emap(IpAddress.fromString(_).toRight("Cannot parse IP address")) + implicit def decodeHostname: Decoder[Hostname] = + Decoder.decodeString.emap(Hostname.fromString(_).toRight("Cannot parse hostname")) + implicit def decodeKeyCIString: KeyDecoder[CIString] = KeyDecoder.decodeKeyString.map(CIString(_)) diff --git a/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala b/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala new file mode 100644 index 00000000..718d7909 --- /dev/null +++ b/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala @@ -0,0 +1,143 @@ +package feral.lambda.events + +/* + * Copyright 2021 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package feral.lambda.events + +import io.circe.literal._ +import munit.FunSuite + +class ApiGatewayV2WebSocketEventSuite extends FunSuite { + + import ApiGatewayV2WebSocketEventSuite._ + + test("decode connect") { + connectEvent.as[ApiGatewayV2WebSocketEvent].toTry.get + } + + test("decode disconnect") { + disconnectEvent.as[ApiGatewayV2WebSocketEvent].toTry.get + } + +} + +object ApiGatewayV2WebSocketEventSuite { + + def connectEvent = json""" + { + "headers": { + "Host": "abcd123.execute-api.us-east-1.amazonaws.com", + "Sec-WebSocket-Extensions": "permessage-deflate; client_max_window_bits", + "Sec-WebSocket-Key": "...", + "Sec-WebSocket-Version": "13", + "X-Amzn-Trace-Id": "...", + "X-Forwarded-For": "192.0.2.1", + "X-Forwarded-Port": "443", + "X-Forwarded-Proto": "https" + }, + "multiValueHeaders": { + "Host": [ + "abcd123.execute-api.us-east-1.amazonaws.com" + ], + "Sec-WebSocket-Extensions": [ + "permessage-deflate; client_max_window_bits" + ], + "Sec-WebSocket-Key": [ + "..." + ], + "Sec-WebSocket-Version": [ + "13" + ], + "X-Amzn-Trace-Id": [ + "..." + ], + "X-Forwarded-For": [ + "192.0.2.1" + ], + "X-Forwarded-Port": [ + "443" + ], + "X-Forwarded-Proto": [ + "https" + ] + }, + "requestContext": { + "routeKey": "$$connect", + "eventType": "CONNECT", + "extendedRequestId": "ABCD1234=", + "requestTime": "09/Feb/2024:18:11:43 +0000", + "messageDirection": "IN", + "stage": "prod", + "connectedAt": 1707502303419, + "requestTimeEpoch": 1707502303420, + "identity": { + "sourceIp": "192.0.2.1" + }, + "requestId": "ABCD1234=", + "domainName": "abcd1234.execute-api.us-east-1.amazonaws.com", + "connectionId": "AAAA1234=", + "apiId": "abcd1234" + }, + "isBase64Encoded": false +} + """ + def disconnectEvent = json""" + { + "headers": { + "Host": "abcd1234.execute-api.us-east-1.amazonaws.com", + "x-api-key": "", + "X-Forwarded-For": "", + "x-restapi": "" + }, + "multiValueHeaders": { + "Host": [ + "abcd1234.execute-api.us-east-1.amazonaws.com" + ], + "x-api-key": [ + "" + ], + "X-Forwarded-For": [ + "" + ], + "x-restapi": [ + "" + ] + }, + "requestContext": { + "routeKey": "$$disconnect", + "disconnectStatusCode": 1005, + "eventType": "DISCONNECT", + "extendedRequestId": "ABCD1234=", + "requestTime": "09/Feb/2024:18:23:28 +0000", + "messageDirection": "IN", + "disconnectReason": "Client-side close frame status not set", + "stage": "prod", + "connectedAt": 1707503007396, + "requestTimeEpoch": 1707503008941, + "identity": { + "sourceIp": "192.0.2.1" + }, + "requestId": "ABCD1234=", + "domainName": "abcd1234.execute-api.us-east-1.amazonaws.com", + "connectionId": "AAAA1234=", + "apiId": "abcd1234" + }, + "isBase64Encoded": false +} + """ +} + From 902562679edfa12edce5885afa337934aff0e86b Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Wed, 19 Jun 2024 02:02:20 +0800 Subject: [PATCH 08/10] Removed an extra line --- .../feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala | 2 -- 1 file changed, 2 deletions(-) diff --git a/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala b/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala index 718d7909..def32b29 100644 --- a/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala +++ b/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala @@ -1,5 +1,3 @@ -package feral.lambda.events - /* * Copyright 2021 Typelevel * From dbef82222e6bffce9221cac45dbba29beb8c3add Mon Sep 17 00:00:00 2001 From: CCH <31771180+Chingles2404@users.noreply.github.com> Date: Wed, 19 Jun 2024 02:07:35 +0800 Subject: [PATCH 09/10] Ran scalafmt --- .../feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala | 1 - 1 file changed, 1 deletion(-) diff --git a/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala b/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala index def32b29..48b9b39f 100644 --- a/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala +++ b/lambda/shared/src/test/scala/feral/lambda/events/ApiGatewayV2WebSocketEventSuite.scala @@ -138,4 +138,3 @@ object ApiGatewayV2WebSocketEventSuite { } """ } - From 771006fd7511e37720a93fdc8f7f975a45afbdfd Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Mon, 8 Jul 2024 14:24:40 -0700 Subject: [PATCH 10/10] Rename APIGatewayV2WebSocketEvent.scala to ApiGatewayV2WebSocketEvent.scala --- ...wayV2WebSocketEvent.scala => ApiGatewayV2WebSocketEvent.scala} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename lambda/shared/src/main/scala/feral/lambda/events/{APIGatewayV2WebSocketEvent.scala => ApiGatewayV2WebSocketEvent.scala} (100%) diff --git a/lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala b/lambda/shared/src/main/scala/feral/lambda/events/ApiGatewayV2WebSocketEvent.scala similarity index 100% rename from lambda/shared/src/main/scala/feral/lambda/events/APIGatewayV2WebSocketEvent.scala rename to lambda/shared/src/main/scala/feral/lambda/events/ApiGatewayV2WebSocketEvent.scala