001/*- 002 * #%L 003 * HAPI FHIR - Server Framework 004 * %% 005 * Copyright (C) 2014 - 2025 Smile CDR, Inc. 006 * %% 007 * Licensed under the Apache License, Version 2.0 (the "License"); 008 * you may not use this file except in compliance with the License. 009 * You may obtain a copy of the License at 010 * 011 * http://www.apache.org/licenses/LICENSE-2.0 012 * 013 * Unless required by applicable law or agreed to in writing, software 014 * distributed under the License is distributed on an "AS IS" BASIS, 015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 * #L% 019 */ 020package ca.uhn.fhir.rest.server.tenant; 021 022import ca.uhn.fhir.i18n.HapiLocalizer; 023import ca.uhn.fhir.i18n.Msg; 024import ca.uhn.fhir.rest.api.server.RequestDetails; 025import ca.uhn.fhir.rest.api.server.SystemRequestDetails; 026import ca.uhn.fhir.rest.server.RestfulServer; 027import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 028import ca.uhn.fhir.util.UrlPathTokenizer; 029import org.slf4j.Logger; 030import org.slf4j.LoggerFactory; 031 032import static org.apache.commons.lang3.StringUtils.defaultIfBlank; 033 034/** 035 * This class is a tenant identification strategy which assumes that a single path 036 * element will be present between the server base URL and individual request. 037 * <p> 038 * For example, 039 * with this strategy enabled, given the following URL on a server with base URL <code>http://example.com/base</code>, 040 * the server will extract the <code>TENANT-A</code> portion of the URL and use it as the tenant identifier. The 041 * request will then proceed to read the resource with ID <code>Patient/123</code>. 042 * </p> 043 * <p> 044 * GET http://example.com/base/TENANT-A/Patient/123 045 * </p> 046 */ 047public class UrlBaseTenantIdentificationStrategy implements ITenantIdentificationStrategy { 048 049 private static final Logger ourLog = LoggerFactory.getLogger(UrlBaseTenantIdentificationStrategy.class); 050 051 @Override 052 public void extractTenant(UrlPathTokenizer theUrlPathTokenizer, RequestDetails theRequestDetails) { 053 String tenantId = null; 054 boolean isSystemRequest = (theRequestDetails instanceof SystemRequestDetails); 055 056 // If we were given no partition for a system request, use DEFAULT: 057 if (!theUrlPathTokenizer.hasMoreTokens()) { 058 if (isSystemRequest) { 059 tenantId = "DEFAULT"; 060 theRequestDetails.setTenantId(tenantId); 061 ourLog.trace("No tenant ID found for system request; using DEFAULT."); 062 } 063 } 064 065 // We were given at least one URL token: 066 else { 067 068 // peek() won't consume this token: 069 tenantId = defaultIfBlank(theUrlPathTokenizer.peek(), null); 070 071 // If it's "metadata" or starts with "$", use DEFAULT partition and don't consume this token: 072 if (tenantId != null && (tenantId.equals("metadata") || isOperation(tenantId))) { 073 tenantId = "DEFAULT"; 074 theRequestDetails.setTenantId(tenantId); 075 ourLog.trace("No tenant ID found for metadata or system request; using DEFAULT."); 076 } 077 078 // It isn't metadata or $, so assume that this first token is the partition name and consume it: 079 else { 080 tenantId = defaultIfBlank(theUrlPathTokenizer.nextTokenUnescapedAndSanitized(), null); 081 if (tenantId != null) { 082 theRequestDetails.setTenantId(tenantId); 083 ourLog.trace("Found tenant ID {} in request string", tenantId); 084 } 085 } 086 } 087 088 // If we get to this point without a tenant, it's an invalid request: 089 if (tenantId == null) { 090 HapiLocalizer localizer = 091 theRequestDetails.getServer().getFhirContext().getLocalizer(); 092 throw new InvalidRequestException( 093 Msg.code(307) + localizer.getMessage(RestfulServer.class, "rootRequest.multitenant")); 094 } 095 } 096 097 private boolean isOperation(String theToken) { 098 return theToken.startsWith("$"); 099 } 100 101 @Override 102 public String massageServerBaseUrl(String theFhirServerBase, RequestDetails theRequestDetails) { 103 String result = theFhirServerBase; 104 if (theRequestDetails.getTenantId() != null) { 105 result += "/" + theRequestDetails.getTenantId(); 106 } 107 return result; 108 } 109 110 @Override 111 public String resolveRelativeUrl(String theRelativeUrl, RequestDetails theRequestDetails) { 112 UrlPathTokenizer tokenizer = new UrlPathTokenizer(theRelativeUrl); 113 // there is no more tokens in the URL - skip url resolution 114 if (!tokenizer.hasMoreTokens() || tokenizer.peek() == null) { 115 return theRelativeUrl; 116 } 117 String nextToken = tokenizer.peek(); 118 // there is no tenant ID in parent request details or tenant ID is already present in URL - skip url resolution 119 if (theRequestDetails.getTenantId() == null || nextToken.equals(theRequestDetails.getTenantId())) { 120 return theRelativeUrl; 121 } 122 123 // token is Resource type or operation - adding tenant ID from parent request details 124 if (isResourceType(nextToken, theRequestDetails) || isOperation(nextToken)) { 125 return theRequestDetails.getTenantId() + "/" + theRelativeUrl; 126 } else { 127 return theRelativeUrl; 128 } 129 } 130 131 private boolean isResourceType(String token, RequestDetails theRequestDetails) { 132 return theRequestDetails.getFhirContext().getResourceTypes().stream().anyMatch(type -> type.equals(token)); 133 } 134}