check in new resource manager

This commit is contained in:
Grahame Grieve 2019-11-12 19:40:42 +11:00
parent 5fe2552a08
commit 4b173cbfb3
6 changed files with 593 additions and 139 deletions

View File

@ -23,6 +23,8 @@ package org.hl7.fhir.r5.context;
import java.io.FileNotFoundException; import java.io.FileNotFoundException;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date; import java.util.Date;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
@ -36,6 +38,7 @@ import org.hl7.fhir.exceptions.DefinitionException;
import org.hl7.fhir.exceptions.FHIRException; import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.exceptions.TerminologyServiceException; import org.hl7.fhir.exceptions.TerminologyServiceException;
import org.hl7.fhir.r5.conformance.ProfileUtilities; import org.hl7.fhir.r5.conformance.ProfileUtilities;
import org.hl7.fhir.r5.context.BaseWorkerContext.MetadataResourceVersionComparator;
import org.hl7.fhir.r5.context.IWorkerContext.ILoggingService.LogCategory; import org.hl7.fhir.r5.context.IWorkerContext.ILoggingService.LogCategory;
import org.hl7.fhir.r5.context.TerminologyCache.CacheToken; import org.hl7.fhir.r5.context.TerminologyCache.CacheToken;
import org.hl7.fhir.r5.model.BooleanType; import org.hl7.fhir.r5.model.BooleanType;
@ -79,6 +82,7 @@ import org.hl7.fhir.utilities.OIDUtils;
import org.hl7.fhir.utilities.TerminologyServiceOptions; import org.hl7.fhir.utilities.TerminologyServiceOptions;
import org.hl7.fhir.utilities.TranslationServices; import org.hl7.fhir.utilities.TranslationServices;
import org.hl7.fhir.utilities.Utilities; import org.hl7.fhir.utilities.Utilities;
import org.hl7.fhir.utilities.VersionUtilities;
import org.hl7.fhir.utilities.validation.ValidationMessage.IssueSeverity; import org.hl7.fhir.utilities.validation.ValidationMessage.IssueSeverity;
import org.hl7.fhir.utilities.validation.ValidationMessage.IssueType; import org.hl7.fhir.utilities.validation.ValidationMessage.IssueType;
@ -86,22 +90,52 @@ import com.google.gson.JsonObject;
public abstract class BaseWorkerContext implements IWorkerContext { public abstract class BaseWorkerContext implements IWorkerContext {
public class MetadataResourceVersionComparator<T extends MetadataResource> implements Comparator<T> {
private List<T> list;
public MetadataResourceVersionComparator(List<T> list) {
this.list = list;
}
@Override
public int compare(T arg1, T arg2) {
String v1 = arg1.getVersion();
String v2 = arg2.getVersion();
if (v1 == null && v2 == null) {
return Integer.compare(list.indexOf(arg1), list.indexOf(arg2)); // retain original order
} else if (v1 == null) {
return -1;
} else if (v2 == null) {
return 1;
} else {
String mm1 = VersionUtilities.getMajMin(v1);
String mm2 = VersionUtilities.getMajMin(v2);
if (mm1 == null || mm2 == null) {
return v1.compareTo(v2);
} else {
return mm1.compareTo(mm2);
}
}
}
}
private Object lock = new Object(); // used as a lock for the data that follows private Object lock = new Object(); // used as a lock for the data that follows
private Map<String, Map<String, Resource>> allResourcesById = new HashMap<String, Map<String, Resource>>(); private Map<String, Map<String, Resource>> allResourcesById = new HashMap<String, Map<String, Resource>>();
// all maps are to the full URI // all maps are to the full URI
private Map<String, CodeSystem> codeSystems = new HashMap<String, CodeSystem>(); private MetadataResourceManager<CodeSystem> codeSystems = new MetadataResourceManager<CodeSystem>();
private Set<String> supportedCodeSystems = new HashSet<String>(); private Set<String> supportedCodeSystems = new HashSet<String>();
private Map<String, ValueSet> valueSets = new HashMap<String, ValueSet>(); private MetadataResourceManager<ValueSet> valueSets = new MetadataResourceManager<ValueSet>();
private Map<String, ConceptMap> maps = new HashMap<String, ConceptMap>(); private MetadataResourceManager<ConceptMap> maps = new MetadataResourceManager<ConceptMap>();
protected Map<String, StructureMap> transforms = new HashMap<String, StructureMap>(); protected MetadataResourceManager<StructureMap> transforms = new MetadataResourceManager<StructureMap>();
private Map<String, StructureDefinition> structures = new HashMap<String, StructureDefinition>(); private MetadataResourceManager<StructureDefinition> structures = new MetadataResourceManager<StructureDefinition>();
private Map<String, ImplementationGuide> guides = new HashMap<String, ImplementationGuide>(); private MetadataResourceManager<ImplementationGuide> guides = new MetadataResourceManager<ImplementationGuide>();
private Map<String, CapabilityStatement> capstmts = new HashMap<String, CapabilityStatement>(); private MetadataResourceManager<CapabilityStatement> capstmts = new MetadataResourceManager<CapabilityStatement>();
private Map<String, SearchParameter> searchParameters = new HashMap<String, SearchParameter>(); private MetadataResourceManager<SearchParameter> searchParameters = new MetadataResourceManager<SearchParameter>();
private Map<String, Questionnaire> questionnaires = new HashMap<String, Questionnaire>(); private MetadataResourceManager<Questionnaire> questionnaires = new MetadataResourceManager<Questionnaire>();
private Map<String, OperationDefinition> operations = new HashMap<String, OperationDefinition>(); private MetadataResourceManager<OperationDefinition> operations = new MetadataResourceManager<OperationDefinition>();
private Map<String, PlanDefinition> plans = new HashMap<String, PlanDefinition>(); private MetadataResourceManager<PlanDefinition> plans = new MetadataResourceManager<PlanDefinition>();
private List<NamingSystem> systems = new ArrayList<NamingSystem>(); private List<NamingSystem> systems = new ArrayList<NamingSystem>();
private UcumService ucumService; private UcumService ucumService;
@ -128,7 +162,8 @@ public abstract class BaseWorkerContext implements IWorkerContext {
txCache = new TerminologyCache(lock, null); txCache = new TerminologyCache(lock, null);
} }
public BaseWorkerContext(Map<String, CodeSystem> codeSystems, Map<String, ValueSet> valueSets, Map<String, ConceptMap> maps, Map<String, StructureDefinition> profiles, Map<String, ImplementationGuide> guides) throws FileNotFoundException, IOException, FHIRException { public BaseWorkerContext(MetadataResourceManager<CodeSystem> codeSystems, MetadataResourceManager<ValueSet> valueSets, MetadataResourceManager<ConceptMap> maps, MetadataResourceManager<StructureDefinition> profiles,
MetadataResourceManager<ImplementationGuide> guides) throws FileNotFoundException, IOException, FHIRException {
super(); super();
this.codeSystems = codeSystems; this.codeSystems = codeSystems;
this.valueSets = valueSets; this.valueSets = valueSets;
@ -142,19 +177,19 @@ public abstract class BaseWorkerContext implements IWorkerContext {
synchronized (other.lock) { // tricky, because you need to lock this as well, but it's really not in use yet synchronized (other.lock) { // tricky, because you need to lock this as well, but it's really not in use yet
allResourcesById.putAll(other.allResourcesById); allResourcesById.putAll(other.allResourcesById);
translator = other.translator; translator = other.translator;
codeSystems.putAll(other.codeSystems); codeSystems.copy(other.codeSystems);
txcaps = other.txcaps; txcaps = other.txcaps;
valueSets.putAll(other.valueSets); valueSets.copy(other.valueSets);
maps.putAll(other.maps); maps.copy(other.maps);
transforms.putAll(other.transforms); transforms.copy(other.transforms);
structures.putAll(other.structures); structures.copy(other.structures);
searchParameters.putAll(other.searchParameters); searchParameters.copy(other.searchParameters);
plans.putAll(other.plans); plans.copy(other.plans);
questionnaires.putAll(other.questionnaires); questionnaires.copy(other.questionnaires);
operations.putAll(other.operations); operations.copy(other.operations);
systems.addAll(other.systems); systems.addAll(other.systems);
guides.putAll(other.guides); guides.copy(other.guides);
capstmts.putAll(other.capstmts); capstmts.copy(other.capstmts);
allowLoadingDuplicates = other.allowLoadingDuplicates; allowLoadingDuplicates = other.allowLoadingDuplicates;
tsServer = other.tsServer; tsServer = other.tsServer;
@ -187,27 +222,27 @@ public abstract class BaseWorkerContext implements IWorkerContext {
if (!allowLoadingDuplicates && hasResource(r.getClass(), url)) if (!allowLoadingDuplicates && hasResource(r.getClass(), url))
throw new DefinitionException("Duplicate Resource " + url); throw new DefinitionException("Duplicate Resource " + url);
if (r instanceof StructureDefinition) if (r instanceof StructureDefinition)
seeMetadataResource((StructureDefinition) m, structures, false); structures.see((StructureDefinition) m);
else if (r instanceof ValueSet) else if (r instanceof ValueSet)
seeMetadataResource((ValueSet) m, valueSets, false); valueSets.see((ValueSet) m);
else if (r instanceof CodeSystem) else if (r instanceof CodeSystem)
seeMetadataResource((CodeSystem) m, codeSystems, false); codeSystems.see((CodeSystem) m);
else if (r instanceof ImplementationGuide) else if (r instanceof ImplementationGuide)
seeMetadataResource((ImplementationGuide) m, guides, false); guides.see((ImplementationGuide) m);
else if (r instanceof CapabilityStatement) else if (r instanceof CapabilityStatement)
seeMetadataResource((CapabilityStatement) m, capstmts, false); capstmts.see((CapabilityStatement) m);
else if (r instanceof SearchParameter) else if (r instanceof SearchParameter)
seeMetadataResource((SearchParameter) m, searchParameters, false); searchParameters.see((SearchParameter) m);
else if (r instanceof PlanDefinition) else if (r instanceof PlanDefinition)
seeMetadataResource((PlanDefinition) m, plans, false); plans.see((PlanDefinition) m);
else if (r instanceof OperationDefinition) else if (r instanceof OperationDefinition)
seeMetadataResource((OperationDefinition) m, operations, false); operations.see((OperationDefinition) m);
else if (r instanceof Questionnaire) else if (r instanceof Questionnaire)
seeMetadataResource((Questionnaire) m, questionnaires, true); questionnaires.see((Questionnaire) m);
else if (r instanceof ConceptMap) else if (r instanceof ConceptMap)
seeMetadataResource((ConceptMap) m, maps, false); maps.see((ConceptMap) m);
else if (r instanceof StructureMap) else if (r instanceof StructureMap)
seeMetadataResource((StructureMap) m, transforms, false); transforms.see((StructureMap) m);
else if (r instanceof NamingSystem) else if (r instanceof NamingSystem)
systems.add((NamingSystem) r); systems.add((NamingSystem) r);
} }
@ -261,26 +296,38 @@ public abstract class BaseWorkerContext implements IWorkerContext {
throw new Error("Delimited versions have exact match for delimiter '"+delimiter+"' : "+newParts+" vs "+oldParts); throw new Error("Delimited versions have exact match for delimiter '"+delimiter+"' : "+newParts+" vs "+oldParts);
} }
protected <T extends MetadataResource> void seeMetadataResource(T r, Map<String, T> map, boolean addId) throws FHIRException { protected <T extends MetadataResource> void seeMetadataResource(T r, Map<String, T> map, List<T> list, boolean addId) throws FHIRException {
if (addId) // if (addId)
map.put(r.getId(), r); // todo: why? // map.put(r.getId(), r); // todo: why?
list.add(r);
if (r.hasUrl()) {
// first, this is the correct reosurce for this version (if it has a version)
if (r.hasVersion()) {
map.put(r.getUrl()+"|"+r.getVersion(), r);
}
// if we haven't get anything for this url, it's the correct version
if (!map.containsKey(r.getUrl())) if (!map.containsKey(r.getUrl()))
map.put(r.getUrl(), r); map.put(r.getUrl(), r);
else { else {
// If this resource already exists, see if it's the newest business version. The default resource to return if not qualified is the most recent business version List<T> rl = new ArrayList<T>();
MetadataResource existingResource = (MetadataResource)map.get(r.getUrl()); for (T t : list) {
if (r.hasVersion() && existingResource.hasVersion() && !r.getVersion().equals(existingResource.getVersion())) { if (t.getUrl().equals(r.getUrl()) && !rl.contains(t)) {
if (laterVersion(r.getVersion(), existingResource.getVersion())) { rl.add(t);
map.remove(r.getUrl()); }
map.put(r.getUrl(), r); }
Collections.sort(rl, new MetadataResourceVersionComparator<T>(list));
map.put(r.getUrl(), rl.get(rl.size()-1));
T latest = null;
for (T t : rl) {
if (VersionUtilities.versionsCompatible(t.getVersion(), r.getVersion())) {
latest = t;
}
}
if (latest != null) { // might be null if it's not using semver
map.put(r.getUrl()+"|"+VersionUtilities.getMajMin(latest.getVersion()), rl.get(rl.size()-1));
}
} }
} else
map.remove(r.getUrl());
map.put(r.getUrl(), r);
// throw new FHIRException("Multiple declarations of resource with same canonical URL (" + r.getUrl() + ") and version (" + (r.hasVersion() ? r.getVersion() : "" ) + ")");
} }
if (r.hasVersion())
map.put(r.getUrl()+"|"+r.getVersion(), r);
} }
@Override @Override
@ -293,7 +340,7 @@ public abstract class BaseWorkerContext implements IWorkerContext {
@Override @Override
public boolean supportsSystem(String system) throws TerminologyServiceException { public boolean supportsSystem(String system) throws TerminologyServiceException {
synchronized (lock) { synchronized (lock) {
if (codeSystems.containsKey(system) && codeSystems.get(system).getContent() != CodeSystemContentMode.NOTPRESENT) if (codeSystems.has(system) && codeSystems.get(system).getContent() != CodeSystemContentMode.NOTPRESENT)
return true; return true;
else if (supportedCodeSystems.contains(system)) else if (supportedCodeSystems.contains(system))
return true; return true;
@ -635,7 +682,7 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<ConceptMap> findMapsForSource(String url) throws FHIRException { public List<ConceptMap> findMapsForSource(String url) throws FHIRException {
synchronized (lock) { synchronized (lock) {
List<ConceptMap> res = new ArrayList<ConceptMap>(); List<ConceptMap> res = new ArrayList<ConceptMap>();
for (ConceptMap map : maps.values()) for (ConceptMap map : maps.getList())
if (((Reference) map.getSource()).getReference().equals(url)) if (((Reference) map.getSource()).getReference().equals(url))
res.add(map); res.add(map);
return res; return res;
@ -706,27 +753,27 @@ public abstract class BaseWorkerContext implements IWorkerContext {
if (uri.contains("#")) if (uri.contains("#"))
uri = uri.substring(0, uri.indexOf("#")); uri = uri.substring(0, uri.indexOf("#"));
if (class_ == Resource.class || class_ == null) { if (class_ == Resource.class || class_ == null) {
if (structures.containsKey(uri)) if (structures.has(uri))
return (T) structures.get(uri); return (T) structures.get(uri);
if (guides.containsKey(uri)) if (guides.has(uri))
return (T) guides.get(uri); return (T) guides.get(uri);
if (capstmts.containsKey(uri)) if (capstmts.has(uri))
return (T) capstmts.get(uri); return (T) capstmts.get(uri);
if (valueSets.containsKey(uri)) if (valueSets.has(uri))
return (T) valueSets.get(uri); return (T) valueSets.get(uri);
if (codeSystems.containsKey(uri)) if (codeSystems.has(uri))
return (T) codeSystems.get(uri); return (T) codeSystems.get(uri);
if (operations.containsKey(uri)) if (operations.has(uri))
return (T) operations.get(uri); return (T) operations.get(uri);
if (searchParameters.containsKey(uri)) if (searchParameters.has(uri))
return (T) searchParameters.get(uri); return (T) searchParameters.get(uri);
if (plans.containsKey(uri)) if (plans.has(uri))
return (T) plans.get(uri); return (T) plans.get(uri);
if (maps.containsKey(uri)) if (maps.has(uri))
return (T) maps.get(uri); return (T) maps.get(uri);
if (transforms.containsKey(uri)) if (transforms.has(uri))
return (T) transforms.get(uri); return (T) transforms.get(uri);
if (questionnaires.containsKey(uri)) if (questionnaires.has(uri))
return (T) questionnaires.get(uri); return (T) questionnaires.get(uri);
for (Map<String, Resource> rt : allResourcesById.values()) { for (Map<String, Resource> rt : allResourcesById.values()) {
for (Resource r : rt.values()) { for (Resource r : rt.values()) {
@ -747,13 +794,13 @@ public abstract class BaseWorkerContext implements IWorkerContext {
} else if (class_ == StructureMap.class) { } else if (class_ == StructureMap.class) {
return (T) transforms.get(uri); return (T) transforms.get(uri);
} else if (class_ == ValueSet.class) { } else if (class_ == ValueSet.class) {
if (valueSets.containsKey(uri+"|"+version)) if (valueSets.has(uri, version))
return (T) valueSets.get(uri+"|"+version); return (T) valueSets.get(uri, version);
else else
return (T) valueSets.get(uri); return (T) valueSets.get(uri);
} else if (class_ == CodeSystem.class) { } else if (class_ == CodeSystem.class) {
if (codeSystems.containsKey(uri+"|"+version)) if (codeSystems.has(uri, version))
return (T) codeSystems.get(uri+"|"+version); return (T) codeSystems.get(uri, version);
else else
return (T) codeSystems.get(uri); return (T) codeSystems.get(uri);
} else if (class_ == ConceptMap.class) { } else if (class_ == ConceptMap.class) {
@ -765,17 +812,10 @@ public abstract class BaseWorkerContext implements IWorkerContext {
return (T) od; return (T) od;
} else if (class_ == SearchParameter.class) { } else if (class_ == SearchParameter.class) {
SearchParameter res = searchParameters.get(uri); SearchParameter res = searchParameters.get(uri);
if (res == null) {
StringBuilder b = new StringBuilder();
for (String s : searchParameters.keySet()) {
b.append(s);
b.append("\r\n");
}
}
return (T) res; return (T) res;
} }
} }
if (class_ == CodeSystem.class && codeSystems.containsKey(uri)) if (class_ == CodeSystem.class && codeSystems.has(uri))
return (T) codeSystems.get(uri); return (T) codeSystems.get(uri);
if (class_ == Questionnaire.class) if (class_ == Questionnaire.class)
@ -943,23 +983,23 @@ public abstract class BaseWorkerContext implements IWorkerContext {
map.remove(id); map.remove(id);
if (fhirType.equals("StructureDefinition")) if (fhirType.equals("StructureDefinition"))
dropMetadataResource(structures, id); structures.drop(id);
else if (fhirType.equals("ImplementationGuide")) else if (fhirType.equals("ImplementationGuide"))
dropMetadataResource(guides, id); guides.drop(id);
else if (fhirType.equals("CapabilityStatement")) else if (fhirType.equals("CapabilityStatement"))
dropMetadataResource(capstmts, id); capstmts.drop(id);
else if (fhirType.equals("ValueSet")) else if (fhirType.equals("ValueSet"))
dropMetadataResource(valueSets, id); valueSets.drop(id);
else if (fhirType.equals("CodeSystem")) else if (fhirType.equals("CodeSystem"))
dropMetadataResource(codeSystems, id); codeSystems.drop(id);
else if (fhirType.equals("OperationDefinition")) else if (fhirType.equals("OperationDefinition"))
dropMetadataResource(operations, id); operations.drop(id);
else if (fhirType.equals("Questionnaire")) else if (fhirType.equals("Questionnaire"))
dropMetadataResource(questionnaires, id); questionnaires.drop(id);
else if (fhirType.equals("ConceptMap")) else if (fhirType.equals("ConceptMap"))
dropMetadataResource(maps, id); maps.drop(id);
else if (fhirType.equals("StructureMap")) else if (fhirType.equals("StructureMap"))
dropMetadataResource(transforms, id); transforms.drop(id);
else if (fhirType.equals("NamingSystem")) else if (fhirType.equals("NamingSystem"))
for (int i = systems.size()-1; i >= 0; i--) { for (int i = systems.size()-1; i >= 0; i--) {
if (systems.get(i).getId().equals(id)) if (systems.get(i).getId().equals(id))
@ -984,15 +1024,15 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<MetadataResource> allConformanceResources() { public List<MetadataResource> allConformanceResources() {
synchronized (lock) { synchronized (lock) {
List<MetadataResource> result = new ArrayList<MetadataResource>(); List<MetadataResource> result = new ArrayList<MetadataResource>();
result.addAll(structures.values()); structures.listAllM(result);
result.addAll(guides.values()); guides.listAllM(result);
result.addAll(capstmts.values()); capstmts.listAllM(result);
result.addAll(codeSystems.values()); codeSystems.listAllM(result);
result.addAll(valueSets.values()); valueSets.listAllM(result);
result.addAll(maps.values()); maps.listAllM(result);
result.addAll(transforms.values()); transforms.listAllM(result);
result.addAll(plans.values()); plans.listAllM(result);
result.addAll(questionnaires.values()); questionnaires.listAllM(result);
return result; return result;
} }
} }
@ -1016,7 +1056,7 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<ConceptMap> listMaps() { public List<ConceptMap> listMaps() {
List<ConceptMap> m = new ArrayList<ConceptMap>(); List<ConceptMap> m = new ArrayList<ConceptMap>();
synchronized (lock) { synchronized (lock) {
m.addAll(maps.values()); maps.listAll(m);
} }
return m; return m;
} }
@ -1024,7 +1064,7 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<StructureMap> listTransforms() { public List<StructureMap> listTransforms() {
List<StructureMap> m = new ArrayList<StructureMap>(); List<StructureMap> m = new ArrayList<StructureMap>();
synchronized (lock) { synchronized (lock) {
m.addAll(transforms.values()); transforms.listAll(m);
} }
return m; return m;
} }
@ -1038,7 +1078,7 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<StructureDefinition> listStructures() { public List<StructureDefinition> listStructures() {
List<StructureDefinition> m = new ArrayList<StructureDefinition>(); List<StructureDefinition> m = new ArrayList<StructureDefinition>();
synchronized (lock) { synchronized (lock) {
m.addAll(structures.values()); structures.listAll(m);
} }
return m; return m;
} }
@ -1138,55 +1178,44 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<StructureDefinition> getStructures() { public List<StructureDefinition> getStructures() {
List<StructureDefinition> res = new ArrayList<>(); List<StructureDefinition> res = new ArrayList<>();
synchronized (lock) { // tricky, because you need to lock this as well, but it's really not in use yet synchronized (lock) { // tricky, because you need to lock this as well, but it's really not in use yet
res.addAll(structures.values()); structures.listAll(res);
} }
return res; return res;
} }
public String getLinkForUrl(String corePath, String url) { public String getLinkForUrl(String corePath, String url) {
for (CodeSystem r : codeSystems.values()) if (codeSystems.has(url))
if (url.equals(r.getUrl())) return codeSystems.get(url).getUserString("path");
return r.getUserString("path");
for (ValueSet r : valueSets.values()) if (valueSets.has(url))
if (url.equals(r.getUrl())) return valueSets.get(url).getUserString("path");
return r.getUserString("path");
for (ConceptMap r : maps.values()) if (maps.has(url))
if (url.equals(r.getUrl())) return maps.get(url).getUserString("path");
return r.getUserString("path");
for (StructureMap r : transforms.values()) if (transforms.has(url))
if (url.equals(r.getUrl())) return transforms.get(url).getUserString("path");
return r.getUserString("path");
for (StructureDefinition r : structures.values()) if (structures.has(url))
if (url.equals(r.getUrl())) return structures.get(url).getUserString("path");
return r.getUserString("path");
for (ImplementationGuide r : guides.values()) if (guides.has(url))
if (url.equals(r.getUrl())) return guides.get(url).getUserString("path");
return r.getUserString("path");
for (CapabilityStatement r : capstmts.values()) if (capstmts.has(url))
if (url.equals(r.getUrl())) return capstmts.get(url).getUserString("path");
return r.getUserString("path");
for (SearchParameter r : searchParameters.values()) if (searchParameters.has(url))
if (url.equals(r.getUrl())) return searchParameters.get(url).getUserString("path");
return r.getUserString("path");
for (Questionnaire r : questionnaires.values()) if (questionnaires.has(url))
if (url.equals(r.getUrl())) return questionnaires.get(url).getUserString("path");
return r.getUserString("path");
for (OperationDefinition r : operations.values()) if (operations.has(url))
if (url.equals(r.getUrl())) return operations.get(url).getUserString("path");
return r.getUserString("path");
for (PlanDefinition r : plans.values()) if (plans.has(url))
if (url.equals(r.getUrl())) return plans.get(url).getUserString("path");
return r.getUserString("path");
if (url.equals("http://loinc.org")) if (url.equals("http://loinc.org"))
return corePath+"loinc.html"; return corePath+"loinc.html";
@ -1201,7 +1230,7 @@ public abstract class BaseWorkerContext implements IWorkerContext {
public List<ImplementationGuide> allImplementationGuides() { public List<ImplementationGuide> allImplementationGuides() {
List<ImplementationGuide> res = new ArrayList<>(); List<ImplementationGuide> res = new ArrayList<>();
res.addAll(guides.values()); guides.listAll(res);
return res; return res;
} }

View File

@ -0,0 +1,180 @@
package org.hl7.fhir.r5.context;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
import org.hl7.fhir.r5.context.BaseWorkerContext.MetadataResourceVersionComparator;
import org.hl7.fhir.r5.model.CodeSystem;
import org.hl7.fhir.r5.model.MetadataResource;
import org.hl7.fhir.utilities.VersionUtilities;
/**
* This manages a cached list of resources, and provides high speed access by URL / URL+version, and assumes that patch version doesn't matter for access
* note, though, that not all resources have semver versions
*
* @author graha
*
*/
public class MetadataResourceManager<T extends MetadataResource> {
public class MetadataResourceVersionComparator<T extends MetadataResource> implements Comparator<T> {
@Override
public int compare(T arg1, T arg2) {
String v1 = arg1.getVersion();
String v2 = arg2.getVersion();
if (v1 == null && v2 == null) {
return Integer.compare(list.indexOf(arg1), list.indexOf(arg2)); // retain original order
} else if (v1 == null) {
return -1;
} else if (v2 == null) {
return 1;
} else {
String mm1 = VersionUtilities.getMajMin(v1);
String mm2 = VersionUtilities.getMajMin(v2);
if (mm1 == null || mm2 == null) {
return v1.compareTo(v2);
} else {
return mm1.compareTo(mm2);
}
}
}
}
private List<T> list = new ArrayList<>();
private Map<String, T> map = new HashMap<>();
public void copy(MetadataResourceManager<T> source) {
list.clear();
map.clear();
list.addAll(source.list);
map.putAll(source.map);
}
public void see(T r) {
if (!r.hasId()) {
r.setId(UUID.randomUUID().toString());
}
if (map.containsKey(r.getId())) {
drop(r.getId());
}
list.add(r);
map.put(r.getId(), r); // we do this so we can drop by id
if (r.hasUrl()) {
// first, this is the correct reosurce for this version (if it has a version)
if (r.hasVersion()) {
map.put(r.getUrl()+"|"+r.getVersion(), r);
}
updateList(r.getUrl(), r.getVersion());
}
}
private void updateList(String url, String version) {
List<T> rl = new ArrayList<T>();
for (T t : list) {
if (url.equals(t.getUrl()) && !rl.contains(t)) {
rl.add(t);
}
}
if (rl.size() > 0) {
// sort by version as much as we are able
Collections.sort(rl, new MetadataResourceVersionComparator<T>());
// the current is the latest
map.put(url, rl.get(rl.size()-1));
// now, also, the latest for major/minor
if (version != null) {
T latest = null;
for (T t : rl) {
if (VersionUtilities.versionsCompatible(t.getVersion(), version)) {
latest = t;
}
}
if (latest != null) { // might be null if it's not using semver
map.put(url+"|"+VersionUtilities.getMajMin(latest.getVersion()), rl.get(rl.size()-1));
}
}
}
}
public T get(String url) {
return map.get(url);
}
public boolean has(String url) {
return map.containsKey(url);
}
public T get(String system, String version) {
if (map.containsKey(system+"|"+version))
return map.get(system+"|"+version);
String mm = VersionUtilities.getMajMin(version);
if (mm != null)
return map.get(system+"|"+mm);
else
return null;
}
public boolean has(String system, String version) {
if (map.containsKey(system+"|"+version))
return true;
String mm = VersionUtilities.getMajMin(version);
if (mm != null)
return map.containsKey(system+"|"+mm);
else
return false;
}
public int size() {
return list.size();
}
public void drop(String id) {
T res = map.get(id);
if (res != null) {
list.remove(res);
map.remove(id);
map.remove(res.getUrl());
if (res.hasVersion()) {
map.remove(res.getUrl()+"|"+res.getVersion());
String mm = VersionUtilities.getMajMin(res.getVersion());
if (mm != null) {
map.remove(res.getUrl()+"|"+mm);
}
}
updateList(res.getUrl(), res.getVersion());
}
}
public void listAll(List<T> result) {
result.addAll(list);
}
public void listAllM(List<MetadataResource> result) {
result.addAll(list);
}
public void clear() {
list.clear();
map.clear();
}
public List<T> getList() {
return list;
}
public Set<String> keys() {
return map.keySet();
}
}

View File

@ -608,7 +608,7 @@ public class SimpleWorkerContext extends BaseWorkerContext implements IWorkerCon
} }
public String listMapUrls() { public String listMapUrls() {
return Utilities.listCanonicalUrls(transforms.keySet()); return Utilities.listCanonicalUrls(transforms.keys());
} }

View File

@ -19,6 +19,7 @@ import org.junit.runners.Suite.SuiteClasses;
ShexGeneratorTests.class, ShexGeneratorTests.class,
BaseDateTimeTypeTest.class, BaseDateTimeTypeTest.class,
OpenApiGeneratorTest.class, OpenApiGeneratorTest.class,
MetadataResourceManagerTester.class,
SnapShotGenerationTests.class}) SnapShotGenerationTests.class})
public class AllR5Tests { public class AllR5Tests {

View File

@ -0,0 +1,226 @@
package org.hl7.fhir.r5.test;
import static org.junit.Assert.*;
import org.hl7.fhir.r5.context.MetadataResourceManager;
import org.hl7.fhir.r5.model.CodeSystem;
import org.hl7.fhir.r5.model.ValueSet;
import org.junit.Assert;
import org.junit.Test;
public class MetadataResourceManagerTester {
private MetadataResourceManager<ValueSet> mrm = new MetadataResourceManager<>();
@Test
public void testSingleNoVersion() {
ValueSet vs = new ValueSet();
vs.setId("2345");
vs.setUrl("http://url/ValueSet/234");
// no version
mrm.clear();
mrm.see(vs);
Assert.assertEquals(mrm.size(), 1);
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertNotNull(mrm.get("2345"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
mrm.see(vs);
Assert.assertEquals(mrm.size(), 1);
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertNotNull(mrm.get("2345"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
mrm.drop("2344");
Assert.assertEquals(mrm.size(), 1);
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertNotNull(mrm.get("2345"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
mrm.drop("2345");
Assert.assertEquals(mrm.size(), 0);
Assert.assertNull(mrm.get("http://url/ValueSet/234"));
Assert.assertNull(mrm.get("2345"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
}
@Test
public void testSingleWithVersion() {
ValueSet vs = new ValueSet();
vs.setId("2345");
vs.setUrl("http://url/ValueSet/234");
vs.setVersion("4.0.1");
mrm.clear();
mrm.see(vs);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
}
@Test
public void testSingleWithVersionNotSemVer() {
ValueSet vs = new ValueSet();
vs.setId("2345");
vs.setUrl("http://url/ValueSet/234");
vs.setVersion("20140403");
mrm.clear();
mrm.see(vs);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "20140403"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "20140402"));
Assert.assertNull(mrm.get("http://url/ValueSet/234", "2014"));
}
@Test
public void testSingleWithVersions1() {
ValueSet vs1 = new ValueSet();
vs1.setId("2345");
vs1.setUrl("http://url/ValueSet/234");
vs1.setVersion("4.0.1");
vs1.setName("1");
ValueSet vs2 = new ValueSet();
vs2.setId("2346");
vs2.setUrl("http://url/ValueSet/234");
vs2.setVersion("4.0.2");
vs2.setName("2");
mrm.clear();
mrm.see(vs1);
Assert.assertEquals(mrm.size(), 1);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertEquals(mrm.get("2345").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.0").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.1").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.2").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0").getName(), "1");
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
mrm.see(vs2);
Assert.assertEquals(mrm.size(), 2);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertEquals(mrm.get("2345").getName(), "1");
Assert.assertNotNull(mrm.get("2346"));
Assert.assertEquals(mrm.get("2346").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.0").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.1").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.2").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0").getName(), "2");
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
mrm.drop("2346"); // vs2;
Assert.assertEquals(mrm.size(), 1);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertNull(mrm.get("2346"));
Assert.assertEquals(mrm.get("2345").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.0").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.1").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.2").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0").getName(), "1");
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
}
@Test
public void testSingleWithVersions2() {
ValueSet vs1 = new ValueSet();
vs1.setId("2345");
vs1.setUrl("http://url/ValueSet/234");
vs1.setVersion("4.0.1");
vs1.setName("1");
ValueSet vs2 = new ValueSet();
vs2.setId("2346");
vs2.setUrl("http://url/ValueSet/234");
vs2.setVersion("4.0.2");
vs2.setName("2");
mrm.clear();
mrm.see(vs1);
Assert.assertEquals(mrm.size(), 1);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertEquals(mrm.get("2345").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.0").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.1").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.2").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0").getName(), "1");
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
mrm.see(vs2);
Assert.assertEquals(mrm.size(), 2);
Assert.assertNotNull(mrm.get("2345"));
Assert.assertEquals(mrm.get("2345").getName(), "1");
Assert.assertNotNull(mrm.get("2346"));
Assert.assertEquals(mrm.get("2346").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.0").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.1").getName(), "1");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.2").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0").getName(), "2");
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
mrm.drop("2345"); // vs1;
Assert.assertEquals(mrm.size(), 1);
Assert.assertNull(mrm.get("2345"));
Assert.assertNotNull(mrm.get("2346"));
Assert.assertEquals(mrm.get("2346").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.0").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.1"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.1").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0.2"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0.2").getName(), "2");
Assert.assertNotNull(mrm.get("http://url/ValueSet/234", "4.0"));
Assert.assertEquals(mrm.get("http://url/ValueSet/234", "4.0").getName(), "2");
Assert.assertNull(mrm.get("http://url/ValueSet/234", "4.1"));
}
}

View File

@ -115,7 +115,13 @@ public class VersionUtilities {
} }
public static boolean versionsCompatible(String v1, String v2) { public static boolean versionsCompatible(String v1, String v2) {
return v1.substring(0, 3).equals(v2.substring(0, 3)); String mm1 = getMajMin(v1);
String mm2 = getMajMin(v2);
if (mm1 == null || mm2 == null) {
return false;
} else {
return mm1.equals(mm2);
}
} }
public static boolean isCorePackage(String s) { public static boolean isCorePackage(String s) {
@ -125,4 +131,16 @@ public class VersionUtilities {
return Utilities.existsInList(s, "hl7.fhir.core","hl7.fhir.r2.core", "hl7.fhir.r2b.core", "hl7.fhir.r3.core", "hl7.fhir.r4.core"); return Utilities.existsInList(s, "hl7.fhir.core","hl7.fhir.r2.core", "hl7.fhir.r2b.core", "hl7.fhir.r3.core", "hl7.fhir.r4.core");
} }
public static String getMajMin(String version) {
if (version == null)
return null;
if (Utilities.charCount(version, '.') == 2) {
String[] p = version.split("\\.");
return p[0]+"."+p[1];
} else {
return null;
}
}
} }