From f4fc65311c146e2a7397486c7e40fe16efb34e1d Mon Sep 17 00:00:00 2001 From: Maciej Winnicki Date: Mon, 26 Mar 2018 09:45:44 +0200 Subject: [PATCH 1/4] Add Reliablity Guarantees section in README --- README.md | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/README.md b/README.md index 07cff2d..1d3adba 100644 --- a/README.md +++ b/README.md @@ -42,6 +42,7 @@ yet ready for production applications._ 1. [Comparison](#comparison) 1. [Architecture](#architecture) 1. [System Overview](#system-overview) + 1. [Reliability Guarantees](#reliability-guarantees) 1. [Clustering](#clustering) 1. [Background](#background) @@ -842,6 +843,18 @@ directly. └────────────────────────────────────────────────────────────────────────────────────────────────────────────────┘ ``` +## Reliability Guarantees + +### Events are not durable + +The event received by Event Gateway is stored only in memory, it's not persisted to disk during processing. This means that in case of hardware failure or software crash the event may not be delivered to the subscriber. For a synchronous subscription (`http` or `invoke` event) it can manifest by error message returned to the requester. If there is multiple subscribes to the same custom event type, in case of failure the event may not be delivered to all of them. + +### Events are delivered _at most once_ + +Event Gateway tries to deliver an event only once, there is no retry mechanism. This and lack of durability implicates that event will be delivered to the subscriber _at most once_. Even though Event Gateway itself doesn't retry failed function invocations, AWS SDK used by few providers does that internally by default. The retry logic there happens only for very specific reasons and should not cause delivering the same event multiple times. Please find more information in [AWS documentation on API retries](https://docs.aws.amazon.com/general/latest/gr/api-retries.html). + +AWS Lambda provider uses `RequestResponse` invocation type which means that retry logic for asynchronous AWS events doesn't apply here. Among others it means, that failed deliveries of custom events are not sent to DLQ. Please find more information in [Understanding Retry Behavior](https://docs.aws.amazon.com/lambda/latest/dg/retries-on-errors.html), "Synchronous invocation" section. + ## Background SOA came along with a new set of challenges. In monolithic architectures, it was simple to call a built-in library or From 87e2c6e8767d6485cfcc171bda2d297c45abe31a Mon Sep 17 00:00:00 2001 From: Maciej Winnicki Date: Wed, 28 Mar 2018 14:11:23 +0200 Subject: [PATCH 2/4] update --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 1d3adba..72c206a 100644 --- a/README.md +++ b/README.md @@ -851,7 +851,7 @@ The event received by Event Gateway is stored only in memory, it's not persisted ### Events are delivered _at most once_ -Event Gateway tries to deliver an event only once, there is no retry mechanism. This and lack of durability implicates that event will be delivered to the subscriber _at most once_. Even though Event Gateway itself doesn't retry failed function invocations, AWS SDK used by few providers does that internally by default. The retry logic there happens only for very specific reasons and should not cause delivering the same event multiple times. Please find more information in [AWS documentation on API retries](https://docs.aws.amazon.com/general/latest/gr/api-retries.html). +Event Gateway attempts delivery fulfillment for an event only once and consequently any event received successfully by the Event Gateway is guaranteed to be received by the subscriber _at most once_. That said, the nature of Event Gateway provider implementation could result in retries under specific circumstances, but these should not cause delivering the same event multiple times. For example, Providers for AWS Services that use the AWS SDK are subject to auto retry logic that's built into the SDK ([AWS documentation on API retries](https://docs.aws.amazon.com/general/latest/gr/api-retries.html)). AWS Lambda provider uses `RequestResponse` invocation type which means that retry logic for asynchronous AWS events doesn't apply here. Among others it means, that failed deliveries of custom events are not sent to DLQ. Please find more information in [Understanding Retry Behavior](https://docs.aws.amazon.com/lambda/latest/dg/retries-on-errors.html), "Synchronous invocation" section. From 943e9fba21f86294eb1cd28a896a2390e86e885b Mon Sep 17 00:00:00 2001 From: Maciej Winnicki Date: Wed, 28 Mar 2018 23:50:07 +0200 Subject: [PATCH 3/4] rewrite the sentence --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 72c206a..f7aca91 100644 --- a/README.md +++ b/README.md @@ -847,7 +847,7 @@ directly. ### Events are not durable -The event received by Event Gateway is stored only in memory, it's not persisted to disk during processing. This means that in case of hardware failure or software crash the event may not be delivered to the subscriber. For a synchronous subscription (`http` or `invoke` event) it can manifest by error message returned to the requester. If there is multiple subscribes to the same custom event type, in case of failure the event may not be delivered to all of them. +The event received by Event Gateway is stored only in memory, it's not persisted to disk before processing. This means that in case of hardware failure or software crash the event may not be delivered to the subscriber. For a synchronous subscription (`http` or `invoke` event) it can manifest as error message returned to the requester. For asynchronous custom event with multiple subscribers it means that the event may not be delivered to all of the subscribers. ### Events are delivered _at most once_ From a2097bc7923d5ecbdf2df8bd5f303a48aef6ec06 Mon Sep 17 00:00:00 2001 From: Maciej Winnicki Date: Thu, 29 Mar 2018 11:35:44 +0200 Subject: [PATCH 4/4] add documentation about error responses from Events API --- README.md | 23 +++++++++++++---------- docs/openapi/openapi-events-api.yaml | 8 ++++++-- 2 files changed, 19 insertions(+), 12 deletions(-) diff --git a/README.md b/README.md index f7aca91..57c1674 100644 --- a/README.md +++ b/README.md @@ -59,7 +59,7 @@ using the Event Gateway locally. There is a [official Docker image](https://hub.docker.com/r/serverless/event-gateway/). -``` +```bash docker run -p 4000:4000 -p 4001:4001 serverless/event-gateway -dev ``` @@ -96,7 +96,7 @@ event. ##### curl example -```http +```bash curl --request POST \ --url http://localhost:4001/v1/spaces/default/functions \ --header 'content-type: application/json' \ @@ -128,7 +128,7 @@ eventGateway.registerFunction({ ##### curl example -```http +```bash curl --request POST \ --url http://localhost:4000/ \ --header 'content-type: application/json' \ @@ -163,7 +163,7 @@ path property indicated URL path which Events API will be listening on. ##### curl example -```http +```bash curl --request POST \ --url http://localhost:4001/v1/spaces/default/subscriptions \ --header 'content-type: application/json' \ @@ -191,7 +191,7 @@ eventGateway.subscribe({ ##### curl example -```http +```bash curl --request POST \ --url http://localhost:4000/ \ --header 'content-type: application/json' \ @@ -219,7 +219,7 @@ only one `http` subscription for the same `method` and `path` pair. ##### curl example -```http +```bash curl --request POST \ --url http://localhost:4001/v1/spaces/default/subscriptions \ --header 'content-type: application/json' \ @@ -249,7 +249,7 @@ eventGateway.subscribe({ One additional concept in the Event Gateway are Spaces. Spaces provide isolation between resources. Space is a coarse-grained sandbox in which entities (Functions and Subscriptions) can interact freely. All actions are -possible within a space: publishing, subscribing and invoking. All access cross-space is disabled. +possible within a space: publishing, subscribing and invoking. Space is not about access control/authentication/authorization. It's only about isolation. It doesn't enforce any specific subscription path. @@ -356,9 +356,10 @@ arbitrary payload, subscribed function receives an event in [HTTP Event](#http-e **Response** -Status code: +HTTP subscription response depends on [response object](#respond-to-an-http-event) returned by the backing function. In case of failure during function invocation following error response are possible: -* `200 OK` with payload with function response +* `404 Not Found` if there is no backing function registered for requested HTTP endpoint +* `500 Internal Server Error` if the function invocation failed or the backing function didn't return [HTTP response object](#respond-to-an-http-event) ##### CORS @@ -416,7 +417,9 @@ arbitrary payload, invoked function receives an event in above schema, where req Status code: -* `200 OK` with payload with function response +* `200 OK` with payload returned by invoked function +* `404 Not Found` if there is no function registered or `invoke` subscription created for requested function +* `500 Internal Server Error` if the function invocation failed ### CORS diff --git a/docs/openapi/openapi-events-api.yaml b/docs/openapi/openapi-events-api.yaml index 597e291..564c35a 100644 --- a/docs/openapi/openapi-events-api.yaml +++ b/docs/openapi/openapi-events-api.yaml @@ -71,5 +71,9 @@ paths: type: string default: default responses: - 202: - description: "event accepted" \ No newline at end of file + 200: + description: "function invoked successfully, response body contains payload returned by the function" + 404: + description: "no function registered or 'invoke' subscription created for requested function" + 500: + description: "function invocation failed" \ No newline at end of file