In this exercise you'll learn how to add custom processing of specific OData operations on your services. It's done by adding custom implementation logic (in JavaScript) via well-defined hooks into the service API.
Along the way you'll also learn how to use debugging features in VS Code, with the launch configuration provided by the cds init
command that you used in an earlier exercise.
With custom implementation logic you can turn an out-of-the-box service into more or less whatever you need, in a clean and minimal way.
At the end of these steps you will have modified the CatalogService
service in such a way that books with a high stock value will be discounted (and for the sake of simplicity with this simple data model, the discount will be shown in the book's title).
Custom logic for a given service definition is provided in a JavaScript file that shares the same base name as that service definition file. For example, for a service definition file my-service.cds
the custom logic should be placed in my-service.js
.
This custom logic file is normally placed in the same directory as the service definition file (i.e. side-by-side with it).
👉 Create a new file service.js
in the srv/
directory. You should end up with something like this:
👉 In the new file service.js
, add the following code:
module.exports = srv => {
console.log('Service name:', srv.name)
}
You can see that this custom logic handler file is in the form of a module, which exports a single function. That function (defined using ES6 arrow functions) has a single parameter srv
to receive the server object on invocation.
Following the automatic service restart, you should see a few interesting new lines in the output:
[cds] - connect to datasource - sqlite:bookshop
[cds] - serving CatalogService at /catalog - with impl: srv/service.js
[cds] - serving Stats at /stats - with impl: srv/service.js
Service name: CatalogService
Service name: Stats
[cds] - launched in: 928.482ms
[cds] - server listening on http://localhost:4004 ...
[ terminate with ^C ]
The two lines containing "serving <service> at <endpoint> ..." that we've seen before now have extra information showing that there's a JavaScript implementation that complements the service definition.
Note that as the relationship between the service definition and the handler is at file level, the new service.js
file is deemed a handler for both services (CatalogService
and Stats
) in that service definition file. In fact, we can see that two lines from the call to console.log
confirm that - the function defined in the module is called twice - once for each service (the first time srv.name
is "CatalogService", and the second time it's "Stats").
The project already comes with some configuration that works with VS Code for debugging - if you're curious, have a look in the .vscode/
directory in the root of the project.
It means that you can easily set a breakpoint and launch the service in debug mode using standard VS Code features.
👉 Before proceeding with the main part of this step, make sure the service is not running - go to the integrated terminal and stop it with Ctrl-C.
👉 Now, set a breakpoint on the console.log
line you added in the custom logic handler, by clicking in the margin to the left of the line numbers (or hitting F9 when on the line), to set a red mark as shown:
👉 Start the service in debug mode by using menu option "Debug -> Start Debugging", or simply hit F5. VS Code should switch to the debug perspective on the left hand side, and the service should start running, pausing at the console.log
line as shown:
At this point you can explore a little bit.
👉 Switch to the Debug Console (next to the integrated terminal) and examine the srv
object, which reflects a rich API. Try examining the values of the following, by typing them into the Debug Console input area.
srv.name
Object.keys(srv.entities)
srv.path
👉 Use the debug control buttons to continue:
👉 When you've finished exploring, use the "Stop" debug control button to terminate the service.
At this point we're confident enough to start adding custom logic, by registering custom handlers. The custom logic should cause a discount message ("5% off!") to appear with the titles of books that are highly stocked (and therefore are those which we want to discount in order to get rid of).
👉 Add the following code directly after the call to console.log
in the service.js
file. As you do, notice in the code the comments that the custom logic is implemented in two different ways, using two different programming styles - you only need one of them. Comment out (or delete) one of them, leaving the one you prefer:
if (srv.name === 'CatalogService') {
srv.after ('READ', 'Books', xs => {
// CHOOSE ONLY ONE OF THESE ...
// AND LET US KNOW YOUR PREFERENCE AND WHY! :-)
// option 1 start
xs.map(x => x.stock > 500 && (x.title = `(5% off!) ${x.title}`))
// option 1 end
// option 2 start
let newBooks = [];
xs.forEach(x => {
if (x.stock > 500) {
x.title = '(5% off!) ' + x.title
}
newBooks.push(x)
})
return newBooks
// option 2 end
})
}
👉 Restart the service (you can choose to do it normally or in debug mode so you can explore with breakpoints in this code) and check that the titles for certain books have been modified to show a discount, like this:
You have added custom logic and learned how to debug a service in VS Code. The options available for adding custom logic are rich and plentiful - we recommend you look further into the documentation for more information.
- What other hooks do you think might be useful in customizing a service?
- How many times is the function (that is supplied to the
after
hook) called?