Remote Services
Remote Services are CQN-based clients to remote APIs that a CAP application consumes. This section describes how to configure and use these services.
The CAP Java SDK supports Remote Services for OData V2 and V4 APIs out of the box. The CQN query APIs enable late-cut microservices with simplified mocking capabilities. Regarding multitenant applications, these APIs keep you extensible, even towards remote APIs. In addition, they free developers from having to map CQN to OData themselves.
Cross-cutting aspects like security are provided by configuration. Applications do not need to provide additional code. The CAP Java SDK leverages the SAP Cloud SDK and in particular its destination capabilities to cover these aspects.
Destinations in the Cloud SDK are the means to express and define connectivity to a remote endpoint including authentication details. Cloud SDK destinations can be created from various sources such as SAP BTP Destination Service or Service Bindings. They can also be defined and registered programmatically in code. The application can choose the best fitting option for their scenario. Every Remote Service internally uses a destination for connectivity.
On top of that CAP integrates nicely with Cloud SDK, for example, ensuring automatic propagation of tenant and user information from the Request Context to the Cloud SDK.
CAP's clear recommendation is to use Remote Services over directly using the SAP Cloud SDK. However, if you can't leverage CQN-based Remote Services, refer to native consumption with Cloud SDK for details.
TIP
To learn more about how to use Remote Services end to end read the Consuming Services cookbook.
Configuring Remote Services
To enable Remote Services for OData V2 or V4 APIs in an application, add the following Maven dependency to your project:
<dependency>
<groupId>com.sap.cds</groupId>
<artifactId>cds-feature-remote-odata</artifactId>
<scope>runtime</scope>
</dependency>
Remote Services need to be configured explicitly in your application configuration. The configuration needs to define two main aspects:
- The CDS service definition of the remote API from the CDS model.
- The (BTP or programmatic) destination or service binding of the remote API and its protocol type.
The following example, shows how you can configure Remote Services in Spring Boot's application.yaml based on a destination:
cds:
remote.services:
API_BUSINESS_PARTNER:
type: "odata-v2"
destination:
name: "s4-business-partner-api"
Remote Services use a CDS service definition from the CDS model as a specification of the remote API. This API specification is required to properly translate CQN statements into respective OData V2 and V4 requests.
By default the CDS service definition is looked up in the CDS model using the name of the Remote Service. The name can be explicitly configured using the name
property. It defaults to the YAML key of the remote service configuration section (here: API_BUSINESS_PARTNER
).
The type
property defines the protocol used by the remote API. The CAP Java SDK currently supports odata-v4
(default) or odata-v2
.
TIP
You can use the cds import
command to generate a CDS service definition from an EDMX API specification. To learn more about this, have a look at the section Importing Service Definitions.
Learn about all cds.remote.services
configuration possibilities in our CDS Properties Reference.
Configuring CDS Service Name
The CDS service definition is, by default, looked up in the CDS model using the name of the Remote Service.
However, the name of the Remote Service needs to be unique, as it's also used to look up the service in Java. Therefore, it's possible to explicitly configure the name of the CDS service definition from the CDS model using the model
property. This is especially useful when creating multiple Remote Services for the same API with different destinations:
cds:
remote.services:
bupa-abc:
model: "API_BUSINESS_PARTNER"
destination:
name: "s4-business-partner-api-abc"
bupa-def:
model: "API_BUSINESS_PARTNER"
destination:
name: "s4-business-partner-api-def"
Using Service Bindings
If the remote API is running on SAP BTP, it's likely that you can leverage Service Binding-based Remote Services. The CAP Java SDK extracts the relevant information from the service binding to connect to the remote API. Service-binding-based Remote Services are simple to use, as the service binding abstracts from several aspects of remote service communication. For instance, it provides authentication information and the URL of the service. In contrast to destinations, it can be created and refreshed as part of the application lifecycle, that is, application deployment. Hence, the location and security aspects of remote services are transparent to CAP applications in the case of service bindings.
Binding to a Reuse Service
If the remote API is exposed by a BTP reuse service, a service broker typically provides means to create service instances of the BTP service. The CAP application requires a service binding to this service to consume the remote API as a Remote Service.
These service instances of BTP services provide the URL of the remote API in their service binding. Therefore, you only need to specify the binding name in the application.yaml
configuration, like in the following example:
cds:
remote.services:
SomeReuseService:
binding:
name: some-service-binding
If binding structure isn't understood ...
In some cases, SAP Cloud SDK doesn't understand the service binding structure of the specific BTP service. In that case it's required to contribute a mapping by the means of Cloud SDK's PropertySupplier
. This PropertySupplier
needs to be registered with the Cloud SDK once at application startup.
static {
OAuth2ServiceBindingDestinationLoader.registerPropertySupplier(
options -> options.getServiceBinding().getTags().contains("<bindingTag>"),
SomeReuseServiceOAuth2PropertySupplier::new);
}
The <bindingTag>
needs to be replaced by the concrete name of the tag provided in the binding of the BTP service. Alternatively, a check on the service name can be chosen as well. The class SomeReuseServiceOAuth2PropertySupplier
needs to be provided by you extending the Cloud SDK base class DefaultOAuth2PropertySupplier
.
Learn more about registering OAuth2PropertySupplier in the SAP Cloud SDK documentation.
Binding to a Service with Shared Identity
If the remote API is available within the same SaaS application and using the same (shared) service instance of XSUAA or Identity (IAS) for authentication, no service broker-based reuse service is required. The Remote Service can be configured using the shared service instance as binding (here: shared-xsuaa
):
cds:
remote.services:
OtherCapService:
binding:
name: shared-xsuaa
options:
url: https://url-of-the-second-cap-application
The plain service binding of XSUAA or IAS does not contain the URL of the remote API. Therefore, it needs to be explicitly configured in the options
section. Since the URL is typically not known during development, you can define it as an environment variable. For the previous example, use CDS_REMOTE_SERVICES_OTHERCAPSERVICE_BINDING_OPTIONS_URL
.
Learn more about Binding From Environment Variables in the Spring Boot documentation.
TIP
Remote APIs which require IAS-based authentication might expect certificate based client authentication in addition to the IAS-based JWT token, see ProofOfPossession validation. CAP Remote Services automatically takes care of this by initiating a mutual TLS handshake with the remote API.
Configuring the Authentication Strategy
While service bindings typically provide authentication details, they don't predetermine the user propagation and authentication strategy, for example, technical user or named user flow. The parameter onBehalfOf
in the binding
configuration section allows to define these strategies.
The following options are available:
currentUser
: Use the user of the current Request Context. This propagates the named user if available or falls back to a (tenant-specific) technical user otherwise. (default)systemUser
: Use a (tenant-specific) technical user, based on the tenant set in the current Request Context.systemUserProvider
: Use a technical user of the provider tenant. This is especially helpful on an internal communication channel that is not authorized tenant-specifically.
Using Destinations
If your remote API is not using Service Bindings, you typically need to separately obtain the URL and additional metadata like credentials from the service provider. You can store these in destinations of SAP BTP Destination Service or programmatically register a destination with Cloud SDK to make them available for usage in your CAP application.
Based on the following configuration, a destination with name s4-business-partner-api
is looked up using the Cloud SDK:
cds:
remote.services:
API_BUSINESS_PARTNER:
type: "odata-v2"
destination:
name: s4-business-partner-api
If your CAP application is using IAS and you want to call a remote API that is provided by another IAS-based application (ie. Application2Application scenario), you can utilize a simplified security configuration in the destination. As a pre-requisite, your CAP application and the called application need to trust the same IAS tenant and you need to define a dependency in IAS to consume the respective API provided by the remote API.
Create a destination configuration with the following parameters:
- URL:
<url-of-the-remote-api>
- Authentication:
NoAuthentication
- Additional Properties:
- cloudsdk.ias-dependency-name:
<name-of-the-ias-dependency>
- cloudsdk.ias-dependency-name:
At runtime, this destination configuration will use the bound identity
service instance's credentials to request a token for the remote API.
The CAP Java SDK obtains the destination for a Remote Service from the DestinationAccessor
using the name that is configured in the Remote Service's destination configuration.
If you're using the SAP BTP Destination Service, this is the name you used when you defined the destination there. To properly resolve the destination from SAP BTP Destination Service additional Cloud SDK dependencies are required.
In multitenant scenarios, the SAP BTP Destination Service tries to look up the destination from the subaccount of the current tenant, set on the RequestContext
. This is not restricted to subscriber tenants, but also includes the provider tenant. Retrieval strategies are part of a set of configuration options provided by the Cloud SDK, which are exposed by CAP Java as part of the configuration for Remote Services. For details refer to the section about destination strategies.
TIP
As a prerequisite for destination lookup in subscriber accounts, the CAP application needs to define a dependency to the Destination service for their subscriptions, for example, in the SaaS registry. This can be enabled by setting the cds.multiTenancy.dependencies.destination
to true
in the configuration.
Learn more about destinations in the SAP Cloud SDK documentation.
Configuring the URL
The destination or service binding configuration provides the base URL to the OData V2 or V4 service, that should be used by the Remote Service. The full service URL however is built from three parts:
- The URL provided by the destination or the service binding configuration.
- An optional URL suffix provided in the Remote Service http configuration under the
suffix
property. - The name of the service, either obtained from the optional
service
configuration property or the fully qualified name of the CDS service definition.
Consider this example:
cds:
remote.services:
API_BUSINESS_PARTNER:
http:
suffix: "/sap/opu/odata/sap"
destination:
name: s4-business-partner-api
In this case, the destination with name s4-business-partner-api
would be obtained from the DestinationAccessor
. Given that this destination holds the URL https://s4.sap.com
, the resulting service URL for OData requests would be https://s4.sap.com/sap/opu/odata/sap/API_BUSINESS_PARTNER
.
Consuming Remote Services
Remote Services can be used in your CAP application just like any other service that accepts CQN queries:
@Autowired
@Qualifier(ApiBusinessPartner_.CDS_NAME)
CqnService bupa;
CqnSelect select = Select.from(ABusinessPartnerAddress_.class)
.where(a -> a.BusinessPartner().eq("4711"));
ABusinessPartnerAddress address = bupa.run(select)
.single(ABusinessPartnerAddress.class);
TIP
To learn more about how to build and run CQN queries, see sections Building CQN Queries and Executing CQN Queries.
Keep in mind that Remote Services are simply clients to remote APIs. CAP doesn't automatically forward CQN queries to these services. Developers need to explicitly call and use these Remote Services in their code. However, as Remote Services are based on the common CQN query APIs it's easy to use them in event handlers of your Application Services.
WARNING
In case data from Remote Services should be combined with data from the database custom coding is required. Refer to the Integrate and Extend guide for more details.
Cloud SDK Integration
Maven Dependencies
The CAP Java SDK only includes the minimum SAP Cloud SDK dependencies required out of the box. In case you want to leverage features from SAP Cloud SDK, like the programmatic destination registration or integration with SAP BTP Destination Service, you need to add additional dependencies.
It's recommended to add the SAP Cloud SDK BOM to the dependency management section of your application's parent POM. If you're also using the CDS Services BOM or the Spring Boot dependencies BOM, it's recommended to add the SAP Cloud SDK BOM after these:
<dependencyManagement>
<!-- CDS Services BOM -->
<!-- Spring Boot dependencies BOM -->
<dependencies>
<dependency>
<groupId>com.sap.cloud.sdk</groupId>
<artifactId>sdk-bom</artifactId>
<version>use-latest-version-here</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
Learn more about dependency management of SAP Cloud SDK.
To enable programmatic destination registration, add this additional dependency to your project:
<dependency>
<groupId>com.sap.cloud.sdk.cloudplatform</groupId>
<artifactId>cloudplatform-connectivity</artifactId>
</dependency>
To integrate with SAP BTP Destination Service on Cloud Foundry, add this additional dependency to your project:
<dependency>
<groupId>com.sap.cloud.sdk.cloudplatform</groupId>
<artifactId>scp-cf</artifactId>
</dependency>
Configuring Destination Strategies
When loading destinations from SAP BTP Destination Service, you can specify a destination retrieval strategy and a token exchange strategy.
These strategies can be set in the destination configuration of the Remote Service:
cds:
remote.services:
API_BUSINESS_PARTNER:
destination:
name: "s4-business-partner-api"
retrievalStrategy: "AlwaysProvider"
tokenExchangeStrategy: "ExchangeOnly"
TIP
Values for destination strategies have to be provided in pascal case.
Programmatic Destination Registration
You can also programmatically build destinations and add them to the DestinationAccessor
to make them available for Remote Services. You can easily register an event handler that is executed during startup of the application and build custom destinations:
@Component
@ServiceName(ApplicationLifecycleService.DEFAULT_NAME)
public class DestinationConfiguration implements EventHandler {
@Value("${api-hub.api-key:}")
private String apiKey;
@Before(event = ApplicationLifecycleService.EVENT_APPLICATION_PREPARED)
public void initializeDestinations() {
if(apiKey != null && !apiKey.isEmpty()) {
DefaultHttpDestination httpDestination = DefaultHttpDestination
.builder("https://sandbox.api.sap.com/s4hanacloud")
.header("APIKey", apiKey)
.name("s4-business-partner-api").build();
DestinationAccessor.prependDestinationLoader(
new DefaultDestinationLoader().registerDestination(httpDestination));
}
}
}
Find out how to create destinations for different authentication typesLearn more about using destinations
Note that you can leverage Spring Boot's configuration possibilities to inject credentials into the destination configuration. The same mechanism can also be used for the URL of the destination by also reading it from your application configuration (for example environment variables or application.yaml). This is especially useful when integrating micro-services, which may have different URLs in productive environments and test environments.
Native Service Consumption
If you need to call an endpoint that you cannot consume as a Remote Service, you can fall back to leverage Cloud SDK APIs. Based on the Cloud SDK's HttpClientAccessor
API, you can resolve an HttpClient
that you can use to execute plain HTTP requests against the remote API.
However, this involves low-level operations like payload de-/serialization. Usage of CAP's Remote Service is encouraged whenever possible to free the developer from these.
Learn more about HttpClientAccessor in the SAP Cloud SDK documentation.
Using Service Bindings
If the URL and credentials of the remote API are available as a service binding, you can create a Cloud SDK destination for the service binding using the ServiceBindingDestinationLoader
API. Based on this, it's possible to create an instance of HttpClient
using the HttpClientAccessor
:
ServiceBinding binding = ...;
HttpDestination destination = ServiceBindingDestinationLoader.defaultLoaderChain().getDestination(
ServiceBindingDestinationOptions
.forService(binding)
.onBehalfOf(OnBehalfOf.TECHNICAL_USER_CURRENT_TENANT)
.build());
HttpClient httpClient = HttpClientAccessor.getHttpClient(destination);
...
Learn more about HttpClientAccessor in the SAP Cloud SDK documentation.
To be able to resolve a service binding into a Cloud SDK destination, a OAuth2PropertySupplier
might need to be registered with Cloud SDK.
static {
OAuth2ServiceBindingDestinationLoader.registerPropertySupplier(
options -> options.getServiceBinding().getTags().contains("<bindingTag>"),
BizPartnerOAuth2PropertySupplier::new);
}
Learn more about registering OAuth2PropertySupplier in the SAP Cloud SDK documentation.
Using Destinations
If the URL and credentials of the remote API are configured as a destination in SAP BTP Destination Service, you can use Cloud SDK's DestinationAccessor
API to load the destination based on its name. In a second step, HttpClientAccessor
is used to create an instance of HttpClient
:
HttpDestination destination = DestinationAccessor.getDestination("<destinationName>").asHttp();
HttpClient httpClient = HttpClientAccessor.getHttpClient(destination);
...
Destination destination = DestinationAccessor.getDestination("<destinationName>");
HttpClient httpClient = HttpClientAccessor.getHttpClient(destination);
...
Programmatic Destinations
The following example code snippets show how to programmatically create a destination for different authentication types. You can register these destinations with the DestinationAccessor
to use them with Remote Services or use them natively with the HttpClientAccessor
to obtain HttpClient
instances.
Use the following example if the remote API supports basic authentication:
DefaultHttpDestination
.builder("https://example.org")
.basicCredentials("user", "password")
.name("my-destination").build();
Use the following example if you can directly forward the token from the current security context:
DefaultHttpDestination
.builder("https://example.org")
.authenticationType(AuthenticationType.TOKEN_FORWARDING)
.name("my-destination").build();
Use the following example if you want to call the remote API using a technical user:
ClientCredentials clientCredentials =
new ClientCredentials("clientid", "clientsecret");
OAuth2DestinationBuilder
.forTargetUrl("https://example.org")
.withTokenEndpoint("https://xsuaa.url")
.withClient(clientCredentials, OnBehalfOf.TECHNICAL_USER_CURRENT_TENANT)
.property("name", "my-destination")
.build();
Use the following example if you need to exchange the token from the security context (that is, user token exchange):
ClientCredentials clientCredentials =
new ClientCredentials("clientid", "clientsecret");
OAuth2DestinationBuilder
.forTargetUrl("https://example.org")
.withTokenEndpoint("https://xsuaa.url")
.withClient(clientCredentials, OnBehalfOf.NAMED_USER_CURRENT_TENANT)
.property("name", "my-destination")
.build();