org.ehrbase.rest.BaseController Maven / Gradle / Ivy
/*
* Copyright (c) 2024 vitasystems GmbH.
*
* This file is part of project EHRbase
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.ehrbase.rest;
import com.nedap.archie.rm.datavalues.quantity.datetime.DvDateTime;
import java.net.URI;
import java.time.LocalDateTime;
import java.time.OffsetDateTime;
import java.time.ZoneOffset;
import java.time.format.DateTimeParseException;
import java.util.*;
import java.util.stream.Stream;
import org.apache.commons.lang3.StringUtils;
import org.ehrbase.api.exception.InvalidApiParameterException;
import org.ehrbase.api.exception.NotAcceptableException;
import org.ehrbase.api.exception.ObjectNotFoundException;
import org.ehrbase.openehr.sdk.response.dto.ehrscape.CompositionFormat;
import org.ehrbase.rest.openehr.format.CompositionRepresentation;
import org.ehrbase.rest.openehr.format.OpenEHRMediaType;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.http.HttpHeaders;
import org.springframework.http.MediaType;
import org.springframework.util.MimeTypeUtils;
import org.springframework.web.servlet.support.ServletUriComponentsBuilder;
import org.springframework.web.util.UriComponentsBuilder;
import org.springframework.web.util.UriUtils;
/**
* This base controller implements the basic functionality for all specific controllers. This
* includes error handling and utils.
*/
public abstract class BaseController {
// HTTP Headers
public static final String OPENEHR_AUDIT_DETAILS = "openEHR-AUDIT_DETAILS";
public static final String OPENEHR_VERSION = "openEHR-VERSION";
public static final String PREFER = "Prefer";
public static final String RETURN_MINIMAL = "return=minimal";
public static final String RETURN_REPRESENTATION = "return=representation";
// Fixed header identifiers
public static final String CONTENT_TYPE = HttpHeaders.CONTENT_TYPE;
public static final String ACCEPT = HttpHeaders.ACCEPT;
public static final String REQ_CONTENT_TYPE = "Client may request content format";
public static final String REQ_ACCEPT = "Client should specify expected format";
// response headers
public static final String RESP_CONTENT_TYPE_DESC = "Format of response";
public static final String LOCATION = HttpHeaders.LOCATION;
public static final String ETAG = HttpHeaders.ETAG;
public static final String LAST_MODIFIED = HttpHeaders.LAST_MODIFIED;
public static final String IF_MATCH = HttpHeaders.IF_MATCH;
// constants of all API resources
public static final String EHR = "ehr";
public static final String EHR_STATUS = "ehr_status";
public static final String VERSIONED_EHR_STATUS = "versioned_ehr_status";
public static final String VERSIONED_COMPOSITION = "versioned_composition";
public static final String COMPOSITION = "composition";
public static final String DIRECTORY = "directory";
public static final String CONTRIBUTION = "contribution";
public static final String QUERY = "query";
public static final String DEFINITION = "definition";
public static final String TEMPLATE = "template";
public static final String API_CONTEXT_PATH = "${openehr-api.context-path:/rest/openehr}";
public static final String API_CONTEXT_PATH_WITH_VERSION = API_CONTEXT_PATH + "/v1";
public static final String ADMIN_API_CONTEXT_PATH = "${admin-api.context-path:/rest/admin}";
public String getContextPath() {
return ServletUriComponentsBuilder.fromCurrentContextPath().build().toUriString();
}
@Value(API_CONTEXT_PATH_WITH_VERSION)
protected String apiContextPathWithVersion;
/**
* Returns a URI for list of segments.
* The segments are appended to the base path and encoded to ensure safe usage in a URI.
*
* @param pathSegments List of segments to append to the base URL
* @return URI for the given base URL and segments
*/
protected URI createLocationUri(String... pathSegments) {
return UriComponentsBuilder.fromHttpUrl(getContextPath())
.path(UriUtils.encodePath(apiContextPathWithVersion, "UTF-8"))
.pathSegment(pathSegments)
.build()
.toUri();
}
/**
* Helper to parse an input UUID int string format.
*
* @param uuidString to parse
* @param error to raise in case the given UUID string is invalid
* @return uuid parse from the input uuidString
* @throws InvalidApiParameterException when the given uuidString
is invalid
*/
protected UUID parseUUID(String uuidString, String error) {
try {
return UUID.fromString(uuidString);
} catch (IllegalArgumentException e) {
throw new InvalidApiParameterException(error);
}
}
/**
* Helper to allow string UUID input from controllers, which throws an ObjectNotFound exception
* when no UUID representation can be created. This case is equal to no matching object.
*
* @param ehrIdString Input String representation of the ehrId
* @return UUID representation of the ehrId
* @throws ObjectNotFoundException when no UUID can't be created from input
*/
protected UUID getEhrUuid(String ehrIdString) {
return extractUUIDFromStringWithError(
ehrIdString, "ehr", "EHR not found, in fact, only UUID-type IDs are supported");
}
/**
* Helper to allow string UUID input from controllers, which throws an ObjectNotFound exception
* when no UUID representation can be created. This case is equal to no matching object.
*
* @param compositionVersionedObjectUidString Input String representation
* @return UUID representation
* @throws ObjectNotFoundException when no UUID can't be created from input
*/
protected UUID getCompositionVersionedObjectUidString(String compositionVersionedObjectUidString) {
return extractUUIDFromStringWithError(
compositionVersionedObjectUidString,
COMPOSITION,
"Composition not found, in fact, only UUID-type versionedObjectUids are supported");
}
/**
* Helper to allow string UUID input from controllers, which throws an ObjectNotFound exception
* when no UUID representation can be created. This case is equal to no matching object.
*
* @param compositionVersionedObjectUidString Input String representation
* @return UUID representation
* @throws ObjectNotFoundException when no UUID can't be created from input
*/
protected UUID getContributionVersionedObjectUidString(String compositionVersionedObjectUidString) {
return extractUUIDFromStringWithError(
compositionVersionedObjectUidString,
CONTRIBUTION,
"Contribution not found, in fact, only UUID-type versionedObjectUids are supported");
}
// Internal abstraction layer helper, so calling methods above can invoke with meaningful error
// messages depending on context.
private UUID extractUUIDFromStringWithError(String uuidString, String type, String error) {
UUID uuid;
try {
uuid = UUID.fromString(uuidString);
} catch (IllegalArgumentException e) {
throw new ObjectNotFoundException(type, error);
}
return uuid;
}
/**
* Extracts the {@link CompositionRepresentation} from the REST request's input {@link MediaType} style
* content type header string and validates them against the given format as {@link CompositionFormat}.
*
* @param contentType String representation of REST request's input {@link MediaType} style content type header
* @param format String representation of the REST request format
parameter that is
* interpreted as an {@link CompositionFormat}
* @return {@link CompositionFormat} expressing the content type
* @throws NotAcceptableException when content type or composition format is not supported or the input is invalid
*/
protected CompositionRepresentation extractCompositionRepresentation(String contentType, String format) {
final CompositionRepresentation representation;
try {
final Optional parsedFormat =
Optional.ofNullable(format).filter(s -> !s.isEmpty()).map(CompositionFormat::valueOf);
final MediaType mediaType = resolveContentType(
contentType,
MediaType.APPLICATION_JSON,
MediaType.APPLICATION_XML,
OpenEHRMediaType.APPLICATION_WT_FLAT_SCHEMA_JSON,
OpenEHRMediaType.APPLICATION_WT_STRUCTURED_SCHEMA_JSON);
representation =
CompositionRepresentation.selectFromMediaTypeWithFormat(mediaType, parsedFormat.orElse(null));
} catch (IllegalArgumentException e) {
throw new NotAcceptableException(
"Invalid compositions format [%s] only [XML, JSON, FLAT, STRUCTURED] are supported at the moment"
.formatted(format));
}
return representation;
}
/**
* Extracts the UUID base from a versioned UID. Or, if
*
* @param versionUid raw versionUid in format [UUID]::[VERSION]
* @return uuid [UUID]
part
*/
protected UUID extractVersionedObjectUidFromVersionUid(String versionUid) {
if (!versionUid.contains("::")) {
return UUID.fromString(versionUid);
}
return UUID.fromString(versionUid.substring(0, versionUid.indexOf("::")));
}
protected Optional extractVersionFromVersionUid(String versionUid) {
// extract the version from string of format "$UUID::$SYSTEM::$VERSION"
// via making a substring starting at last occurrence of "::" + 2
int lastOccurrence = versionUid.lastIndexOf("::");
if (lastOccurrence > 0 && versionUid.indexOf("::") != lastOccurrence) {
int version = Integer.parseInt(versionUid.substring(lastOccurrence + 2));
if (version < 1) {
throw new InvalidApiParameterException("Version can't be zero or negative.");
}
return Optional.of(version);
}
return Optional.empty();
}
/**
* Resolves the Content-Type based on Accept header. Validates if the given acceptHeader
in
* either application/json
or application/xml
.
* In case acceptHeader
is given application/json
will be selected as a default.
*
* @param acceptHeader Accept header value
* @return Content-Type of the response
*/
protected MediaType resolveContentType(String acceptHeader) {
return resolveContentType(acceptHeader, MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML);
}
/**
* Resolves the Content-Type based on Accept header using the supported predicate.
*
* @param acceptHeader Accept header value
* @param defaultMediaType Default Content-Type
* @param supportedMediaTypes supported Content-Types
* @return Content-Type of the response
*/
protected MediaType resolveContentType(
String acceptHeader, MediaType defaultMediaType, MediaType... supportedMediaTypes) {
List mediaTypes = MediaType.parseMediaTypes(acceptHeader);
if (mediaTypes.isEmpty()) {
return defaultMediaType;
}
MimeTypeUtils.sortBySpecificity(mediaTypes);
MediaType contentType = mediaTypes.stream()
.filter(type -> Stream.concat(Stream.of(defaultMediaType), Arrays.stream(supportedMediaTypes))
.anyMatch(type::isCompatibleWith))
.findFirst()
.orElseThrow(() -> new InvalidApiParameterException("Wrong Content-Type header in request"));
if (contentType.equals(MediaType.ALL)) {
return defaultMediaType;
}
return contentType;
}
protected static Optional decodeVersionAtTime(String versionAtTimeParam) {
return Optional.ofNullable(versionAtTimeParam)
.filter(StringUtils::isNotBlank)
// revert application/x-www-form-urlencoded
.map(s -> s.replace(' ', '+'))
.map(s -> {
try {
DvDateTime dvDateTime = new DvDateTime(s);
if (dvDateTime.getValue() instanceof OffsetDateTime offsetDateTime) {
return offsetDateTime;
} else if (dvDateTime.getValue() instanceof LocalDateTime localDateTime) {
return localDateTime.atOffset(ZoneOffset.UTC);
} else {
throw new IllegalArgumentException(
"Value '%s' is not valid for version_at_time parameter. Value must be in the extended ISO 8601 format."
.formatted(versionAtTimeParam));
}
} catch (DateTimeParseException e) {
throw new IllegalArgumentException(
"Value '%s' is not valid for version_at_time parameter. Value must be in the extended ISO 8601 format."
.formatted(versionAtTimeParam),
e);
}
});
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy