synopsis | status |
---|---|
CAP provides intrinsic support for emitting and receiving events. This is complemented by Messaging Services connecting to message brokers to exchange event messages across remote services.
|
released |
{{ $frontmatter.synopsis }}
[[toc]]
We're starting with an introduction to the core concepts in CAP. If you want to skip the introduction, you can fast-forward to the samples part starting at Books Reviews Sample.
As introduced in About CAP, everything happening at runtime is in response to events, and all service implementations take place in event handlers. All CAP services intrinsically support emitting and reacting to events, as shown in this simple code snippet (you can copy & run it in cds repl
):
let srv = new cds.Service
// Receiving Events
srv.on ('some event', msg => console.log('1st listener received:', msg))
srv.on ('some event', msg => console.log('2nd listener received:', msg))
// Emitting Events
await srv.emit ('some event', { foo:11, bar:'12' })
::: tip Intrinsic support for events The core of CAP's processing model: all services are event emitters. Events can be sent to them, emitted by them, and event handlers register with them to react to such events. :::
In contrast to the previous code sample, emitters and receivers of events are decoupled, in different services and processes. And as all active things in CAP are services, so are usually emitters and receivers of events. Typical patterns look like that:
class Emitter extends cds.Service { async someMethod() {
// inform unknown receivers about something happened
await this.emit ('some event', { some:'payload' })
}}
class Receiver extends cds.Service { async init() {
// connect to and register for events from Emitter
const Emitter = await cds.connect.to('Emitter')
Emitter.on ('some event', msg => {...})
}}
::: tip Emitters usually emit messages to themselves to inform potential listeners about certain events. Receivers connect to Emitters to register handlers to such emitted events. :::
A Request in CAP is actually a specialization of an Event Message. The same intrinsic mechanisms of sending and reacting to events are used for asynchronous communication in inverse order. A typical flow:
Asynchronous communication looks similar, just with reversed roles:
::: tip Event Listeners vs Interceptors
Requests are handled the same ways as events, with one major difference: While on
handlers for events are listeners (all are called), handlers for synchronous requests are interceptors (only the topmost is called by the framework). An interceptor then decides whether to pass down control to next
handlers or not.
:::
To sum up, handling events in CAP is done in the same way as you would handle requests in a service provider. Also, emitting event messages is similar to sending requests. The major difference is that the initiative is inverted: While Consumers connect to Services in synchronous communications, the Receivers connect to Emitters in asynchronous ones; Emitters in turn don't know Receivers.
::: tip Blurring the line between synchronous and asynchronous API In essence, services receive events. The emitting service itself or other services can register handlers for those events in order to implement the logic of how to react to these events. :::
Using messaging has two major advantages:
::: tip Resilience If a receiving service goes offline for a while, event messages are safely stored, and guaranteed to be delivered to the receiver as soon as it goes online again. :::
::: tip Decoupling Emitters of event messages are decoupled from the receivers and don't need to know them at the time of sending. This way a service is able to emit events that other services can register on in the future, for example, to implement extension points. :::
The following explanations walk us through a books review example from cap/samples:
- @capire/bookshop provides the well-known basic bookshop app.
- @capire/reviews provides an independent service to manage reviews.
- @capire/bookstore combines both into a composite application.
::: tip Follow the instructions in cap/samples/readme for getting the samples and exercising the following steps. :::
Package @capire/reviews
essentially provides a ReviewsService
, declared like that:
service ReviewsService {
// Sync API
entity Reviews as projection on my.Reviews excluding { likes }
action like (review: Reviews:ID);
action unlike (review: Reviews:ID);
// Async API
event reviewed : {
subject : Reviews:subject;
count : Integer;
rating : Decimal; // new avg rating
}
}
Learn more about declaring events in CDS.{.learn-more}
As you can read from the definitions, the service's synchronous API allows to create, read, update, and delete user Reviews
for arbitrary review subjects. In addition, the service's asynchronous API declares the reviewed
event that shall be emitted whenever a subject's average rating changes.
::: tip Services in CAP combine synchronous and asynchronous APIs. Events are declared on conceptual level focusing on domain, instead of low-level wire protocols. :::
Find the code to emit events in @capire/reviews/srv/reviews-service.js:
class ReviewsService extends cds.ApplicationService { async init() {
// Emit a `reviewed` event whenever a subject's avg rating changes
this.after (['CREATE','UPDATE','DELETE'], 'Reviews', (req) => {
let { subject } = req.data, count, rating //= ...
return this.emit ('reviewed', { subject, count, rating })
})
}}
Learn more about srv.emit()
in Node.js.{.learn-more}
Learn more about srv.emit()
in Java.{.learn-more}
Method srv.emit()
is used to emit event messages. As you can see, emitters usually emit messages to themselves, that is, this
, to inform potential listeners about certain events. Emitters don't know the receivers of the events they emit. There might be none, there might be local ones in the same process, or remote ones in separate processes.
::: tip Messaging on Conceptual Level
Simply use srv.emit()
to emit events, and let the CAP framework care for wire protocols like CloudEvents, transports via message brokers, multitenancy handling, and so forth.
:::
Find the code to receive events in @capire/bookstore/srv/mashup.js (which is the basic bookshop app enhanced by reviews, hence integration with ReviewsService
):
// Update Books' average ratings when reviews are updated
ReviewsService.on ('reviewed', (msg) => {
const { subject, count, rating } = msg.data
// ...
})
Learn more about registering event handlers in Node.js.{.learn-more} Learn more about registering event handlers in Java.{.learn-more}
The message payload is in the data
property of the inbound msg
object.
As emitting and handling events is an intrinsic feature of the CAP core runtimes, there's nothing else required when emitters and receivers live in the same process.
Let's see that in action...
Run the following command to start a reviews-enhanced bookshop as an all-in-one server process:
cds watch bookstore
It produces a trace output like that:
[cds] - mocking ReviewsService { path: '/reviews', impl: '../reviews/srv/reviews-service.js' }
[cds] - mocking OrdersService { path: '/orders', impl: '../orders/srv/orders-service.js' }
[cds] - serving CatalogService { path: '/browse', impl: '../bookshop/srv/cat-service.js' }
[cds] - serving AdminService { path: '/admin', impl: '../bookshop/srv/admin-service.js' }
[cds] - server listening on { url: 'http://localhost:4004' }
[cds] - launched at 5/25/2023, 4:53:46 PM, version: 7.0.0, in: 991.573ms
As apparent from the output, both, the two bookshop services CatalogService
and AdminService
as well as our new ReviewsService
, are served in the same process (mocked, as the ReviewsService
is configured as required service in bookstore/package.json).
Now, open http://localhost:4004/reviews to display the Vue.js UI that is provided with the reviews service sample:
- Choose one of the reviews.
- Change the 5-star rating with the dropdown.
- Choose Submit.
- Enter bob to authenticate.
→ In the terminal window you should see a server reaction like this:
[cds] - PATCH /reviews/Reviews/148ddf2b-c16a-4d52-b8aa-7d581460b431
< emitting: reviewed { subject: '201', count: 2, rating: 4.5 }
> received: reviewed { subject: '201', count: 2, rating: 4.5 }
Which means the ReviewsService
emitted a reviewed
message that was received by the enhanced CatalogService
.
Open http://localhost:4004/bookshop to see the list of books served by CatalogService
and refresh to see the updated average rating and reviews count:
When emitters and receivers live in separate processes, you need to add a message channel to forward event messages. CAP provides messaging services, which take care for that message channel behind the scenes as illustrated in the following graphic:
::: tip Uniform, Agnostic Messaging CAP provides messaging services, which transport messages behind the scenes using different messaging channels and brokers. All of this happens without the need to touch your code, which stays on conceptual level. :::
For quick tests during development, CAP provides a simple file-based messaging service implementation. Configure that as follows for the [development]
profile:
"cds": {
"requires": {
"messaging": {
"[development]": { "kind": "file-based-messaging" }
},
}
}
Learn more about cds.env
profiles.{.learn-more}
In our samples, you find that in @capire/reviews/package.json as well as @capire/bookstore/package.json, which you'll run in the next step as separate processes.
First start the reviews
service separately:
cds watch reviews
The trace output should contain these lines, confirming that you're using file-based-messaging
, and that the ReviewsService
is served by that process at port 4005:
[cds] - connect to messaging > file-based-messaging { file: '~/.cds-msg-box' }
[cds] - serving ReviewsService { path: '/reviews', impl: '../reviews/srv/reviews-service.js' }
[cds] - server listening on { url: 'http://localhost:4005' }
[cds] - launched at 5/25/2023, 4:53:46 PM, version: 7.0.0, in: 593.274ms
Then, in a separate terminal start the bookstore
server as before:
cds watch bookstore
This time the trace output is different to when you started all in a single server. The output confirms that you're using file-based-messaging
, and that you now connected to the separately started ReviewsService
at port 4005:
[cds] - connect to messaging > file-based-messaging { file: '~/.cds-msg-box' }
[cds] - mocking OrdersService { path: '/orders', impl: '../orders/srv/orders-service.js' }
[cds] - serving CatalogService { path: '/browse', impl: '../reviews/srv/cat-service.js' }
[cds] - serving AdminService { path: '/admin', impl: '../reviews/srv/admin-service.js' }
[cds] - connect to ReviewsService > odata { url: 'http://localhost:4005/reviews' }
[cds] - server listening on { url: 'http://localhost:4004' }
[cds] - launched at 5/25/2023, 4:55:46 PM, version: 7.0.0, in: 1.053s
Similar to before, open http://localhost:4005/vue/index.html to add or update reviews.
→ In the terminal window for the reviews
server you should see this:
[cds] - PATCH /reviews/Reviews/74191a20-f197-4829-bd47-c4676710e04a
< emitting: reviewed { subject: '251', count: 1, rating: 3 }
→ In the terminal window for the bookstore
server you should see this:
> received: reviewed { subject: '251', count: 1, rating: 3 }
::: tip Agnostic Messaging APIs
Without touching any code the event emitted from the ReviewsService
got transported via file-based-messaging
channel behind the scenes and was received in the bookstore
as before, when you used in-process eventing → which was to be shown (QED).
:::
You can simulate a server outage to demonstrate the value of messaging for resilience as follows:
- Terminate the
bookstore
server with Ctrl + C in the respective terminal. - Add or update more reviews as described before.
- Restart the receiver with
cds watch bookstore
.
→ You should see some trace output like that:
[cds] - server listening on { url: 'http://localhost:4004' }
[cds] - launched at 5/25/2023, 10:45:42 PM, version: 7.0.0, in: 1.023s
[cds] - [ terminate with ^C ]
> received: reviewed { subject: '207', count: 1, rating: 2 }
> received: reviewed { subject: '207', count: 1, rating: 2 }
> received: reviewed { subject: '207', count: 1, rating: 2 }
::: tip Resilience by Design All messages emitted while the receiver was down stayed in the messaging queue and are delivered when the server is back. :::
You can watch the messages flowing through the message queue by opening ~/.cds-msg-box in a text editor. When the receiver is down and therefore the message not already consumed, you can see the event messages emitted by the ReviewsService
in entries like that:
ReviewsService.reviewed {"data":{"subject":"201","count":4,"rating":5}, "headers": {...}}
By default CAP uses a single message channel for all messages.
For example: If you consume messages from SAP S/4HANA in an enhanced version of bookstore
, as well as emit messages a customer could subscribe and react to in a customer extension, the overall topology would look like that:
Now, sometimes you want to use separate channels for different emitters or receivers. Let's assume you want to have a dedicated channel for all events from SAP S/4HANA, and yet another separate one for all outgoing events, to which customer extensions can subscribe too. This situation is illustrated in this graphic:
This is possible when using low-level messaging, but comes at the price of loosing all advantages of conceptual-level messaging as explained in the following.
To avoid falling back to low-level messaging, CAP provides the composite-messaging
implementation, which basically acts like a transparent dispatcher for both, inbound and outbound messages. The resulting topology would look like that:
::: tip Transparent Topologies
The composite-messaging
implementation allows to flexibly change topologies of message channels at deployment time, without touching source code or models.
:::
You would configure this in bookstore
's package.json as follows:
"cds": {
"requires": {
"messaging": {
"kind": "composite-messaging",
"routes": {
"ChannelA": ["**/ReviewsService/*"],
"ChannelB": ["**/sap/s4/**"]
"ChannelC": ["**/bookshop/**"]
}
},
"ChannelA": {
"kind": "enterprise-messaging", ...
},
"ChannelB": {
"kind": "enterprise-messaging", ...
},
"ChannelC": {
"kind": "enterprise-messaging", ...
}
}
}
In essence, you first configure a messaging service for each channel. In addition, you would configure the default messaging
service to be of kind composite-messaging
.
In the routes
, you can use the glob pattern to define filters for event names, that means:
**
will match any number of characters.*
will match any number of characters except/
and.
.?
will match a single character.
::: tip
You can also refer to events declared in CDS models, by using their fully qualified event name (unless annotation @topic
is used on them).
:::
In the previous sections it's documented how CAP promotes messaging on conceptual levels, staying agnostic to topologies and message brokers. While CAP strongly recommends staying on that level, CAP also offers lower-level messaging, which loses some of the advantages but still stays independent from specific message brokers.
::: tip Messaging as Just Another CAP Service
All messaging implementations are provided through class cds.MessagingService
and broker-specific subclasses of that. This class is in turn a standard CAP service, derived from cds.Service
, hence it's consumed as any other CAP service, and can also be extended by adding event handlers as usual.
:::
As with all other CAP services, add an entry to cds.requires
in your package.json or .cdsrc.json like that:
"cds": {
"requires": {
"messaging": {
"kind": // ...
},
}
}
Learn more about cds.env
and cds.requires
.{.learn-more}
You're free how you name your messaging service. Could be messaging
as in the previous example, or any other name you choose. You can also configure multiple messages services with different names.
Instead of connecting to an emitter service, connect to the messaging service:
const messaging = await cds.connect.to('messaging')
Instead of emitter services emitting to themselves, emit to the messaging service:
await messaging.emit ('ReviewsService.reviewed', { ... })
Instead of registering event handlers with a concrete emitter service, register handlers on the messaging service:
messaging.on ('ReviewsService.reviewed', msg => console.log(msg))
When declaring events in CDS models, be aware that the fully qualified name of the event is used as topic names when emitting to message brokers. Based on the following model, the resulting topic name is my.namespace.SomeEventEmitter.SomeEvent
.
namespace my.namespace;
service SomeEventEmitter {
event SomeEvent { ... }
}
If you want to manually define the topic, you can use the @topic
annotation:
//...
@topic: 'some/very/different/topic-name'
event SomeEvent { ... }
When looking at the previous code samples, you see that in contrast to conceptual messaging you need to provide fully qualified event names now. This is just one of the advantages you lose. Have a look at the following list of advantages you have using conceptual messaging and lose with low-level messaging.
- Service-local event names (as already mentioned)
- Event declarations (as they go with individual services)
- Generated typed API classes for declared events
- Run in-process without any messaging service
::: tip Always prefer conceptual-level API over low-level API variants. Besides the things listed above, this allows you to flexibly change topologies, such as starting with co-located services in a single process, and moving single services out to separate micro services later on. :::
CAP messaging has built-in support for formatting event data compliant to the CloudEvents standard. Enable this using the format
config option as follows:
"cds": {
"requires": {
"messaging": {
"format": "cloudevents"
}
}
}
With this setting, all mandatory and some more basic header fields, like type
, source
, id
, datacontenttype
, specversion
, time
are filled in automatically. The event name is used as type
. The message payload is in the data
property anyways.
::: tip CloudEvents is a wire protocol specification. Application developers shouldn't have to care for such technical details. CAP ensures that for you, by filling in the respective fields behind the scenes. :::
Using SAP Event Mesh {#sap-event-mesh}
CAP has out-of-the-box support for SAP Event Mesh. As an
application developer, all you need to do is configuring CAP to use enterprise-messaging
,
usually in combination with cloudevents
format, as in this excerpt from a package.json:
"cds": {
"requires": {
"messaging": {
"[production]": {
"kind": "enterprise-messaging",
"format": "cloudevents"
}
}
}
}
Learn more about cds.env
profiles{.learn-more}
::: tip Find additional information about deploying SAP Event Mesh on SAP BTP in this guide: → Using SAP Event Mesh in BTP :::
SAP S/4HANA integrates SAP Event Mesh for messaging. That makes it relatively easy for CAP-based applications to receive events from SAP S/4HANA systems.
In contrast to CAP, the asynchronous APIs of SAP S/4HANA are separate from the synchronous ones (OData, REST). So, the effort on the CAP side is to fill this gap. You can achieve it like that, for example, for an already imported SAP S/4HANA BusinessPartner API:
// filling in missing events as found on SAP Business Accelerator Hub
using { API_BUSINESS_PARTNER as S4 } from './API_BUSINESS_PARTNER';
extend service S4 with {
event BusinessPartner.Created @(topic:'sap.s4.beh.businesspartner.v1.BusinessPartner.Created.v1') {
BusinessPartner : String
}
event BusinessPartner.Changed @(topic:'sap.s4.beh.businesspartner.v1.BusinessPartner.Changed.v1') {
BusinessPartner : String
}
}
Learn more about importing SAP S/4HANA service APIs.{.learn-more}
With that gap filled, we can easily receive events from SAP S/4HANA the same way as from CAP services as explained in this guide, for example:
const S4Bupa = await cds.connect.to ('API_BUSINESS_PARTNER')
S4Bupa.on ('BusinessPartner.Changed', msg => {...})
::: tip Find more detailed information specific to receiving events from SAP S/4HANA in this separate guide: → Receiving Events from SAP S/4HANA :::