Skip to content

Commit

Permalink
Merge pull request #476 from Chingles2404/APIGatewayV2WebSocketEvent
Browse files Browse the repository at this point in the history
Add `ApiGatewayV2WebSocketEvent`
  • Loading branch information
armanbilge authored Jul 9, 2024
2 parents 6560650 + 771006f commit 07b0557
Show file tree
Hide file tree
Showing 3 changed files with 295 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,151 @@
/*
* 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 com.comcast.ip4s.Hostname
import io.circe.Decoder

import java.time.Instant

import codecs.decodeInstant
import codecs.decodeHostname

sealed abstract class ApiGatewayV2WebSocketEvent {
def stageVariables: Option[Map[String, String]]
def requestContext: WebSocketRequestContext
def body: Option[String]
def isBase64Encoded: Boolean
}

object ApiGatewayV2WebSocketEvent {
def apply(
stageVariables: Option[Map[String, String]],
requestContext: WebSocketRequestContext,
body: Option[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: Option[Map[String, String]],
requestContext: WebSocketRequestContext,
body: Option[String],
isBase64Encoded: Boolean
) 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: Instant
def connectionId: String
def domainName: Hostname
def eventType: WebSocketEventType
def extendedRequestId: String
def messageId: Option[String]
def requestTime: Instant
def routeKey: String
}

object WebSocketRequestContext {
def apply(
stage: String,
requestId: String,
apiId: String,
connectedAt: Instant,
connectionId: String,
domainName: Hostname,
eventType: WebSocketEventType,
extendedRequestId: String,
messageId: Option[String],
requestTime: Instant,
routeKey: String
): WebSocketRequestContext =
new Impl(
stage,
requestId,
apiId,
connectedAt,
connectionId,
domainName,
eventType,
extendedRequestId,
messageId,
requestTime,
routeKey
)

private[events] implicit val decoder: Decoder[WebSocketRequestContext] = Decoder.forProduct11(
"stage",
"requestId",
"apiId",
"connectedAt",
"connectionId",
"domainName",
"eventType",
"extendedRequestId",
"messageId",
"requestTimeEpoch",
"routeKey"
)(WebSocketRequestContext.apply)

private final case class Impl(
stage: String,
requestId: String,
apiId: String,
connectedAt: Instant,
connectionId: String,
domainName: Hostname,
eventType: WebSocketEventType,
extendedRequestId: String,
messageId: Option[String],
requestTime: Instant,
routeKey: String
) extends WebSocketRequestContext {
override def productPrefix = "WebSocketRequestContext"
}
}
4 changes: 4 additions & 0 deletions lambda/shared/src/main/scala/feral/lambda/events/codecs.scala
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand All @@ -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(_))

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +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.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
}
"""
}

0 comments on commit 07b0557

Please sign in to comment.