diff --git a/caching-service/README.md b/caching-service/README.md new file mode 100644 index 0000000000..6295410190 --- /dev/null +++ b/caching-service/README.md @@ -0,0 +1,69 @@ +# 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. + +The initial implementation of the service will depend on VSAM to store the key/value pairs, as VSAM is a native z/OS solution to storing key/value pairs. Eventually, there will be other implementations for solutions such as MQs. As such, this needs to be taken into account for the initial design document. + +## 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](cachingServiceStructure.png "Architecture of the service") + +## 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](#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 + +TO BE DONE + +### 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 mock. + +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. diff --git a/caching-service/build.gradle b/caching-service/build.gradle new file mode 100644 index 0000000000..c497bd4b6f --- /dev/null +++ b/caching-service/build.gradle @@ -0,0 +1,82 @@ +buildscript { + repositories mavenRepositories + dependencies { + classpath("org.springframework.boot:spring-boot-gradle-plugin:${springBootVersion}") + classpath("gradle.plugin.com.gorylenko.gradle-git-properties:gradle-git-properties:${gradleGitPropertiesVersion}") + } +} + +normalization { + runtimeClasspath { + ignore("**/*git.properties*") + ignore("**/*build-info.properties*") + } +} + +apply plugin: 'org.springframework.boot' +apply plugin: 'com.gorylenko.gradle-git-properties' + +springBoot { + // This statement tells the Gradle Spring Boot plugin to generate a file + // build/resources/main/META-INF/build-info.properties that is picked up by Spring Boot to display + // via /info endpoint + buildInfo { + properties { + // Generate extra build info: + additionalProperties = [ + by: System.properties['user.name'], + operatingSystem: "${System.properties['os.name']} (${System.properties['os.version']})", + number: System.getenv('BUILD_NUMBER') ? System.getenv('BUILD_NUMBER') : "n/a", + machine: InetAddress.localHost.hostName + ] + } + } +} + +gitProperties { + gitPropertiesDir = new File("${project.rootDir}/${name}/build/resources/main/META-INF") +} + +dependencies { + compile project(':common-service-core') + compile project(':zaas-client') + + implementation project(':onboarding-enabler-spring') + + compile libraries.jjwt + compile libraries.jjwt_impl + compile libraries.jjwt_jackson + + implementation libraries.springFox + implementation libraries.spring_boot_starter + implementation libraries.spring_boot_starter_actuator + implementation libraries.spring_boot_starter_web + implementation libraries.spring_boot_starter_websocket + + implementation libraries.bootstrap + implementation libraries.jquery + + implementation libraries.gson + compileOnly libraries.lombok + annotationProcessor libraries.lombok + + testImplementation libraries.spring_boot_starter_test +} + + +bootJar.archiveName = "${bootJar.baseName}.jar" + +jar { + enabled = true + archiveName = "${jar.baseName}-thin.jar" + + def libClassPathEntries = configurations.runtimeClasspath.files.collect { + "lib/" + it.getName() + } + doFirst { + manifest { + attributes "Class-Path": libClassPathEntries.join(" "), + "Main-Class": "org.zowe.apiml.caching.CachingService" + } + } +} diff --git a/caching-service/cachingServiceStructure.png b/caching-service/cachingServiceStructure.png new file mode 100644 index 0000000000..8ad397b90e Binary files /dev/null and b/caching-service/cachingServiceStructure.png differ diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/CachingService.java b/caching-service/src/main/java/org/zowe/apiml/caching/CachingService.java new file mode 100644 index 0000000000..56e457169c --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/CachingService.java @@ -0,0 +1,25 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.zowe.apiml.enable.EnableApiDiscovery; + +@SpringBootApplication +@EnableApiDiscovery +public class CachingService { + + public static void main(String[] args) { + SpringApplication app = new SpringApplication(CachingService.class); + app.setLogStartupInfo(false); + app.run(args); + } +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/api/CachingController.java b/caching-service/src/main/java/org/zowe/apiml/caching/api/CachingController.java new file mode 100644 index 0000000000..157842a861 --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/api/CachingController.java @@ -0,0 +1,256 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.api; + +import io.swagger.annotations.ApiOperation; +import lombok.RequiredArgsConstructor; +import org.apache.commons.lang3.StringUtils; +import org.springframework.context.annotation.Import; +import org.springframework.http.HttpStatus; +import org.springframework.http.MediaType; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.*; +import org.zowe.apiml.caching.exceptions.CachingPayloadException; +import org.zowe.apiml.caching.model.KeyValue; +import org.zowe.apiml.caching.service.Storage; +import org.zowe.apiml.message.core.Message; +import org.zowe.apiml.message.core.MessageService; +import org.zowe.apiml.zaasclient.config.DefaultZaasClientConfiguration; +import org.zowe.apiml.zaasclient.exception.ZaasClientErrorCodes; +import org.zowe.apiml.zaasclient.exception.ZaasClientException; +import org.zowe.apiml.zaasclient.service.ZaasClient; +import org.zowe.apiml.zaasclient.service.ZaasToken; + +import javax.servlet.http.Cookie; +import javax.servlet.http.HttpServletRequest; +import java.util.Arrays; + +@RestController +@RequiredArgsConstructor +@RequestMapping("/api/v1") +@Import(DefaultZaasClientConfiguration.class) +public class CachingController { + private static final String TOKEN_COOKIE_PREFIX = "apimlAuthenticationToken"; + private static final String KEY_NOT_IN_CACHE_MESSAGE = "org.zowe.apiml.cache.keyNotInCache"; + + private final Storage storage; + private final ZaasClient zaasClient; + private final MessageService messageService; + + @GetMapping(value = "/cache/{key}", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ApiOperation(value = "Retrieves a specific value in the cache", + notes = "Value returned is for the provided {key}") + @ResponseBody + public ResponseEntity getValue(@PathVariable String key, HttpServletRequest request) { + String serviceId; + try { + ZaasToken token = queryTokenFromRequest(request); + serviceId = token.getUserId(); + } catch (ZaasClientException e) { + return handleZaasClientException(e, request); + } + + if (key == null) { + return noKeyProvidedResponse(serviceId); + } + + KeyValue readPair = storage.read(serviceId, key); + + if (readPair == null) { + Message message = messageService.createMessage(KEY_NOT_IN_CACHE_MESSAGE, key, serviceId); + return new ResponseEntity<>(message.mapToView(), HttpStatus.NOT_FOUND); + } + + return new ResponseEntity<>(readPair, HttpStatus.OK); + } + + @GetMapping(value = "/cache", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ApiOperation(value = "Retrieves all values in the cache", + notes = "Values returned for the calling service") + @ResponseBody + public ResponseEntity getAllValues(HttpServletRequest request) { + String serviceId; + try { + ZaasToken token = queryTokenFromRequest(request); + serviceId = token.getUserId(); + } catch (ZaasClientException e) { + return handleZaasClientException(e, request); + } + + return new ResponseEntity<>(storage.readForService(serviceId), HttpStatus.OK); + } + + @PostMapping(value = "/cache", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ApiOperation(value = "Create a new key in the cache", + notes = "A new key-value pair will be added to the cache") + @ResponseBody + public ResponseEntity createKey(@RequestBody KeyValue keyValue, HttpServletRequest request) { + String serviceId; + try { + ZaasToken token = queryTokenFromRequest(request); + serviceId = token.getUserId(); + + checkForInvalidPayload(keyValue); + } catch (ZaasClientException e) { + return handleZaasClientException(e, request); + } catch (CachingPayloadException e) { + return invalidPayloadResponse(e, keyValue); + } + + KeyValue createdPair = storage.create(serviceId, keyValue); + + if (createdPair == null) { + Message message = messageService.createMessage("org.zowe.apiml.cache.keyCollision", keyValue.getKey()); + return new ResponseEntity<>(message.mapToView(), HttpStatus.CONFLICT); + } + + return new ResponseEntity<>(HttpStatus.CREATED); + } + + @PutMapping(value = "/cache", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ApiOperation(value = "Update key in the cache", + notes = "Value at the key in the provided key-value pair will be updated to the provided value") + @ResponseBody + public ResponseEntity update(@RequestBody KeyValue keyValue, HttpServletRequest request) { + String serviceId; + try { + ZaasToken token = queryTokenFromRequest(request); + serviceId = token.getUserId(); + + checkForInvalidPayload(keyValue); + } catch (ZaasClientException e) { + return handleZaasClientException(e, request); + } catch (CachingPayloadException e) { + return invalidPayloadResponse(e, keyValue); + } + + KeyValue updatedPair = storage.update(serviceId, keyValue); + + if (updatedPair == null) { + Message message = messageService.createMessage(KEY_NOT_IN_CACHE_MESSAGE, keyValue.getKey(), serviceId); + return new ResponseEntity<>(message.mapToView(), HttpStatus.NOT_FOUND); + } + + return new ResponseEntity<>(HttpStatus.NO_CONTENT); + } + + @DeleteMapping(value = "/cache/{key}", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ApiOperation(value = "Delete key from the cache", + notes = "Will delete key-value pair for the provided {key}") + @ResponseBody + public ResponseEntity delete(@PathVariable String key, HttpServletRequest request) { + String serviceId; + try { + ZaasToken token = queryTokenFromRequest(request); + serviceId = token.getUserId(); + } catch (ZaasClientException e) { + return handleZaasClientException(e, request); + } + + if (key == null) { + return noKeyProvidedResponse(serviceId); + } + + KeyValue deletedPair = storage.delete(serviceId, key); + + if (deletedPair == null) { + Message message = messageService.createMessage(KEY_NOT_IN_CACHE_MESSAGE, key, serviceId); + return new ResponseEntity<>(message.mapToView(), HttpStatus.NOT_FOUND); + } + + return new ResponseEntity<>(HttpStatus.NO_CONTENT); + } + + private void checkForInvalidPayload(KeyValue keyValue) throws CachingPayloadException { + if (keyValue == null) { + throw new CachingPayloadException("No KeyValue provided in the payload"); + } + + if (keyValue.getValue() == null) { + throw new CachingPayloadException("No value provided in the payload"); + } + + String key = keyValue.getKey(); + if (key == null) { + throw new CachingPayloadException("No key provided in the payload"); + } + if (!StringUtils.isAlphanumeric(key)) { + throw new CachingPayloadException("Key is not alphanumeric"); + } + } + + private ResponseEntity invalidPayloadResponse(CachingPayloadException e, KeyValue keyValue) { + Message message = messageService.createMessage("org.zowe.apiml.cache.invalidPayload", keyValue, e.getMessage()); + return new ResponseEntity<>(message.mapToView(), HttpStatus.BAD_REQUEST); + } + + private ResponseEntity noKeyProvidedResponse(String serviceId) { + Message message = messageService.createMessage("org.zowe.apiml.cache.keyNotProvided", serviceId); + return new ResponseEntity<>(message.mapToView(), HttpStatus.BAD_REQUEST); + } + + private ZaasToken queryTokenFromRequest(HttpServletRequest request) throws ZaasClientException { + String jwtToken = getJwtTokenFromCookie(request); + ZaasToken zaasToken = zaasClient.query(jwtToken); + + if (zaasToken == null) { + throw new ZaasClientException(ZaasClientErrorCodes.INVALID_JWT_TOKEN, "Queried token is null"); + } + if (zaasToken.isExpired()) { + throw new ZaasClientException(ZaasClientErrorCodes.EXPIRED_JWT_EXCEPTION, "Queried token is expired"); + } + + return zaasToken; + } + + private String getJwtTokenFromCookie(HttpServletRequest request) { + Cookie[] cookies = request.getCookies(); + if (cookies == null) { + return null; + } + return Arrays.stream(cookies) + .filter(cookie -> cookie.getName().equals(TOKEN_COOKIE_PREFIX)) + .filter(cookie -> !cookie.getValue().isEmpty()) + .findFirst() + .map(Cookie::getValue) + .orElse(null); + } + + private ResponseEntity handleZaasClientException(ZaasClientException e, HttpServletRequest request) { + String requestUrl = request.getRequestURL().toString(); + Message message; + HttpStatus statusCode; + + switch (e.getErrorCode()) { + case TOKEN_NOT_PROVIDED: + statusCode = HttpStatus.BAD_REQUEST; + message = messageService.createMessage("org.zowe.apiml.security.query.tokenNotProvided", requestUrl); + break; + case INVALID_JWT_TOKEN: + statusCode = HttpStatus.UNAUTHORIZED; + message = messageService.createMessage("org.zowe.apiml.security.query.invalidToken", requestUrl); + break; + case EXPIRED_JWT_EXCEPTION: + statusCode = HttpStatus.UNAUTHORIZED; + message = messageService.createMessage("org.zowe.apiml.security.expiredToken", requestUrl); + break; + case SERVICE_UNAVAILABLE: + statusCode = HttpStatus.NOT_FOUND; + message = messageService.createMessage("org.zowe.apiml.cache.gatewayUnavailable", requestUrl, e.getMessage()); + break; + default: + statusCode = HttpStatus.INTERNAL_SERVER_ERROR; + message = messageService.createMessage("org.zowe.apiml.common.internalRequestError", requestUrl, e.getMessage(), e.getCause()); + break; + } + + return new ResponseEntity<>(message.mapToView(), statusCode); + } +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/config/MessageConfiguration.java b/caching-service/src/main/java/org/zowe/apiml/caching/config/MessageConfiguration.java new file mode 100644 index 0000000000..ab1392f20e --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/config/MessageConfiguration.java @@ -0,0 +1,25 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.config; + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.zowe.apiml.message.core.MessageService; +import org.zowe.apiml.message.yaml.YamlMessageServiceInstance; + +@Configuration +public class MessageConfiguration { + @Bean + public MessageService messageService() { + MessageService messageService = YamlMessageServiceInstance.getInstance(); + messageService.loadMessages("/caching-log-messages.yml"); + return messageService; + } +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/config/StorageConfiguration.java b/caching-service/src/main/java/org/zowe/apiml/caching/config/StorageConfiguration.java new file mode 100644 index 0000000000..fc6a02ea42 --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/config/StorageConfiguration.java @@ -0,0 +1,25 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.config; + +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.zowe.apiml.caching.service.Storage; +import org.zowe.apiml.caching.service.inmemory.InMemoryStorage; + +@Configuration +public class StorageConfiguration { + @ConditionalOnMissingBean(Storage.class) + @Bean + public Storage inMemory() { + return new InMemoryStorage(); + } +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/config/SwaggerConfig.java b/caching-service/src/main/java/org/zowe/apiml/caching/config/SwaggerConfig.java new file mode 100644 index 0000000000..292528e008 --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/config/SwaggerConfig.java @@ -0,0 +1,57 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.config; + +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import springfox.documentation.builders.PathSelectors; +import springfox.documentation.builders.RequestHandlerSelectors; +import springfox.documentation.service.ApiInfo; +import springfox.documentation.spi.DocumentationType; +import springfox.documentation.spring.web.plugins.Docket; +import springfox.documentation.swagger2.annotations.EnableSwagger2; + +import java.util.Collections; + +@Configuration +@EnableSwagger2 +public class SwaggerConfig { + + @Value("${apiml.service.title}") + private String apiTitle; + + @Value("${apiml.service.apiInfo[0].version}") + private String apiVersion; + + @Value("${apiml.service.description}") + private String apiDescription; + + @Bean + public Docket api() { + return new Docket(DocumentationType.SWAGGER_2) + .select() + .apis(RequestHandlerSelectors.any()) + .paths(PathSelectors.ant("/api/v1/**")) + .build() + .apiInfo( + new ApiInfo( + apiTitle, + apiDescription, + apiVersion, + null, + null, + null, + null, + Collections.emptyList() + ) + ); + } +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/exceptions/CachingPayloadException.java b/caching-service/src/main/java/org/zowe/apiml/caching/exceptions/CachingPayloadException.java new file mode 100644 index 0000000000..b123884551 --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/exceptions/CachingPayloadException.java @@ -0,0 +1,16 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.exceptions; + +public class CachingPayloadException extends Exception { + public CachingPayloadException(String message) { + super(message); + } +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/model/KeyValue.java b/caching-service/src/main/java/org/zowe/apiml/caching/model/KeyValue.java new file mode 100644 index 0000000000..490e8b277d --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/model/KeyValue.java @@ -0,0 +1,22 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.model; + +import com.fasterxml.jackson.annotation.JsonInclude; +import lombok.Data; +import lombok.RequiredArgsConstructor; + +@RequiredArgsConstructor +@JsonInclude(JsonInclude.Include.NON_EMPTY) +@Data +public class KeyValue { + private final String key; + private final String value; +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/service/Storage.java b/caching-service/src/main/java/org/zowe/apiml/caching/service/Storage.java new file mode 100644 index 0000000000..7c44a11aa4 --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/service/Storage.java @@ -0,0 +1,64 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.service; + +import org.zowe.apiml.caching.model.KeyValue; + +import java.util.Map; + +/** + * Every supported storage backend needs to have an implementation of the Storage. + */ +public interface Storage { + /** + * Store new KeyValue pair in the storage. If there is a key collision null is returned. + * + * @param serviceId Id of the service to store the value for + * @param toCreate KeyValue pair to be created. + * @return The stored KeyValue pair or null. + */ + KeyValue create(String serviceId, KeyValue toCreate); + + /** + * Returns the keys associated with the provided keys. + * + * @param serviceId Id of the service to read value for + * @param key key to lookup + * @return KeyValue associated with the value + */ + KeyValue read(String serviceId, String key); + + /** + * Replaces the value for the given key with the new value. If there is no existing key/value pair null is returned. + * + * @param serviceId Id of the service to store the value for. + * @param toUpdate Value to store instead of the original one. + * @return Updated key/value pair or null. + */ + KeyValue update(String serviceId, KeyValue toUpdate); + + /** + * Delete the key/value pair if it exists within the context of the service. If there is none existing null + * is returned. + * + * @param serviceId Id of the service to delete the value for. + * @param toDelete Key to delete from the storage. + * @return Deleted key/value pair or null. + */ + KeyValue delete(String serviceId, String toDelete); + + /** + * Return all the key/value pairs for given service id. + * + * @param serviceId Id of the service to load all key/value pairs + * @return Map with the key/value pairs or null if there is none existing. + */ + Map readForService(String serviceId); +} diff --git a/caching-service/src/main/java/org/zowe/apiml/caching/service/inmemory/InMemoryStorage.java b/caching-service/src/main/java/org/zowe/apiml/caching/service/inmemory/InMemoryStorage.java new file mode 100644 index 0000000000..85bf3be181 --- /dev/null +++ b/caching-service/src/main/java/org/zowe/apiml/caching/service/inmemory/InMemoryStorage.java @@ -0,0 +1,77 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.service.inmemory; + +import org.zowe.apiml.caching.model.KeyValue; +import org.zowe.apiml.caching.service.Storage; + +import java.util.HashMap; +import java.util.Map; + +public class InMemoryStorage implements Storage { + private Map> storage = new HashMap<>(); + + public InMemoryStorage() { + } + + protected InMemoryStorage(Map> storage) { + this.storage = storage; + } + + @Override + public KeyValue create(String serviceId, KeyValue toCreate) { + storage.computeIfAbsent(serviceId, k -> new HashMap<>()); + Map serviceStorage = storage.get(serviceId); + serviceStorage.put(toCreate.getKey(), toCreate); + return toCreate; + } + + @Override + public KeyValue read(String serviceId, String key) { + Map serviceSpecificStorage = storage.get(serviceId); + if (serviceSpecificStorage == null) { + return null; + } + + return serviceSpecificStorage.get(key); + } + + @Override + public KeyValue update(String serviceId, KeyValue toUpdate) { + String keyToUpdate = toUpdate.getKey(); + if (isKeyNotInCache(serviceId, keyToUpdate)) { + return null; + } + + Map serviceStorage = storage.get(serviceId); + serviceStorage.put(keyToUpdate, toUpdate); + return toUpdate; + } + + @Override + public KeyValue delete(String serviceId, String toDelete) { + if (isKeyNotInCache(serviceId, toDelete)) { + return null; + } + + Map serviceSpecificStorage = storage.get(serviceId); + return serviceSpecificStorage.remove(toDelete); + } + + @Override + public Map readForService(String serviceId) { + return storage.get(serviceId); + } + + private boolean isKeyNotInCache(String serviceId, String keyToTest) { + Map serviceSpecificStorage = storage.get(serviceId); + return serviceSpecificStorage == null || serviceSpecificStorage.get(keyToTest) == null; + } +} diff --git a/caching-service/src/main/resources/application.yml b/caching-service/src/main/resources/application.yml new file mode 100644 index 0000000000..2b01989ef9 --- /dev/null +++ b/caching-service/src/main/resources/application.yml @@ -0,0 +1,82 @@ +spring: + application: + name: Caching service + +apiml: + enabled: true + service: + preferIpAddress: false + + serviceId: cachingservice + title: Caching service for internal usage. + description: Service that provides caching API. + + discoveryServiceUrls: https://localhost:10011/eureka/ + + scheme: https + + hostname: localhost + port: 10016 + baseUrl: ${apiml.service.scheme}://${apiml.service.hostname}:${apiml.service.port} + contextPath: /${apiml.service.serviceId} + + homePageRelativeUrl: ${apiml.service.contextPath} + statusPageRelativeUrl: ${apiml.service.contextPath}/application/info + healthCheckRelativeUrl: ${apiml.service.contextPath}/application/health + + routes: + - gateway-url: "api/v1" + service-url: ${apiml.service.contextPath}/api/v1 + apiInfo: + - apiId: org.zowe.cachingservice + version: 1.0.0 + gatewayUrl: api/v1 + swaggerUrl: ${apiml.service.scheme}://${apiml.service.hostname}:${apiml.service.port}${apiml.service.contextPath}/v2/api-docs + documentationUrl: https://www.zowe.org + catalog: + tile: + id: zowe + title: Zowe Applications + description: Applications which are part of Zowe. + version: 1.0.0 + ssl: + enabled: true + verifySslCertificatesOfServices: true + protocol: ${server.ssl.protocol} + keyStoreType: ${server.ssl.keyStoreType} + trustStoreType: ${server.ssl.trustStoreType} + + keyAlias: ${server.ssl.keyAlias} + keyPassword: ${server.ssl.keyPassword} + keyStore: ${server.ssl.keyStore} + keyStorePassword: ${server.ssl.keyStorePassword} + trustStore: ${server.ssl.trustStore} + trustStorePassword: ${server.ssl.trustStorePassword} + customMetadata: + apiml: + enableUrlEncodedCharacters: true + gatewayPort: 10010 + gatewayAuthEndpoint: /api/v1/gateway/auth + corsEnabled: false + +server: + port: ${apiml.service.port} + + servlet: + contextPath: /${apiml.service.serviceId} + + ssl: + enabled: true + clientAuth: want + protocol: TLSv1.2 + enabled-protocols: TLSv1.2 + ciphers: TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384,TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256,TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256,TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 + keyStoreType: PKCS12 + trustStoreType: PKCS12 + + keyAlias: localhost + keyPassword: password + keyStore: keystore/localhost/localhost.keystore.p12 + keyStorePassword: password + trustStore: keystore/localhost/localhost.truststore.p12 + trustStorePassword: password diff --git a/caching-service/src/main/resources/banner.txt b/caching-service/src/main/resources/banner.txt new file mode 100644 index 0000000000..dc1287451e --- /dev/null +++ b/caching-service/src/main/resources/banner.txt @@ -0,0 +1 @@ +Caching Service diff --git a/caching-service/src/main/resources/caching-log-messages.yml b/caching-service/src/main/resources/caching-log-messages.yml new file mode 100644 index 0000000000..fe0e5a532a --- /dev/null +++ b/caching-service/src/main/resources/caching-log-messages.yml @@ -0,0 +1,59 @@ +messages: + # General messages (100 - 199) + - key: org.zowe.apiml.security.expiredToken + number: ZWEAT100 + type: ERROR + text: "Token is expired for URL '%s'" + reason: "The validity of the token is expired." + action: "Obtain a new token by performing an authentication request." + + # Query messages (130 - 140) + - key: org.zowe.apiml.security.query.invalidToken + number: ZWEAG130 + type: ERROR + text: "Token is not valid for URL '%s'" + reason: "The token is not valid." + action: "Provide a valid token." + + - key: org.zowe.apiml.security.query.tokenNotProvided + number: ZWEAG131 + type: ERROR + text: "No authorization token provided for URL '%s'" + reason: "No authorization token is provided." + action: "Provide a valid authorization token." + + - key: org.zowe.apiml.cache.invalidPayload + number: ZWECS130 + type: ERROR + text: "Payload '%s' is not valid: '%s'." + reason: "The payload is not in valid JSON format." + action: "Provide a payload in JSON format." + + - key: org.zowe.apiml.cache.keyNotInCache + number: ZWECS131 + type: ERROR + text: "Key '%s' is not in the cache for service '%s'" + reason: "Cache does not contain the provided key." + action: "Add a key-value pair to the cache using the key or operate on an existing key in the cache." + + - key: org.zowe.apiml.cache.keyNotProvided + number: ZWECS132 + type: ERROR + text: "No cache key provided." + reason: "No cache key was provided." + action: "Provide a key that is in the cache." + + - key: org.zowe.apiml.cache.keyCollision + number: ZWECS133 + type: ERROR + text: "Adding key '%s' resulted in a collision in the cache." + reason: "Key is already in the cache." + action: "Update or delete the key, or add a different key." + + # Service specific messages (700 - 799) + - key: org.zowe.apiml.cache.gatewayUnavailable + number: ZWECS700 + type: ERROR + text: "Gateway service is not available at URL '%s'. Error returned: '%s'" + reason: "The gateway service is not available." + action: "Make sure that the gateway service is running and is accessible by the URL provided in the message." diff --git a/caching-service/src/test/java/org/zowe/apiml/caching/api/CachingControllerTest.java b/caching-service/src/test/java/org/zowe/apiml/caching/api/CachingControllerTest.java new file mode 100644 index 0000000000..dd63022a25 --- /dev/null +++ b/caching-service/src/test/java/org/zowe/apiml/caching/api/CachingControllerTest.java @@ -0,0 +1,312 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.api; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.Arguments; +import org.junit.jupiter.params.provider.MethodSource; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.mock.web.MockHttpServletRequest; +import org.zowe.apiml.caching.model.KeyValue; +import org.zowe.apiml.caching.service.Storage; +import org.zowe.apiml.message.api.ApiMessageView; +import org.zowe.apiml.message.core.MessageService; +import org.zowe.apiml.message.yaml.YamlMessageService; +import org.zowe.apiml.zaasclient.exception.ZaasClientErrorCodes; +import org.zowe.apiml.zaasclient.exception.ZaasClientException; +import org.zowe.apiml.zaasclient.service.ZaasClient; +import org.zowe.apiml.zaasclient.service.ZaasToken; + +import javax.servlet.http.Cookie; +import java.util.HashMap; +import java.util.Map; +import java.util.stream.Stream; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.Is.is; +import static org.hamcrest.core.IsNull.nullValue; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; + +public class CachingControllerTest { + private static final String SERVICE_ID = "test-service"; + private static final String KEY = "key"; + private static final String VALUE = "value"; + + private static final KeyValue KEY_VALUE = new KeyValue(KEY, VALUE); + private static final ZaasToken TOKEN = new ZaasToken(); + + private MockHttpServletRequest mockRequest; + + private Storage mockStorage; + private ZaasClient mockZaasClient; + private final MessageService messageService = new YamlMessageService("/caching-log-messages.yml"); + private CachingController underTest; + + @BeforeEach + void setUp() throws ZaasClientException { + mockRequest = new MockHttpServletRequest(); + mockStorage = mock(Storage.class); + mockZaasClient = mock(ZaasClient.class); + when(mockZaasClient.query(any())).thenReturn(TOKEN); + + TOKEN.setUserId(SERVICE_ID); + underTest = new CachingController(mockStorage, mockZaasClient, messageService); + } + + @Test + void givenStorageReturnsValidValue_whenGetByKey_thenReturnProperValue() { + when(mockStorage.read(SERVICE_ID, KEY)).thenReturn(KEY_VALUE); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.OK)); + + KeyValue body = (KeyValue) response.getBody(); + assertThat(body.getValue(), is(VALUE)); + } + + @Test + void givenNoKey_whenGetByKey_thenResponseBadRequest() { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.keyNotProvided", SERVICE_ID).mapToView(); + + ResponseEntity response = underTest.getValue(null, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.BAD_REQUEST)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenStoreWithNoKey_whenGetByKey_thenResponseNotFound() { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.keyNotInCache", KEY, SERVICE_ID).mapToView(); + when(mockStorage.read(any(), any())).thenReturn(null); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.NOT_FOUND)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenStorageReturnsValidValues_whenGetByService_thenReturnProperValues() { + Map values = new HashMap<>(); + values.put(KEY, new KeyValue("key2", VALUE)); + when(mockStorage.readForService(SERVICE_ID)).thenReturn(values); + + ResponseEntity response = underTest.getAllValues(mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.OK)); + + Map result = (Map) response.getBody(); + assertThat(result, is(values)); + } + + @Test + void givenStorage_whenCreateKey_thenResponseCreated() { + when(mockStorage.create(SERVICE_ID, KEY_VALUE)).thenReturn(KEY_VALUE); + + ResponseEntity response = underTest.createKey(KEY_VALUE, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.CREATED)); + assertThat(response.getBody(), is(nullValue())); + } + + @Test + void givenStorageWithExistingKey_whenCreateKey_thenResponseConflict() { + when(mockStorage.create(SERVICE_ID, KEY_VALUE)).thenReturn(null); + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.keyCollision", KEY).mapToView(); + + ResponseEntity response = underTest.createKey(KEY_VALUE, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.CONFLICT)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenStorageWithKey_whenUpdateKey_thenResponseNoContent() { + when(mockStorage.update(SERVICE_ID, KEY_VALUE)).thenReturn(KEY_VALUE); + + ResponseEntity response = underTest.update(KEY_VALUE, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.NO_CONTENT)); + assertThat(response.getBody(), is(nullValue())); + } + + @Test + void givenStorageWithNoKey_whenUpdateKey_thenResponseNotFound() { + when(mockStorage.update(SERVICE_ID, KEY_VALUE)).thenReturn(null); + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.keyNotInCache", KEY, SERVICE_ID).mapToView(); + + ResponseEntity response = underTest.update(KEY_VALUE, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.NOT_FOUND)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenStorageWithKey_whenDeleteKey_thenResponseNoContent() { + when(mockStorage.delete(any(), any())).thenReturn(KEY_VALUE); + + ResponseEntity response = underTest.delete(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.NO_CONTENT)); + assertThat(response.getBody(), is(nullValue())); + } + + @Test + void givenNoKey_whenDeleteKey_thenResponseBadRequest() { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.keyNotProvided").mapToView(); + + ResponseEntity response = underTest.delete(null, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.BAD_REQUEST)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenStorageWithNoKey_whenDeleteKey_thenResponseNotFound() { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.keyNotInCache", KEY, SERVICE_ID).mapToView(); + when(mockStorage.delete(any(), any())).thenReturn(null); + + ResponseEntity response = underTest.delete(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.NOT_FOUND)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenNoPayload_whenValidatePayload_thenResponseBadRequest() { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.invalidPayload", + null, "No KeyValue provided in the payload").mapToView(); + + ResponseEntity response = underTest.createKey(null, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.BAD_REQUEST)); + assertThat(response.getBody(), is(expectedBody)); + } + + @ParameterizedTest + @MethodSource("provideStringsForGivenBadKeyValue") + void givenBadKeyValue_whenValidatePayload_thenResponseBadRequest(String key, String value, String errMessage) { + KeyValue keyValue = new KeyValue(key, value); + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.invalidPayload", + keyValue, errMessage).mapToView(); + + ResponseEntity response = underTest.createKey(keyValue, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.BAD_REQUEST)); + assertThat(response.getBody(), is(expectedBody)); + } + + private static Stream provideStringsForGivenBadKeyValue() { + return Stream.of( + Arguments.of("key", null, "No value provided in the payload"), + Arguments.of(null, "value", "No key provided in the payload"), + Arguments.of("key ", "value", "Key is not alphanumeric") + ); + } + + @Test + void givenNoToken_whenQueryToken_thenResponseBadRequest() throws ZaasClientException { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.security.query.tokenNotProvided", + mockRequest.getRequestURL().toString()).mapToView(); + when(mockZaasClient.query(any())).thenThrow(new ZaasClientException(ZaasClientErrorCodes.TOKEN_NOT_PROVIDED)); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.BAD_REQUEST)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenInvalidToken_whenQueryToken_thenResponseUnauthorized() throws ZaasClientException { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.security.query.invalidToken", + mockRequest.getRequestURL().toString()).mapToView(); + when(mockZaasClient.query(any())).thenThrow(new ZaasClientException(ZaasClientErrorCodes.INVALID_JWT_TOKEN)); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.UNAUTHORIZED)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenExpiredToken_whenQueryToken_thenResponseUnauthorized() throws ZaasClientException { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.security.expiredToken", + mockRequest.getRequestURL().toString()).mapToView(); + when(mockZaasClient.query(any())).thenThrow(new ZaasClientException(ZaasClientErrorCodes.EXPIRED_JWT_EXCEPTION)); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.UNAUTHORIZED)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenNoGateway_whenQueryToken_thenResponseUnauthorized() throws ZaasClientException { + ZaasClientException zaasException = new ZaasClientException(ZaasClientErrorCodes.SERVICE_UNAVAILABLE, "This is an error"); + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.cache.gatewayUnavailable", + mockRequest.getRequestURL().toString(), zaasException.getMessage()).mapToView(); + when(mockZaasClient.query(any())).thenThrow(zaasException); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.NOT_FOUND)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenRandomError_whenQueryToken_thenResponseInternalError() throws ZaasClientException { + Throwable errCause = new Exception("This is an error"); + ZaasClientException zaasException = new ZaasClientException(ZaasClientErrorCodes.GENERIC_EXCEPTION, errCause); + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.common.internalRequestError", + mockRequest.getRequestURL().toString(), zaasException.getMessage(), zaasException.getCause()).mapToView(); + when(mockZaasClient.query(any())).thenThrow(zaasException); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.INTERNAL_SERVER_ERROR)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenToken_whenTokenQueryReturnsNull_thenResponseUnauthorized() throws ZaasClientException { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.security.query.invalidToken", + mockRequest.getRequestURL().toString()).mapToView(); + when(mockZaasClient.query(any())).thenReturn(null); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.UNAUTHORIZED)); + assertThat(response.getBody(), is(expectedBody)); + } + + @Test + void givenToken_whenTokenQueryReturnsExpiredToken_thenResponseUnauthorized() throws ZaasClientException { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.security.expiredToken", + mockRequest.getRequestURL().toString()).mapToView(); + ZaasToken expiredToken = new ZaasToken(); + expiredToken.setExpired(true); + when(mockZaasClient.query(any())).thenReturn(expiredToken); + + ResponseEntity response = underTest.getValue(KEY, mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.UNAUTHORIZED)); + assertThat(response.getBody(), is(expectedBody)); + } + + @ParameterizedTest + @MethodSource("cookieTestProvider") + void givenCookieWithWrongAuthentication_whenQueryToken_thenResponseUnauthorized + (String cookieName, String cookieValue, String queryToken) throws ZaasClientException { + ApiMessageView expectedBody = messageService.createMessage("org.zowe.apiml.security.query.invalidToken", + mockRequest.getRequestURL().toString()).mapToView(); + + Cookie[] cookies = new Cookie[]{new Cookie(cookieName, cookieValue)}; + mockRequest.setCookies(cookies); + when(mockZaasClient.query(queryToken)).thenReturn(null); + + ResponseEntity response = underTest.getAllValues(mockRequest); + assertThat(response.getStatusCode(), is(HttpStatus.UNAUTHORIZED)); + assertThat(response.getBody(), is(expectedBody)); + } + + private static Stream cookieTestProvider() { + return Stream.of( + Arguments.of("my", "cookie", null), + Arguments.of("apimlAuthenticationToken", "bad_token", "bad_token"), + Arguments.of("apimlAuthenticationToken", "", null) + ); + } +} diff --git a/caching-service/src/test/java/org/zowe/apiml/caching/service/inmemory/InMemoryStorageTest.java b/caching-service/src/test/java/org/zowe/apiml/caching/service/inmemory/InMemoryStorageTest.java new file mode 100644 index 0000000000..748a1c46eb --- /dev/null +++ b/caching-service/src/test/java/org/zowe/apiml/caching/service/inmemory/InMemoryStorageTest.java @@ -0,0 +1,128 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ +package org.zowe.apiml.caching.service.inmemory; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.zowe.apiml.caching.model.KeyValue; + +import java.util.HashMap; +import java.util.Map; + +import static org.hamcrest.CoreMatchers.nullValue; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.Is.is; + +public class InMemoryStorageTest { + private InMemoryStorage underTest; + + private Map> testingStorage; + private final String serviceId = "acme"; + + @BeforeEach + void setUp() { + testingStorage = new HashMap<>(); + underTest = new InMemoryStorage(testingStorage); + } + + @Test + void givenDefaultStorageConstructor_whenStorageConstructed_thenCanUseStorage() { + underTest = new InMemoryStorage(); + underTest.create(serviceId, new KeyValue("key", "value")); + + KeyValue result = underTest.read(serviceId, "key"); + assertThat(result.getKey(), is("key")); + assertThat(result.getValue(), is("value")); + } + + @Test + void givenThereIsNoValueForService_whenValueIsStored_thenItIsStored() { + underTest.create(serviceId, new KeyValue("username", "ValidName")); + + KeyValue result = testingStorage.get(serviceId).get("username"); + assertThat(result.getKey(), is("username")); + assertThat(result.getValue(), is("ValidName")); + } + + @Test + void givenThereIsValueForService_whenValueIsUpdated_thenItIsReplaced() { + Map serviceStorage = new HashMap<>(); + testingStorage.put(serviceId, serviceStorage); + serviceStorage.put("username", new KeyValue("username", "Name 1")); + underTest.update(serviceId, new KeyValue("username", "ValidName")); + + KeyValue result = testingStorage.get(serviceId).get("username"); + assertThat(result.getKey(), is("username")); + assertThat(result.getValue(), is("ValidName")); + } + + @Test + void givenThereIsNoServiceCache_whenValueIsUpdated_thenNullIsReturned() { + KeyValue result = underTest.update(serviceId, new KeyValue("username", "Name 1")); + assertThat(result, is(nullValue())); + } + + @Test + void givenThereIsNoKey_whenValueIsUpdated_thenNullIsReturned() { + testingStorage.put(serviceId, new HashMap<>()); + KeyValue result = underTest.update(serviceId, new KeyValue("bad key", "Name 1")); + assertThat(result, is(nullValue())); + } + + @Test + void givenValueWasAlreadyAddedToTheStorage_whenRequested_thenItWillBeReturned() { + Map serviceStorage = new HashMap<>(); + testingStorage.put(serviceId, serviceStorage); + serviceStorage.put("username", new KeyValue("username", "Name 1")); + + KeyValue result = underTest.read(serviceId, "username"); + assertThat(result.getKey(), is("username")); + assertThat(result.getValue(), is("Name 1")); + } + + @Test + void givenNoValueWasStoredForTheService_whenRequested_thenNullWillBeReturned() { + KeyValue result = underTest.read(serviceId, "username"); + assertThat(result, is(nullValue())); + } + + @Test + void givenServiceHasStoredValues_whenLoadingAllForService_thenAllAreReturned() { + Map serviceStorage = new HashMap<>(); + testingStorage.put(serviceId, serviceStorage); + serviceStorage.put("username", new KeyValue("username", "Name 1")); + + Map result = underTest.readForService(serviceId); + assertThat(result.containsKey("username"), is(true)); + } + + @Test + void givenKeyDoesntExist_whenDeletionRequested_thenNullIsReturned() { + testingStorage.put(serviceId, new HashMap<>()); + KeyValue result = underTest.delete(serviceId, "nonexistent"); + assertThat(result, is(nullValue())); + } + + @Test + void givenServiceStorageDoesntExist_whenDeletionRequest_thenNullIsReturned() { + KeyValue result = underTest.delete(serviceId, "nonexistent"); + assertThat(result, is(nullValue())); + } + + @Test + void givenKeyExists_whenDeletionRequested_thenKeyValueIsReturnedAndKeyIsRemoved() { + Map serviceStorage = new HashMap<>(); + testingStorage.put(serviceId, serviceStorage); + serviceStorage.put("username", new KeyValue("username", "Name 1")); + + underTest.delete(serviceId, "username"); + assertThat(serviceStorage.containsKey("username"), is(false)); + } +} diff --git a/caching-service/src/test/resources/application.yml b/caching-service/src/test/resources/application.yml new file mode 100644 index 0000000000..e6cedc2582 --- /dev/null +++ b/caching-service/src/test/resources/application.yml @@ -0,0 +1,82 @@ +spring: + application: + name: Caching service + +apiml: + enabled: true + service: + preferIpAddress: false + + serviceId: cachingservice + title: Caching service for the internal usage. + description: Service which provides caching API + + discoveryServiceUrls: https://localhost:10011/eureka/ + + scheme: https + + hostname: localhost + port: 10016 + baseUrl: ${apiml.service.scheme}://${apiml.service.hostname}:${apiml.service.port} + contextPath: /${apiml.service.serviceId} + + homePageRelativeUrl: ${apiml.service.contextPath} + statusPageRelativeUrl: ${apiml.service.contextPath}/application/info + healthCheckRelativeUrl: ${apiml.service.contextPath}/application/health + + routes: + - gateway-url: "api/v1" + service-url: ${apiml.service.contextPath}/api/v1 + apiInfo: + - apiId: org.zowe.cachingservice + version: 1.0.0 + gatewayUrl: api/v1 + swaggerUrl: ${apiml.service.scheme}://${apiml.service.hostname}:${apiml.service.port}${apiml.service.contextPath}/v1/api-docs + documentationUrl: https://www.zowe.org + catalog: + tile: + id: zowe + title: Zowe applications + description: Applications which are part of the Zowe + version: 1.0.0 + ssl: + enabled: true + verifySslCertificatesOfServices: true + protocol: ${server.ssl.protocol} + keyStoreType: ${server.ssl.keyStoreType} + trustStoreType: ${server.ssl.trustStoreType} + + keyAlias: ${server.ssl.keyAlias} + keyPassword: ${server.ssl.keyPassword} + keyStore: ${server.ssl.keyStore} + keyStorePassword: ${server.ssl.keyStorePassword} + trustStore: ${server.ssl.trustStore} + trustStorePassword: ${server.ssl.trustStorePassword} + customMetadata: + apiml: + enableUrlEncodedCharacters: true + gatewayPort: 10010 + gatewayAuthEndpoint: /api/v1/gateway/auth + corsEnabled: false + +server: + port: 10016 + + servlet: + contextPath: / + + ssl: + enabled: true + clientAuth: want + protocol: TLSv1.2 + enabled-protocols: TLSv1.2 + ciphers: TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384,TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256,TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256,TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 + keyStoreType: PKCS12 + trustStoreType: PKCS12 + + keyAlias: localhost + keyPassword: password + keyStore: ../keystore/localhost/localhost.keystore.p12 + keyStorePassword: password + trustStore: ../keystore/localhost/localhost.truststore.p12 + trustStorePassword: password diff --git a/discoverable-client/src/main/java/org/zowe/apiml/client/api/ZaasClientTestController.java b/discoverable-client/src/main/java/org/zowe/apiml/client/api/ZaasClientTestController.java index 184f9d528f..342961be87 100644 --- a/discoverable-client/src/main/java/org/zowe/apiml/client/api/ZaasClientTestController.java +++ b/discoverable-client/src/main/java/org/zowe/apiml/client/api/ZaasClientTestController.java @@ -15,9 +15,11 @@ import lombok.Data; import lombok.NoArgsConstructor; import org.apache.commons.lang3.StringUtils; +import org.springframework.context.annotation.Import; import org.springframework.http.HttpHeaders; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.*; +import org.zowe.apiml.zaasclient.config.DefaultZaasClientConfiguration; import org.zowe.apiml.zaasclient.exception.ZaasClientException; import org.zowe.apiml.zaasclient.exception.ZaasConfigurationException; import org.zowe.apiml.zaasclient.service.ZaasClient; @@ -28,6 +30,7 @@ value = "/api/v1/zaasClient", consumes = "application/json", tags = {"Zaas client test call"}) +@Import(DefaultZaasClientConfiguration.class) public class ZaasClientTestController { private ZaasClient zaasClient; diff --git a/docs/local-configuration.md b/docs/local-configuration.md index a230df6ea5..f169769eeb 100644 --- a/docs/local-configuration.md +++ b/docs/local-configuration.md @@ -41,6 +41,11 @@ java -jar discovery-service/build/libs/discovery-service.jar --spring.config.add java -jar api-catalog-services/build/libs/api-catalog-services.jar --spring.config.additional-location=file:./config/local/api-catalog-service.yml ``` +### Caching API + +```shell +java -jar caching-service/build/libs/caching-service.jar +``` ### Sample Application - Discoverable Client diff --git a/gradle/coverage.gradle b/gradle/coverage.gradle index 38a271cace..c7ece37a15 100644 --- a/gradle/coverage.gradle +++ b/gradle/coverage.gradle @@ -7,6 +7,7 @@ ext.javaProjectsWithUnitTests = [ 'discovery-service', 'apiml-common', 'apiml-utility', + 'caching-service', 'gateway-service', 'onboarding-enabler-java', 'onboarding-enabler-spring', diff --git a/gradle/license.gradle b/gradle/license.gradle index 36c15f45aa..cb317cacd3 100644 --- a/gradle/license.gradle +++ b/gradle/license.gradle @@ -3,6 +3,7 @@ ext.projectsNeedLicense = [ 'api-catalog-ui', 'apiml-common', 'apiml-security-common', + 'caching-service', 'common-service-core', 'discoverable-client', 'discovery-service', diff --git a/gradle/publish.gradle b/gradle/publish.gradle index debd813240..f5d416036d 100644 --- a/gradle/publish.gradle +++ b/gradle/publish.gradle @@ -7,6 +7,7 @@ ext.javaLibraries = [ 'apiml-utility', 'apiml-common', 'apiml-security-common', + 'caching-service', 'discovery-service', 'gateway-service', 'security-service-client-spring', diff --git a/integration-tests/build.gradle b/integration-tests/build.gradle index f17a93ed3e..7088558b90 100644 --- a/integration-tests/build.gradle +++ b/integration-tests/build.gradle @@ -27,6 +27,7 @@ dependencies { compile libraries.jjwt compile(project(':apiml-security-common')) compile(project(':zaas-client')) + compile(project(':caching-service')) testCompile libraries.junit testCompile libraries.hamcrest diff --git a/integration-tests/src/test/java/org/zowe/apiml/cachingservice/CachingApiIntegrationTest.java b/integration-tests/src/test/java/org/zowe/apiml/cachingservice/CachingApiIntegrationTest.java new file mode 100644 index 0000000000..0e8592d723 --- /dev/null +++ b/integration-tests/src/test/java/org/zowe/apiml/cachingservice/CachingApiIntegrationTest.java @@ -0,0 +1,170 @@ +/* + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + */ + +package org.zowe.apiml.cachingservice; +import io.restassured.RestAssured; +import org.hamcrest.Matchers; +import org.junit.jupiter.api.*; +import org.zowe.apiml.caching.model.KeyValue; +import org.zowe.apiml.gatewayservice.SecurityUtils; +import org.zowe.apiml.util.categories.NotForMainframeTest; +import org.zowe.apiml.util.categories.TestsNotMeantForZowe; +import org.zowe.apiml.util.http.HttpRequestUtils; + +import java.net.URI; + +import static io.restassured.RestAssured.given; +import static io.restassured.http.ContentType.JSON; +import static org.apache.http.HttpStatus.*; +import static org.hamcrest.Matchers.isEmptyString; +import static org.hamcrest.core.Is.is; +import static org.hamcrest.core.IsNot.not; + +@TestsNotMeantForZowe +@TestMethodOrder(MethodOrderer.OrderAnnotation.class) +@NotForMainframeTest +class CachingApiIntegrationTest { + + private static final URI CACHING_PATH = HttpRequestUtils.getUriFromGateway("/cachingservice/api/v1/cache"); + private final static String COOKIE_NAME = "apimlAuthenticationToken"; + private static String jwtToken; + + @BeforeAll + static void setup() { + RestAssured.useRelaxedHTTPSValidation(); + jwtToken = generateToken(); + } + + @Test + @Order(1) + void givenValidKeyValue_whenCallingCreateEndpoint_thenStoreIt() { + KeyValue keyValue = new KeyValue("testKey", "testValue"); + + given() + .contentType(JSON) + .body(keyValue) + .cookie(COOKIE_NAME, jwtToken) + .when() + .post(CACHING_PATH) + .then() + .statusCode(is(SC_CREATED)); + } + + @Test + @Order(2) + void givenEmptyBody_whenCallingCreateEndpoint_thenReturn400() { + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .post(CACHING_PATH) + .then() + .statusCode(is(SC_BAD_REQUEST)); + } + + @Test + @Order(3) + void givenValidKeyParameter_whenCallingGetEndpoint_thenReturnKeyValueEntry() { + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .get(CACHING_PATH + "/testKey") + .then() + .body(not(isEmptyString())) + .statusCode(is(SC_OK)); + } + + @Test + @Order(4) + void givenValidKeyParameter_whenCallingGetAllEndpoint_thenAllTheStoredEntries() { + KeyValue keyValue = new KeyValue("testKey2", "testValue2"); + + given() + .contentType(JSON) + .body(keyValue) + .cookie(COOKIE_NAME, jwtToken) + .when() + .post(CACHING_PATH) + .then() + .statusCode(is(SC_CREATED)); + + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .get(CACHING_PATH) + .then() + .body("testKey", Matchers.is(not(isEmptyString())), + "testKey2", Matchers.is(not(isEmptyString()))) + .statusCode(is(SC_OK)); + } + + @Test + @Order(5) + void givenNonExistingKeyParameter_whenCallingGetEndpoint_thenReturnKeyNotFound() { + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .get(CACHING_PATH + "/invalidKey") + .then() + .body(not(isEmptyString())) + .statusCode(is(SC_NOT_FOUND)); + } + + @Test + @Order(6) + void givenValidKeyParameter_whenCallingUpdateEndpoint_thenReturnUpdateValue() { + KeyValue newValue = new KeyValue("testKey", "newValue"); + + given() + .contentType(JSON) + .body(newValue) + .cookie(COOKIE_NAME, jwtToken) + .when() + .put(CACHING_PATH) + .then() + .statusCode(is(SC_NO_CONTENT)); + + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .get(CACHING_PATH + "/testKey") + .then() + .body("value", Matchers.is("newValue")) + .statusCode(is(SC_OK)); + } + + @Test + @Order(7) + void givenValidKeyParameter_whenCallingDeleteEndpoint_thenDeleteKeyValueFromStore() { + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .delete(CACHING_PATH + "/testKey") + .then() + .statusCode(is(SC_NO_CONTENT)); + + given() + .contentType(JSON) + .cookie(COOKIE_NAME, jwtToken) + .when() + .get(CACHING_PATH + "/testKey") + .then() + .statusCode(is(SC_NOT_FOUND)); + } + + private static String generateToken() { + return SecurityUtils.gatewayToken(); + } +} diff --git a/package.json b/package.json index ef603a1b44..b4d3306083 100644 --- a/package.json +++ b/package.json @@ -6,12 +6,13 @@ "doc": "docs" }, "scripts": { - "api-layer": "concurrently --names \"GS,DS,AC,DC,ZO\" -c cyan,yellow,white,blue,green npm:gateway-service npm:discovery-service npm:api-catalog-service npm:discoverable-client npm:mock-zosmf", - "api-layer-ci": "concurrently --names \"GS,DS,AC,DC,ZO\" -c cyan,yellow,white,blue,green npm:gateway-service-ci npm:discovery-service npm:api-catalog-service npm:discoverable-client npm:mock-zosmf", + "api-layer": "concurrently --names \"GS,DS,AC,DC,ZO,CS\" -c cyan,yellow,white,blue,green npm:gateway-service npm:discovery-service npm:api-catalog-service npm:discoverable-client npm:mock-zosmf npm:caching-service", + "api-layer-ci": "concurrently --names \"GS,DS,AC,DC,ZO,CS\" -c cyan,yellow,white,blue,green npm:gateway-service-ci npm:discovery-service npm:api-catalog-service npm:discoverable-client npm:mock-zosmf npm:caching-service", "api-layer-core": "concurrently --names \"GW,DS,AC\" -c cyan,yellow,white npm:gateway-service npm:discovery-service npm:api-catalog-service", "api-layer-without-gateway": "concurrently --names \"DS,AC,DC\" -c yellow,white,blue npm:discovery-service npm:api-catalog-service npm:discoverable-client", "api-layer-without-discovery": "concurrently --names \"GW,AC,DC\" -c cyan,white,blue npm:gateway-service npm:api-catalog-service npm:discoverable-client", "api-layer-without-catalog": "concurrently --names \"GW,DS,DC\" -c cyan,yellow,blue npm:gateway-service npm:discovery-service npm:discoverable-client", + "caching-service": "java -jar caching-service/build/libs/caching-service.jar", "gateway-service": "java -jar gateway-service/build/libs/gateway-service.jar --spring.config.additional-location=file:./config/local/gateway-service.yml --apiml.security.ssl.verifySslCertificatesOfServices=true", "gateway-service-ci": "java -jar gateway-service/build/libs/gateway-service.jar --spring.config.additional-location=file:./config/local/gateway-service.yml --apiml.security.ssl.verifySslCertificatesOfServices=true --spring.profiles.include=diag --apiml.security.x509.enabled=true", "gateway-service-debug": "java -Xdebug -Xrunjdwp:transport=dt_socket,server=y,address=5010,suspend=y -jar gateway-service/build/libs/gateway-service.jar --spring.config.additional-location=file:./config/local/gateway-service.yml", diff --git a/settings.gradle b/settings.gradle index e41ac93977..e81832e644 100644 --- a/settings.gradle +++ b/settings.gradle @@ -19,6 +19,7 @@ include 'discovery-service' include 'apiml-utility' include 'apiml-common' include 'apiml-security-common' +include 'caching-service' include 'gateway-service' include 'common-service-core' include 'discoverable-client' diff --git a/zaas-client/README.md b/zaas-client/README.md index 7d08902546..d2d7735196 100644 --- a/zaas-client/README.md +++ b/zaas-client/README.md @@ -157,6 +157,15 @@ To use this library use the procedure described in this article. } ``` + Alternatively, you can import `DefaultZaasClientCongfiguration` to use the default configuration file structure: + + ```java + @Import(DefaultZaasClientConfiguration.class) + public class SampleZaasClientImplementation { + private ZaasClient zaasClient; + } + ``` + 4. Create an instance of `ZaasClient` in your class and provide the `configProperties` object like the following: ```java diff --git a/zaas-client/build.gradle b/zaas-client/build.gradle index 787f0e45ee..b8be40c878 100644 --- a/zaas-client/build.gradle +++ b/zaas-client/build.gradle @@ -1,6 +1,7 @@ dependencies { annotationProcessor libraries.lombok + compile "org.springframework.boot:spring-boot-starter-web:${springBootVersion}" compile group: 'org.apache.httpcomponents', name: 'httpclient', version: '4.5.11' compile group: 'com.fasterxml.jackson.core', name: 'jackson-databind', version: '2.10.1' compile group: 'org.slf4j', name: 'slf4j-api', version: '1.7.30' diff --git a/discoverable-client/src/main/java/org/zowe/apiml/client/configuration/ZaasClientConfig.java b/zaas-client/src/main/java/org/zowe/apiml/zaasclient/config/DefaultZaasClientConfiguration.java similarity index 91% rename from discoverable-client/src/main/java/org/zowe/apiml/client/configuration/ZaasClientConfig.java rename to zaas-client/src/main/java/org/zowe/apiml/zaasclient/config/DefaultZaasClientConfiguration.java index 89637d8f08..8fa8ec4b47 100644 --- a/discoverable-client/src/main/java/org/zowe/apiml/client/configuration/ZaasClientConfig.java +++ b/zaas-client/src/main/java/org/zowe/apiml/zaasclient/config/DefaultZaasClientConfiguration.java @@ -7,18 +7,15 @@ * * Copyright Contributors to the Zowe Project. */ -package org.zowe.apiml.client.configuration; +package org.zowe.apiml.zaasclient.config; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.zowe.apiml.zaasclient.config.ConfigProperties; import org.zowe.apiml.zaasclient.exception.ZaasConfigurationException; import org.zowe.apiml.zaasclient.service.ZaasClient; import org.zowe.apiml.zaasclient.service.internal.ZaasClientImpl; -@Configuration -public class ZaasClientConfig { +public class DefaultZaasClientConfiguration { @Value("${apiml.service.hostname}") private String host; @@ -47,10 +44,8 @@ public class ZaasClientConfig { @Value("${apiml.service.ssl.trustStoreType}") private String trustStoreType; - @Bean public ConfigProperties getConfigProperties() { - ConfigProperties configProperties = new ConfigProperties(); configProperties.setApimlHost(host); configProperties.setApimlPort(port); diff --git a/zowe-install/src/main/resources/component-scripts/start.sh b/zowe-install/src/main/resources/component-scripts/start.sh index 870f034c78..c1a5566bef 100644 --- a/zowe-install/src/main/resources/component-scripts/start.sh +++ b/zowe-install/src/main/resources/component-scripts/start.sh @@ -139,3 +139,24 @@ _BPX_JOBNAME=${ZOWE_PREFIX}${GATEWAY_CODE} java -Xms32m -Xmx256m -Xquickstart \ -Djava.protocol.handler.pkgs=com.ibm.crypto.provider \ -cp ${ROOT_DIR}"/components/api-mediation/gateway-service.jar":/usr/include/java_classes/IRRRacf.jar \ org.springframework.boot.loader.PropertiesLauncher & + +if [[ ! -z "$ZOWE_CACHING_SERVICE_START" ]] +then + CACHING_CODE=CS + _BPX_JOBNAME=${ZOWE_PREFIX}${CACHING_CODE} java -Xms16m -Xmx512m -Xquickstart \ + -Dibm.serversocket.recover=true \ + -Dfile.encoding=UTF-8 \ + -Djava.io.tmpdir=/tmp \ + -Dspring.profiles.include=$LOG_LEVEL \ + -Dserver.ssl.enabled=true \ + -Dserver.ssl.keyStore=${KEYSTORE} \ + -Dserver.ssl.keyStoreType=${KEYSTORE_TYPE} \ + -Dserver.ssl.keyStorePassword=${KEYSTORE_PASSWORD} \ + -Dserver.ssl.keyAlias=${KEY_ALIAS} \ + -Dserver.ssl.keyPassword=${KEYSTORE_PASSWORD} \ + -Dserver.ssl.trustStore=${TRUSTSTORE} \ + -Dserver.ssl.trustStoreType=${KEYSTORE_TYPE} \ + -Dserver.ssl.trustStorePassword=${KEYSTORE_PASSWORD} \ + -Djava.protocol.handler.pkgs=com.ibm.crypto.provider \ + -jar ${ROOT_DIR}"/components/api-mediation/caching-service.jar" & +fi