Awakeables
Awakeables pause an invocation while waiting for another process to complete a task. You can use this pattern to let a handler execute a task somewhere else and retrieve the result. This pattern is also known as the callback (task token) pattern.
Creating awakeables
- Java
- Kotlin
- The handler creates an awakeable. This contains a String identifier and a Promise/Awaitable.
Awakeable<String> awakeable = ctx.awakeable(JsonSerdes.STRING);String awakeableId = awakeable.id();ctx.run(() -> triggerTaskAndDeliverId(awakeableId));String payload = awakeable.await();
- The handler triggers a task/process and attaches the awakeable ID (e.g. over Kafka, via HTTP,...).
For example, send an HTTP request to a service that executes the task, and attach the ID to the payload.
You use
ctx.run
to avoid re-triggering the task on retries.
Awakeable<String> awakeable = ctx.awakeable(JsonSerdes.STRING);String awakeableId = awakeable.id();ctx.run(() -> triggerTaskAndDeliverId(awakeableId));String payload = awakeable.await();
- The handler waits until the other process has executed the task. The handler receives the payload and resumes.
Awakeable<String> awakeable = ctx.awakeable(JsonSerdes.STRING);String awakeableId = awakeable.id();ctx.run(() -> triggerTaskAndDeliverId(awakeableId));String payload = awakeable.await();
- The handler creates an awakeable. This contains a String identifier and a Promise/Awaitable.
- The handler triggers a task/process and attaches the awakeable ID (e.g. over Kafka, via HTTP,...).
For example, send an HTTP request to a service that executes the task, and attach the ID to the payload.
You use
ctx.run
to avoid re-triggering the task on retries.
- The handler waits until the other process has executed the task. The handler receives the payload and resumes.
Awakeable<String> awakeable = ctx.awakeable(JsonSerdes.STRING);String awakeableId = awakeable.id();ctx.run(() -> triggerTaskAndDeliverId(awakeableId));String payload = awakeable.await();
- The handler creates an awakeable. This contains a String identifier and a Promise/Awaitable.
val awakeable = ctx.awakeable<String>()val awakeableId: String = awakeable.idctx.runBlock{ triggerTaskAndDeliverId(awakeableId) }val payload: String = awakeable.await()
- The handler triggers a task/process and attaches the awakeable ID (e.g. over Kafka, via HTTP,...).
For example, send an HTTP request to a service that executes the task, and attach the ID to the payload.
You use
ctx.runBlock
to avoid re-triggering the task on retries.
val awakeable = ctx.awakeable<String>()val awakeableId: String = awakeable.idctx.runBlock{ triggerTaskAndDeliverId(awakeableId) }val payload: String = awakeable.await()
- The handler waits until the other process has executed the task. The handler receives the payload and resumes.
val awakeable = ctx.awakeable<String>()val awakeableId: String = awakeable.idctx.runBlock{ triggerTaskAndDeliverId(awakeableId) }val payload: String = awakeable.await()
- The handler creates an awakeable. This contains a String identifier and a Promise/Awaitable.
- The handler triggers a task/process and attaches the awakeable ID (e.g. over Kafka, via HTTP,...).
For example, send an HTTP request to a service that executes the task, and attach the ID to the payload.
You use
ctx.runBlock
to avoid re-triggering the task on retries.
- The handler waits until the other process has executed the task. The handler receives the payload and resumes.
val awakeable = ctx.awakeable<String>()val awakeableId: String = awakeable.idctx.runBlock{ triggerTaskAndDeliverId(awakeableId) }val payload: String = awakeable.await()
Completing awakeables
The external process completes the awakeable by either resolving it with an optional payload or by rejecting it with its ID and a reason for the failure. This throws a terminal error in the waiting handler.
- Java
- Kotlin
- Resolving over HTTP with its ID and an optional payload:
curl localhost:8080/restate/awakeables/prom_1PePOqp/resolve -H 'content-type: application/json' -d '{"hello": "world"}'
- Resolving over HTTP with its ID and an optional payload:
curl localhost:8080/restate/awakeables/prom_1PePOqp/resolve -H 'content-type: application/json' -d '{"hello": "world"}'
- Rejecting over HTTP with its ID and a reason:
curl localhost:8080/restate/awakeables/prom_1PePOqp/reject -H 'content-type: text/plain' \ -d 'Very bad error!'
- Rejecting over HTTP with its ID and a reason:
curl localhost:8080/restate/awakeables/prom_1PePOqp/reject -H 'content-type: text/plain' \ -d 'Very bad error!'
- Resolving via the SDK with its ID and an optional payload:
ctx.awakeableHandle(awakeableId).resolve(JsonSerdes.STRING, "hello");
- Resolving via the SDK with its ID and an optional payload:
ctx.awakeableHandle(awakeableId).resolve(JsonSerdes.STRING, "hello");
- Rejecting via the SDK with its ID and a reason:
ctx.awakeableHandle(awakeableId).reject("my error reason");
- Rejecting via the SDK with its ID and a reason:
ctx.awakeableHandle(awakeableId).reject("my error reason");
For primitive types, you can use the Restate SDK's CoreSerdes
.
For other types, have a look at the serialization docs.
- Resolving over HTTP with its ID and an optional payload:
curl localhost:8080/restate/awakeables/prom_1PePOqp/resolve -H 'content-type: application/json' -d '{"hello": "world"}'
- Resolving over HTTP with its ID and an optional payload:
curl localhost:8080/restate/awakeables/prom_1PePOqp/resolve -H 'content-type: application/json' -d '{"hello": "world"}'
- Rejecting over HTTP with its ID and a reason:
curl localhost:8080/restate/awakeables/prom_1PePOqp/reject -H 'content-type: text/plain' \ -d 'Very bad error!'
- Rejecting over HTTP with its ID and a reason:
curl localhost:8080/restate/awakeables/prom_1PePOqp/reject -H 'content-type: text/plain' \ -d 'Very bad error!'
- Resolving via the SDK with its ID and an optional payload:
ctx.awakeableHandle(awakeableId) .resolve("hello")
- Resolving via the SDK with its ID and an optional payload:
ctx.awakeableHandle(awakeableId) .resolve("hello")
- Rejecting via the SDK with its ID and a reason:
ctx.awakeableHandle(awakeableId) .reject("my error reason")
- Rejecting via the SDK with its ID and a reason:
ctx.awakeableHandle(awakeableId) .reject("my error reason")
By default JSON is used to serialize payloads, using Kotlin serialization. For other types, have a look at the serialization docs.
When running on Function-as-a-Service platforms, such as AWS Lambda, Restate suspends the handler while waiting for the awakeable to be completed. Since you only pay for the time that the handler is actually running, your don't pay while waiting for the external process to return.
Virtual Objects only process a single invocation at a time, so the Virtual Object will be blocked while waiting on the awakeable to be resolved.