001/*- 002 * #%L 003 * HAPI FHIR - Master Data Management 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.mdm.api; 021 022import ca.uhn.fhir.jpa.api.dao.DaoRegistry; 023import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao; 024import ca.uhn.fhir.jpa.searchparam.SearchParameterMap; 025import ca.uhn.fhir.mdm.api.params.GenerateMdmMetricsParameters; 026import ca.uhn.fhir.mdm.model.MdmResourceMetrics; 027import ca.uhn.fhir.mdm.util.MdmSearchParamBuildingUtils; 028import ca.uhn.fhir.rest.api.SearchTotalModeEnum; 029import ca.uhn.fhir.rest.api.server.IBundleProvider; 030import ca.uhn.fhir.rest.api.server.SystemRequestDetails; 031 032public abstract class BaseMdmMetricSvc implements IMdmMetricSvc { 033 034 /** 035 * Count of numbered buckets. 036 * There will also be a NULL bucket, so there will be a total 037 * of BUCKETS + 1 buckets. 038 */ 039 public static final int BUCKETS = 100; 040 041 /** 042 * The NULL label 043 */ 044 public static final String NULL_VALUE = "NULL"; 045 046 /** 047 * The label for the first bucket 048 */ 049 public static final String FIRST_BUCKET = "x_<_%.2f"; 050 051 /** 052 * The label for the nth bucket (2... buckets) 053 */ 054 public static final String NTH_BUCKET = "%.2f_<_x_<=_%.2f"; 055 056 protected final DaoRegistry myDaoRegistry; 057 058 public BaseMdmMetricSvc(DaoRegistry theDaoRegistry) { 059 myDaoRegistry = theDaoRegistry; 060 } 061 062 protected double getBucket(int theBucketId) { 063 return (double) Math.round((float) (100 * theBucketId) / BUCKETS) / 100; 064 } 065 066 protected MdmResourceMetrics generateResourceMetrics(GenerateMdmMetricsParameters theParameters) { 067 String resourceType = theParameters.getResourceType(); 068 @SuppressWarnings("rawtypes") 069 IFhirResourceDao dao = myDaoRegistry.getResourceDao(resourceType); 070 071 // TODO 072 /* 073 * We are using 3 different queries to count: 074 * * all resources 075 * * all golden resources 076 * * all blocked resources. 077 * 078 * This is inefficient and if we want, we can speed it up with 079 * a custom query in the future. 080 */ 081 IBundleProvider outcome = null; 082 SearchParameterMap map = null; 083 084 MdmResourceMetrics metrics = new MdmResourceMetrics(); 085 metrics.setResourceType(resourceType); 086 087 // find golden resources 088 map = MdmSearchParamBuildingUtils.buildBasicGoldenResourceSearchParameterMap(resourceType); 089 setCountOnly(map); 090 outcome = dao.search(map, new SystemRequestDetails()); 091 metrics.setGoldenResourcesCount(outcome.size()); 092 093 // find blocked resources 094 map = MdmSearchParamBuildingUtils.buildSearchParameterForBlockedResourceCount(resourceType); 095 setCountOnly(map); 096 outcome = dao.search(map, new SystemRequestDetails()); 097 metrics.setExcludedResources(outcome.size()); 098 099 // find all resources 100 map = new SearchParameterMap(); 101 setCountOnly(map); 102 outcome = dao.search(map, new SystemRequestDetails()); 103 metrics.setSourceResourcesCount(outcome.size() - metrics.getGoldenResourcesCount()); 104 105 return metrics; 106 } 107 108 private void setCountOnly(SearchParameterMap theMap) { 109 theMap.setCount(0); 110 theMap.setLoadSynchronous(true); 111 theMap.setSearchTotalMode(SearchTotalModeEnum.ACCURATE); 112 } 113}