Skip to content

Files

Latest commit

2246bf1 · Jun 20, 2024

History

History
This branch is 304 commits ahead of, 202 commits behind zowe/api-layer:v3.x.x.

caching-service

Caching Service

To support the High Availability of all components within Zowe, components either need to be stateless, or offload the state to a location accessible by all instances of the service, including those which just started. At the current time, some services are not, and cannot be stateless. For these services, we introduce the Caching Service.

The Caching Service aims to provide an API which offers the possibility to store, retrieve and delete data associated with keys. The service will be used only by internal Zowe applications and will not be exposed to the internet. The Caching Service needs to support a hot-reload scenario in which a client service requests all available service data.

There are two production implementations offered directly by Zowe: VSAM storage running on Z and Redis running off Z. The other service providers may provide further storage solutions.

Architecture

Internal architecture needs to take into consideration, namely the fact that there will be multiple storage solutions. The API, on the other hand, remains the same throughout various storage implementations.

Diagram

How to use

The Caching Service is built on top of the spring enabler, which means that it is dynamically registered to the API Mediation Layer. It appears in the API Catalog under the tile "Zowe Applications".

There are REST APIs available to create, delete, and update key-value pairs in the cache, as well as APIs to read a specific key-value pair or all key-value pairs in the cache.

Storage

There are multiple storage solutions supported by the Caching Service with the option to add custom implementation. Additional Storage Support explains what needs to be done to implement custom solution.

In Memory

This storage is useful for testing and integration verification. Don't use it in production. The key/value pairs are stored only in the memory of one instance of the service and therefore won't persist.

VSAM

VSAM is a first solution as a storage for running the Caching Service on Z (on platform). As the VSAM is specific for the zOS there is no way to run it in a standard development environment. To run this scenario the Caching Service needs to be deployed on platform. More information on how to achieve this is in the Ad hoc mainframe Deployment

Further documentation on how to use the VSAM are here

Performance

Due to the Java access to the VSAM there are performance limitation to this approach. We have been testing in a few scenarios.

Test process:

  1. Load 1000 records into cache concurrently (5 threads)
  2. Update all records for 120 seconds with increasing thread count, up to amount of threads
  3. Read all records for 120 seconds with increasing thread count, up to amount of threads
  4. Read and update all records for 120 seconds with increasing thread count, up to amount of threads
  5. Delete all loaded records from cache concurrently (5 threads)

Tests are run with 3 scenarios: Low load: 5 threads, Medium load: 15 threads, High load: 50 threads

Test subjects:

  • Single Caching Service with VSAM storage
  • Two Caching Services with shared VSAM storage

Results

  • Most important operation is READ.
  • Two Caching Services achieve better READ performance than a single CS.
  • Based on data, the read performance seems acceptable, ranging from 300 ms to 1000 ms.
  • With two Caching Services and high load, the read performance is significantly better.
  • Response times of other operations are also acceptable, yet error rates increase with higher concurrency.
  • Two Caching Services produce higher load on shared resource (VSAM) and have higher error rate.
  • It seems to us that for user-based workloads, the VSAM implementation will suffice. For light automation workloads it might be acceptable as well. For heavy workloads it might not be enough.
  • VSAM does not scale very well beyond 1000 RPM on a node.

Redis

Redis is another valid option for the storage to use. The main goal for Redis is the running of the storage, and the Caching Service off platform.

For development the repository contains docker compose scripts. There are two setups provided.

  1. redis/docker-compose-replica.yml - Starts two containers in master/replica setup.
  2. redis/docker-compose-replica-tls - Starts two containers in master/replica setup with TLS enabled.
  3. redis/docker-compose-sentinel.yml - Starts five containers. Master, replica and three sentinels to coordinate in case sentinel fails.
  4. redis/docker-compose-sentinel-tls.yml - Starts five containers. Master, replica and three sentinels to coordinate in case sentinel fails. TLS is enabled.

The first setup works well for testing. In order to properly configure the Caching Service following configuration is needed either in application.yml in the Caching Service or passed in via command line parameters.

caching:
    storage:
        mode: redis
        redis:
            host: localhost
            port: 6379
            username: default
            password: heslo

In order to enable TLS, the following configuration is required:

caching:
     storage:
         mode: redis
         redis:
             host: localhost
             port: 6379
             username: default
             password: heslo
            ssl:
                enabled: true
                keyStore: ${server.ssl.keyStore}
                keyStorePassword: ${server.ssl.keyStorePassword}
                trustStore: ${server.ssl.trustStore}
                trustStorePassword: ${server.ssl.trustStorePassword}

In order to connect to sentinel, the following configuration can be used:

 caching:
     storage:
         mode: redis
         redis:
             host: localhost
             port: 6379
             username: default
             password: heslo
             sentinel:
                 master: redismaster
                 nodes:
                     - host: localhost
                       port: 26379
                       password: sentinelpassword
                     - host: localhost
                       port: 26380
                       password: sentinelpassword
                     - host: localhost
                       port: 26381
                       password: sentinelpassword

The library used to connect to Redis, Lettuce, uses node registration information to automatically discover instances downstream from the master (in master/replica topology) or the sentinels (in sentinel topology). This means the IP address used to connect from the Caching Service is the IP address used to register, which with the above docker compose files is the container IP address. This means the Caching Service tries to connect using the container IP address, which does not resolve properly. The ports are published, however, so if the container IP addresses are aliased to localhost, the Caching Service can connect. Another option, if running on Linux, is to use a host network in the docker compose file.

Additional Storage Support

To add a new implementation it is necessary to provide the library with the implementation of the Storage.class and properly configure the Spring with the used implementation.

@ConditionalOnProperty(
    value = "caching.storage",
    havingValue = "custom"
)
@Bean
public Storage custom() {
    return new CustomStorage();
}

The example above shows the Configuration within the library that will use different storage than the default InMemory one.

It is possible to provide the custom implementation via the -Dloader.path property provided on startup of the Caching Service.

How do you run for local development

The Caching Service is a Spring Boot application. You can either add it as a run configuration and run it together with other services, or the npm command to run API ML also runs the caching service with in memory storage.

Command to run full set of api-layer: npm run api-layer. If you are looking for the Continuous Integration set up run: npm run api-layer-ci

In local usage, the Caching Service will run at https://localhost:10016. The API path is /cachingservice/api/v1/cache/${path-params-as-needed}. For example, https://localhost:10016/cachingservice/api/v1/cache/my-key retrieves the cache entry using the key 'my-key'.

Configuration properties

The Caching Service uses the standard application.yml structure for configuration.

apiml.service.routes only specifies one API route as there is no need for web socket or UI routes. caching.storage this property is reserved for the setup of the proper storage within the Caching Service.