label | synopsis | status | uacp |
---|---|---|---|
Best Practices |
Learn about Node.js best practices.
|
released |
This page is linked from the Help Portal at https://help.sap.com/products/BTP/65de2977205c403bbc107264b8eccf4b/29c25e504fdb4752b0383d3c407f52a6.html |
From generic Node.js best practices like dependency management and error handling to CAP-specific topics like transaction handling and testing, this video provides some tips and tricks to improve the developer experience and avoid common pitfalls, based on common customer issues. In the following section we explain these best practices.
Projects using CAP need to manage dependencies to the respective tools and libraries in their package.json and/or pom.xml respectively. Follow the guidelines to make sure that you consume the latest fixes and avoid vulnerabilities and version incompatibilities. These guidelines apply to you as a consumer of reuse packages as well as a provider of such reuse packages.
This applies to both, @sap packages as well as open source ones. It makes sure that your projects receive the latest features and important fixes during development. It also makes sure that bundles have a minimal footprint by leveraging npm's dedupe feature.
Example:
"dependencies": {
"@sap/cds": "^5.5.0",
"@sap/some-reuse-package": "^1.1.0",
"express": "^4.17.0"
}
::: tip
We recommend using the caret form, that is, ^1.0.2
to add your dependencies, which are also the default for npm install
, as that clearly captures the minimum patch version.
:::
Let's explain this by looking at counter examples.
Let's assume that you've developed a reuse package that others can use in their projects, and you also use a reuse package. For whatever reason, you decided to violate the previous rules and use exact dependencies in your package.json as follows:
"name": "@sap/your-reuse-package",
"version": "1.1.2",
"dependencies": {
"@sap/cds": "3.0.3",
"@sap/foundation": "2.0.1",
"express": "4.16.3"
}
The effect would be as follows:
- Consuming projects would get duplicate versions of each package that they also use directly, for example,
@sap/cds
,@sap/foundation
, andexpress
. - Consuming projects wouldn't receive important fixes for the packages used in your implementations unless you also provide an update.
- It wouldn't be possible to reuse CDS models from common reuse packages (for example, would already fail for
@sap/cds/common
).
Therefore, the rules when publishing packages for reuse are:
- Keep the open ranges in your package.json (just don't touch them).
- Do an npm update before publishing and test thoroughly. (→ ideally automated in your CI/CD pipeline).
- Do the vulnerability checks for your software and all open-source software used by you or by packages you used (→ Minimize Usage of Open Source Packages).
- Don't do
npm shrinkwrap
→ see also npm's docs: "It's discouraged for library authors to publish this file, ..."
::: tip If both your package and a consuming package reuse the same CDS models, loading those models would fail because it's impossible to automatically merge the two versions, nor is it possible to load two independent versions. The reason for this is that it's reusing models that share the same single definitions. :::
When releasing a service or an application to end consumers, use npm install
or npm update
to produce a package-lock.json file that freezes dependencies. This guarantees that it works correctly as it did the last time you tested it and checked it for vulnerabilities.
Overall, the process for your release should include these steps:
npm config set package-lock true # enables package-lock.json
npm update # update it with latest versions
git add package-lock.json # add it to version control
# conduct all test and vulnerability checks
The package-lock.json file in your project root freezes all dependencies and is deployed with your application. Subsequent npm installs, such as by cloud deployers or build packs, always get the same versions, which you checked upon your release.
This ensures that the deployed tool/service/app doesn't receive new vulnerabilities, for example, through updated open source packages, without you being able to apply the necessary tests as prescribed by our security standards.
:::tip
We recommend running npm update
regularly and frequently during development to ensure that you receive the latest fixes.
Tools like renovate or GitHub's dependabot can help you automate this process.
:::
This rule for keeping open ranges for dependencies during development, as well as when publishing for reuse, also applies for open source packages.
Because open source packages are less reliable with respect to vulnerability checks, this means that end-of-chain projects have to ensure respective checks for all the open source packages they use directly, as well as those they 'inherit' transitively from reuse packages.
So, always take into account these rules:
-
When releasing to end consumers, you always have to conduct vulnerability checks for all open source packages that you used directly or transitively.
-
As a provider of reuse packages you should minimize the usage of open source packages to a reasonable minimum.
Q: Why not freeze open source dependencies when releasing for reuse?
A: Because that would only affect directly consumed packages, while packages from transitive dependencies would still reach your consumers.
A good approach is to also provide certain features in combination with third-party packages, but to keep them, and hence the dependencies, optional; for example, express.js does this.
As providers of evolving SDKs we provide major feature updates, enhancements, and improvements in 6-12 month release cycles. These updates come with an increment of major release numbers.
At the same time, we can't maintain and support unlimited numbers of branches with fixes. The following rules apply:
- Fixes and nonbreaking enhancements are made available frequently in upstream release branches (current major).
- Critical fixes also reach recent majors in a 2-month grace period.
To make sure that you receive ongoing fixes, make sure to also adopt the latest major releases in a timely fashion in your actively maintained projects, that is, following the 6-12 month cycle.
Using npm-shrinkwrap.json — only if you want to publish CLI tools or other 'sealed' production packages to NPM. Unlike package-lock.json, it does get packaged and published to NPM registries. See the NPM documentation for more.
To keep builds as small as possible, the Node.js runtime doesn't bring any potentially unnecessary dependencies and, hence, doesn't automatically mount any express middlewares, such as the popular helmet
and csurf
.
However, application developers can easily mount custom or best-practice express middlewares using the bootstrapping mechanism.
Example:
// local ./server.js
const cds = require('@sap/cds')
const helmet = require('helmet')
cds.on('bootstrap', app => {
app.use(helmet())
})
module.exports = cds.server // > delegate to default server.js
{ style="padding: 0 33px"}
Consult sources such as Express' Production Best Practices: Security documentation for state of the art application security.
Creating a Content Security Policy (CSP) is a major building block in securing your web application.
helmet
provides a default policy out of the box that you can also customize as follows:
cds.on('bootstrap', app => {
app.use(
helmet({
contentSecurityPolicy: {
directives: {
...helmet.contentSecurityPolicy.getDefaultDirectives()
// custom settings
}
}
})
)
})
::: tip Consult OpenUI5 Content Security Policy documentation for the list of directives that SAPUI5 requires. :::
Protect against cross-side request forgery (CSRF) attacks by enabling CSRF token handling either through the App Router or by adding it manually. ::: tip For a SAPUI5 (SAP Fiori/SAP Fiori Elements) application developer, CSRF token handling is transparent. There's no need to program or to configure anything in additional. In case the server rejects the request with 403 and “X-CSRF-Token: required”, the UI sends a HEAD request to the service document to fetch a new token. :::
Learn more about CSRF tokens and SAPUI5 in the Cross-Site Scripting documentation.{.learn-more}
::: warning ❗ The request must never be cacheable. :::
The App Router is configured to require a CSRF token by default for all protected routes and all HTTP requests methods except HEAD and GET. Thus, by adding the App Router as described in the Deployment Guide: Using App Router as Gateway, endpoints are CSRF protected.
Learn more about CSRF protection with the App Router{.learn-more}
On the backend side, except for handling the HEAD request mentioned previously, also the handlers for each CSRF protected method and path should be added. In the following example, the POST method is protected. ::: tip If you use SAP Fiori Elements, requests to the backend are sent as batch requests using the POST method. In this case, an arbitrary POST request should be protected. :::
As already mentioned, in case the server rejects because of a bad CSRF token, the response with a status 403 and a header “X-CSRF-Token: required” should be returned to the UI. For this purpose, the error handling in the following example is extended:
cds.on('bootstrap', async app => {
var csrfProtection = csrf({ cookie: true })
var parseForm = express.urlencoded({ extended: false })
app.use(cookieParser())
// Must: Provide actual <service endpoint>s of served services.
// Optional: Adapt for non-Fiori Elements UIs.
app.head('/<service endpoint>', csrfProtection, function (req, res) {
res.set('X-CSRF-Token', req.csrfToken())
res.send()
})
// Must: Provide actual <service endpoint>s of served services.
// Optional: Adapt for non-Fiori Elements UIs.
app.post('/<service endpoint>/$batch', parseForm, csrfProtection, function (req, res, next) {
next()
})
app.use(function (err, req, res, next) {
if (err.code !== 'EBADCSRFTOKEN') return next(err)
res.status(403)
res.set('X-CSRF-Token', 'required')
res.send()
})
})
Learn more about backend coding in the csurf documentation.{.learn-more} ::: tip If you're using horizontal scaling of Node.js virtual machines, the CSRF handling should be done at the approuter level. :::
With Cross-Origin Resource Sharing (CORS) the server that hosts the UI can tell the browser about servers it trusts to provide resources. In addition, so-called "preflight" requests tell the browser if the cross-origin server will process a request with a specific method and a specific origin. ::: tip CORS should either be configured in the App Router or in the backend, but not in both places. :::
The App Router has full support for CORS. Thus, by adding the App Router as described in the Deployment Guide: Using App Router as Gateway, CORS can be configured in the App Router configuration.
Learn more about CORS handling with the App Router{.learn-more}
If not running in production, CAP's default server allows all origins. For production, you can add CORS to your server as follows:
const ORIGINS = { 'https://example.com': 1 }
cds.on('bootstrap', async app => {
app.use((req, res, next) => {
const { origin } = req.headers
// standard request
if (origin && ORIGINS[origin]) res.set('access-control-allow-origin', origin)
// preflight request
if (origin && ORIGINS[origin] && req.method === 'OPTIONS')
return res.set('access-control-allow-methods', 'GET,HEAD,PUT,PATCH,POST,DELETE').end()
next()
})
})
Learn more about CORS in the MDN Web Docs documentation.{.learn-more}
To proactively identify problems, projects should set up availability monitoring for all the components involved in their solution.
An anonymous ping service should be implemented with the least overhead possible. Hence, it should not use any authentication or authorization mechanism, but simply respond to whoever is asking.
The Node.js runtime does not yet provide an out of the box solution for availability monitoring. However, the anonymous ping endpoint can be easily provided via a custom express middleware as follows.
cds.on('bootstrap', app => {
app.get('/health', (_, res) => {
res.status(200).send('OK')
})
})
Good error handling is important to ensure the correctness and performance of the running app and developer productivity. We will give you a brief overview of common best practices.
We need to distinguish between two types of errors:
- Programming errors: These occur because of some programming mistakes (for example,
cannot read 'foo' of undefined
). They need to be fixed. - Operational errors: These occur during the operation (for example, when a request is sent to an erroneous remote system). They need to be handled.
'Let it crash' is a philosophy coming from the Erlang programming language (Joe Armstrong) which can also be (partially) applied to Node.js.
The most important aspects for programming errors are:
- Fail loudly: Do not hide errors and silently continue. Make sure that unexpected errors are correctly logged. Do not catch errors you can't handle.
- Don't program in a defensive way: Concentrate on your business logic and only handle errors if you know that they occur. Only use
try
/catch
blocks when necessary.
Never attempt to catch and handle unexpected errors, promise rejections, etc. If it's unexpected, you can't handle it correctly. If you could, it would be expected (and should already be handled). Even though your apps should be stateless, you can never be 100% certain that any shared resource wasn't affected by the unexpected error. Hence, you should never keep an app running after such an event, especially in multi-tenant apps that bear the risk of information disclosure.
This will make your code shorter, clearer, and simpler.
If an error occurs, it should be possible to know the origin. If you catch errors and re-throw them without the original information, it becomes hard to find and fix the root cause.
Example:
try {
// something
} catch (e) {
// augment instead of replace details
e.message = 'Oh no! ' + e.message
e.additionalInfo = 'This is just an example.'
// re-throw same object
throw e
}
In rare cases, throwing a new error is necessary, for example, if the original error has sensitive details that should not be propagated any further. This should be kept to an absolute minimum.
The following articles might be of interest:
When using timestamps (for example for managed dates) the Node.js runtime offers a way to easily deal with that without knowing the format of the time string. The req
object contains a property timestamp
that holds the current time (specifically new Date()
, which is comparable to CURRENT_TIMESTAMP
in SQL). It also stays the same until the request finished, so if it is used in multiple places in the same transaction or request it will always be the same.
Example:
srv.before("UPDATE", "EntityName", (req) => {
const now = req.timestamp;
req.data.createdAt = now;
});
Internally the timestamp is a Javascript Date
object, that is converted to the right format, when sent to the database. So if in any case a date string is needed, the best solution would be to initialize a Date object, that is then translated to the correct UTC String for the database.
When using Media Data the Node.js runtime offers a possibility to
return a custom stream object as response to READ
requests like GET /Books/coverImage
.
Example:
srv.on('READ', 'Books', (req, next) => {
if (coverImageIsRequested) {
const readable = new Readable()
return {
value: readable,
$mediaContentType = 'image/jpeg',
$mediaContentDispositionFilename = 'cover.jpg', // > optional
$mediaContentDispositionType = 'inline' // > optional
}
}
return next()
})
In the returned object, value
is an instance of stream.Readable and the properties $mediaContentType
, $mediaContentDispositionFilename
, and $mediaContentDispositionType
are used to set the respective headers.