001/*- 002 * #%L 003 * HAPI FHIR Storage api 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.jpa.graphql; 021 022import ca.uhn.fhir.context.ConfigurationException; 023import ca.uhn.fhir.context.FhirContext; 024import ca.uhn.fhir.context.FhirVersionEnum; 025import ca.uhn.fhir.context.support.DefaultProfileValidationSupport; 026import ca.uhn.fhir.context.support.IValidationSupport; 027import ca.uhn.fhir.i18n.Msg; 028import ca.uhn.fhir.model.api.annotation.Description; 029import ca.uhn.fhir.rest.annotation.GraphQL; 030import ca.uhn.fhir.rest.annotation.GraphQLQueryBody; 031import ca.uhn.fhir.rest.annotation.GraphQLQueryUrl; 032import ca.uhn.fhir.rest.annotation.IdParam; 033import ca.uhn.fhir.rest.annotation.Initialize; 034import ca.uhn.fhir.rest.api.RequestTypeEnum; 035import ca.uhn.fhir.rest.api.server.RequestDetails; 036import ca.uhn.fhir.rest.server.RestfulServer; 037import ca.uhn.fhir.rest.server.exceptions.BaseServerResponseException; 038import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 039import ca.uhn.fhir.rest.server.exceptions.UnclassifiedServerFailureException; 040import ca.uhn.fhir.rest.server.servlet.ServletRequestDetails; 041import jakarta.annotation.Nonnull; 042import jakarta.annotation.Nullable; 043import org.apache.commons.lang3.ObjectUtils; 044import org.apache.commons.lang3.Validate; 045import org.hl7.fhir.instance.model.api.IBaseResource; 046import org.hl7.fhir.instance.model.api.IIdType; 047import org.hl7.fhir.utilities.graphql.IGraphQLEngine; 048import org.hl7.fhir.utilities.graphql.IGraphQLStorageServices; 049import org.hl7.fhir.utilities.graphql.ObjectValue; 050import org.hl7.fhir.utilities.graphql.Package; 051import org.hl7.fhir.utilities.graphql.Parser; 052import org.slf4j.Logger; 053import org.slf4j.LoggerFactory; 054 055import java.util.function.Supplier; 056 057public class GraphQLProvider { 058 private static final Logger ourLog = LoggerFactory.getLogger(GraphQLProvider.class); 059 060 private final Supplier<IGraphQLEngine> myEngineFactory; 061 private final IGraphQLStorageServices myStorageServices; 062 063 /** 064 * Constructor which uses a default context and validation support object 065 * 066 * @param theStorageServices The storage services (this object will be used to retrieve various resources as required by the GraphQL engine) 067 */ 068 public GraphQLProvider(IGraphQLStorageServices theStorageServices) { 069 this(FhirContext.forR4(), null, theStorageServices); 070 } 071 072 /** 073 * Constructor which uses the given worker context 074 * 075 * @param theFhirContext The HAPI FHIR Context object 076 * @param theValidationSupport The HAPI Validation Support object, or null 077 * @param theStorageServices The storage services (this object will be used to retrieve various resources as required by the GraphQL engine) 078 */ 079 public GraphQLProvider( 080 @Nonnull FhirContext theFhirContext, 081 @Nullable IValidationSupport theValidationSupport, 082 @Nonnull IGraphQLStorageServices theStorageServices) { 083 Validate.notNull(theFhirContext, "theFhirContext must not be null"); 084 Validate.notNull(theStorageServices, "theStorageServices must not be null"); 085 086 switch (theFhirContext.getVersion().getVersion()) { 087 case DSTU3: { 088 IValidationSupport validationSupport = theValidationSupport; 089 validationSupport = ObjectUtils.defaultIfNull( 090 validationSupport, new DefaultProfileValidationSupport(theFhirContext)); 091 org.hl7.fhir.dstu3.hapi.ctx.HapiWorkerContext workerContext = 092 new org.hl7.fhir.dstu3.hapi.ctx.HapiWorkerContext(theFhirContext, validationSupport); 093 myEngineFactory = () -> new org.hl7.fhir.dstu3.utils.GraphQLEngine(workerContext); 094 break; 095 } 096 case R4: { 097 IValidationSupport validationSupport = theValidationSupport; 098 validationSupport = ObjectUtils.defaultIfNull( 099 validationSupport, new DefaultProfileValidationSupport(theFhirContext)); 100 org.hl7.fhir.r4.hapi.ctx.HapiWorkerContext workerContext = 101 new org.hl7.fhir.r4.hapi.ctx.HapiWorkerContext(theFhirContext, validationSupport); 102 myEngineFactory = () -> new org.hl7.fhir.r4.utils.GraphQLEngine(workerContext); 103 break; 104 } 105 case R4B: { 106 IValidationSupport validationSupport = theValidationSupport; 107 validationSupport = ObjectUtils.defaultIfNull( 108 validationSupport, new DefaultProfileValidationSupport(theFhirContext)); 109 org.hl7.fhir.r4b.hapi.ctx.HapiWorkerContext workerContext = 110 new org.hl7.fhir.r4b.hapi.ctx.HapiWorkerContext(theFhirContext, validationSupport); 111 myEngineFactory = () -> new org.hl7.fhir.r4b.utils.GraphQLEngine(workerContext); 112 break; 113 } 114 case R5: { 115 IValidationSupport validationSupport = theValidationSupport; 116 validationSupport = ObjectUtils.defaultIfNull( 117 validationSupport, new DefaultProfileValidationSupport(theFhirContext)); 118 org.hl7.fhir.r5.hapi.ctx.HapiWorkerContext workerContext = 119 new org.hl7.fhir.r5.hapi.ctx.HapiWorkerContext(theFhirContext, validationSupport); 120 myEngineFactory = () -> new org.hl7.fhir.r5.utils.GraphQLEngine(workerContext); 121 break; 122 } 123 case DSTU2: 124 case DSTU2_HL7ORG: 125 case DSTU2_1: 126 default: { 127 throw new UnsupportedOperationException(Msg.code(1143) + "GraphQL not supported for version: " 128 + theFhirContext.getVersion().getVersion()); 129 } 130 } 131 132 myStorageServices = theStorageServices; 133 } 134 135 @Description( 136 value = 137 "This operation invokes a GraphQL expression for fetching an joining a graph of resources, returning them in a custom format.") 138 @GraphQL(type = RequestTypeEnum.GET) 139 public String processGraphQlGetRequest( 140 ServletRequestDetails theRequestDetails, @IdParam IIdType theId, @GraphQLQueryUrl String theQueryUrl) { 141 if (theQueryUrl != null) { 142 return processGraphQLRequest(theRequestDetails, theId, theQueryUrl); 143 } 144 throw new InvalidRequestException(Msg.code(1144) + "Unable to parse empty GraphQL expression"); 145 } 146 147 @Description( 148 value = 149 "This operation invokes a GraphQL expression for fetching an joining a graph of resources, returning them in a custom format.") 150 @GraphQL(type = RequestTypeEnum.POST) 151 public String processGraphQlPostRequest( 152 ServletRequestDetails theServletRequestDetails, 153 RequestDetails theRequestDetails, 154 @IdParam IIdType theId, 155 @GraphQLQueryBody String theQueryBody) { 156 if (theQueryBody != null) { 157 return processGraphQLRequest(theServletRequestDetails, theId, theQueryBody); 158 } 159 throw new InvalidRequestException(Msg.code(1145) + "Unable to parse empty GraphQL expression"); 160 } 161 162 public String processGraphQLRequest(ServletRequestDetails theRequestDetails, IIdType theId, String theQuery) { 163 Package parsedGraphQLRequest; 164 try { 165 parsedGraphQLRequest = Parser.parse(theQuery); 166 } catch (Exception e) { 167 throw new InvalidRequestException(Msg.code(1146) + "Unable to parse GraphQL Expression: " + e); 168 } 169 170 return processGraphQLRequest(theRequestDetails, theId, parsedGraphQLRequest); 171 } 172 173 protected String processGraphQLRequest( 174 ServletRequestDetails theRequestDetails, IIdType theId, Package parsedGraphQLRequest) { 175 IGraphQLEngine engine = myEngineFactory.get(); 176 engine.setAppInfo(theRequestDetails); 177 engine.setServices(myStorageServices); 178 engine.setGraphQL(parsedGraphQLRequest); 179 180 try { 181 182 if (theId != null) { 183 IBaseResource focus = 184 myStorageServices.lookup(theRequestDetails, theId.getResourceType(), theId.getIdPart()); 185 engine.setFocus(focus); 186 } 187 engine.execute(); 188 189 StringBuilder outputBuilder = new StringBuilder(); 190 ObjectValue output = engine.getOutput(); 191 output.write(outputBuilder, 0, "\n"); 192 193 return outputBuilder.toString(); 194 195 } catch (Exception e) { 196 StringBuilder b = new StringBuilder(); 197 b.append("Unable to execute GraphQL Expression: "); 198 int statusCode = 500; 199 if (e instanceof BaseServerResponseException) { 200 b.append("HTTP "); 201 statusCode = ((BaseServerResponseException) e).getStatusCode(); 202 b.append(statusCode); 203 b.append(" "); 204 } else { 205 // This means it's a bug, so let's log 206 ourLog.error("Failure during GraphQL processing", e); 207 } 208 b.append(e.getMessage()); 209 throw new UnclassifiedServerFailureException(statusCode, Msg.code(1147) + b); 210 } 211 } 212 213 @Initialize 214 public void initialize(RestfulServer theServer) { 215 ourLog.trace("Initializing GraphQL provider"); 216 if (!theServer.getFhirContext().getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.DSTU3)) { 217 throw new ConfigurationException(Msg.code(1148) + "Can not use " 218 + getClass().getName() + " provider on server with FHIR " 219 + theServer.getFhirContext().getVersion().getVersion().name() + " context"); 220 } 221 } 222}