Merge pull request #760 from hapifhir/gg-202203-xver-extensions
Gg 202203 xver extensions
This commit is contained in:
commit
27e3dcfaa8
|
@ -1,8 +1,10 @@
|
||||||
## Validator Changes
|
## Validator Changes
|
||||||
|
|
||||||
* no changes
|
* fix handling of xver extensions with choice sub-extensions
|
||||||
|
|
||||||
## Other code changes
|
## Other code changes
|
||||||
|
|
||||||
|
* more presenting extensions when rendering.
|
||||||
|
* Get package id from 'package' field when 'npm-package' unavailable
|
||||||
* Ignore META-INF contents from other dependencies while building Validator jar
|
* Ignore META-INF contents from other dependencies while building Validator jar
|
||||||
* Fix to r5 TestingUtilities to allow NarrativeGenerationTests to pass.
|
* Fix to r5 TestingUtilities to allow NarrativeGenerationTests to pass.
|
||||||
|
|
|
@ -0,0 +1,189 @@
|
||||||
|
package org.hl7.fhir.convertors.misc;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import org.hl7.fhir.utilities.json.JsonTrackingParser;
|
||||||
|
|
||||||
|
import com.google.gson.JsonObject;
|
||||||
|
|
||||||
|
public class JsonProcessor {
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
new JsonProcessor().process(args[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void process(String source) throws IOException {
|
||||||
|
JsonObject json = JsonTrackingParser.parseJsonFile(source);
|
||||||
|
process(json);
|
||||||
|
JsonTrackingParser.write(json, new File(source), true);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private void process(JsonObject json) {
|
||||||
|
process(json, "ActivityDefinition.status");
|
||||||
|
process(json, "CapabilityStatement.status");
|
||||||
|
process(json, "CodeSystem.status");
|
||||||
|
process(json, "CompartmentDefinition.status");
|
||||||
|
process(json, "ConceptMap.status");
|
||||||
|
process(json, "DataElement.status");
|
||||||
|
process(json, "ExpansionProfile.status");
|
||||||
|
process(json, "GraphDefinition.status");
|
||||||
|
process(json, "ImplementationGuide.status");
|
||||||
|
process(json, "Library.status");
|
||||||
|
process(json, "Measure.status");
|
||||||
|
process(json, "MessageDefinition.status");
|
||||||
|
process(json, "OperationDefinition.status");
|
||||||
|
process(json, "PlanDefinition.status");
|
||||||
|
process(json, "Questionnaire.status");
|
||||||
|
process(json, "SearchParameter.status");
|
||||||
|
process(json, "ServiceDefinition.status");
|
||||||
|
process(json, "StructureDefinition.status");
|
||||||
|
process(json, "StructureMap.status");
|
||||||
|
process(json, "TestScript.status");
|
||||||
|
process(json, "ValueSet.status");
|
||||||
|
process(json, "ActivityDefinition.experimental");
|
||||||
|
process(json, "CapabilityStatement.experimental");
|
||||||
|
process(json, "CodeSystem.experimental");
|
||||||
|
process(json, "CompartmentDefinition.experimental");
|
||||||
|
process(json, "ConceptMap.experimental");
|
||||||
|
process(json, "DataElement.experimental");
|
||||||
|
process(json, "ExpansionProfile.experimental");
|
||||||
|
process(json, "GraphDefinition.experimental");
|
||||||
|
process(json, "ImplementationGuide.experimental");
|
||||||
|
process(json, "Library.experimental");
|
||||||
|
process(json, "Measure.experimental");
|
||||||
|
process(json, "MessageDefinition.experimental");
|
||||||
|
process(json, "OperationDefinition.experimental");
|
||||||
|
process(json, "PlanDefinition.experimental");
|
||||||
|
process(json, "Questionnaire.experimental");
|
||||||
|
process(json, "SearchParameter.experimental");
|
||||||
|
process(json, "ServiceDefinition.experimental");
|
||||||
|
process(json, "StructureDefinition.experimental");
|
||||||
|
process(json, "StructureMap.experimental");
|
||||||
|
process(json, "TestScript.experimental");
|
||||||
|
process(json, "ValueSet.experimental");
|
||||||
|
process(json, "Identifier.use");
|
||||||
|
process(json, "Quantity.comparator");
|
||||||
|
process(json, "Address.use");
|
||||||
|
process(json, "ContactPoint.use");
|
||||||
|
process(json, "HumanName.use");
|
||||||
|
process(json, "BackboneElement.modifierExtension");
|
||||||
|
process(json, "DomainResource.modifierExtension");
|
||||||
|
process(json, "Resource.implicitRules");
|
||||||
|
process(json, "Account.status");
|
||||||
|
process(json, "AllergyIntolerance.clinicalStatus");
|
||||||
|
process(json, "AllergyIntolerance.verificationStatus");
|
||||||
|
process(json, "Appointment.status");
|
||||||
|
process(json, "AppointmentResponse.participantStatus");
|
||||||
|
process(json, "Basic.code");
|
||||||
|
process(json, "BodySite.active");
|
||||||
|
process(json, "CarePlan.status");
|
||||||
|
process(json, "CarePlan.intent");
|
||||||
|
process(json, "CarePlan.activity.detail.status");
|
||||||
|
process(json, "CarePlan.activity.detail.prohibited");
|
||||||
|
process(json, "CareTeam.status");
|
||||||
|
process(json, "ChargeItem.status");
|
||||||
|
process(json, "Claim.status");
|
||||||
|
process(json, "ClaimResponse.status");
|
||||||
|
process(json, "ClinicalImpression.status");
|
||||||
|
process(json, "Communication.status");
|
||||||
|
process(json, "Communication.notDone");
|
||||||
|
process(json, "CommunicationRequest.status");
|
||||||
|
process(json, "Composition.status");
|
||||||
|
process(json, "Composition.confidentiality");
|
||||||
|
process(json, "Composition.section.mode");
|
||||||
|
process(json, "ConceptMap.group.element.target.equivalence");
|
||||||
|
process(json, "Condition.clinicalStatus");
|
||||||
|
process(json, "Condition.verificationStatus");
|
||||||
|
process(json, "Consent.status");
|
||||||
|
process(json, "Contract.status");
|
||||||
|
process(json, "Coverage.status");
|
||||||
|
process(json, "DetectedIssue.status");
|
||||||
|
process(json, "Device.status");
|
||||||
|
process(json, "DeviceRequest.status");
|
||||||
|
process(json, "DeviceRequest.intent");
|
||||||
|
process(json, "DeviceUseStatement.status");
|
||||||
|
process(json, "DiagnosticReport.status");
|
||||||
|
process(json, "DocumentManifest.status");
|
||||||
|
process(json, "DocumentReference.status");
|
||||||
|
process(json, "DocumentReference.relatesTo");
|
||||||
|
process(json, "EligibilityRequest.status");
|
||||||
|
process(json, "EligibilityResponse.status");
|
||||||
|
process(json, "Encounter.status");
|
||||||
|
process(json, "Endpoint.status");
|
||||||
|
process(json, "EnrollmentRequest.status");
|
||||||
|
process(json, "EnrollmentResponse.status");
|
||||||
|
process(json, "EpisodeOfCare.status");
|
||||||
|
process(json, "ExplanationOfBenefit.status");
|
||||||
|
process(json, "FamilyMemberHistory.status");
|
||||||
|
process(json, "FamilyMemberHistory.notDone");
|
||||||
|
process(json, "FamilyMemberHistory.estimatedAge");
|
||||||
|
process(json, "Flag.status");
|
||||||
|
process(json, "Goal.status");
|
||||||
|
process(json, "Group.characteristic.exclude");
|
||||||
|
process(json, "GuidanceResponse.status");
|
||||||
|
process(json, "HealthcareService.active");
|
||||||
|
process(json, "Immunization.status");
|
||||||
|
process(json, "Immunization.notGiven");
|
||||||
|
process(json, "List.status");
|
||||||
|
process(json, "List.mode");
|
||||||
|
process(json, "List.entry.deleted");
|
||||||
|
process(json, "Location.status");
|
||||||
|
process(json, "Location.mode");
|
||||||
|
process(json, "MeasureReport.status");
|
||||||
|
process(json, "MedicationAdministration.status");
|
||||||
|
process(json, "MedicationAdministration.notGiven");
|
||||||
|
process(json, "MedicationDispense.status");
|
||||||
|
process(json, "MedicationRequest.status");
|
||||||
|
process(json, "MedicationRequest.intent");
|
||||||
|
process(json, "MedicationRequest.substitution.allowed");
|
||||||
|
process(json, "MedicationStatement.status");
|
||||||
|
process(json, "MedicationStatement.taken");
|
||||||
|
process(json, "NamingSystem.status");
|
||||||
|
process(json, "NutritionOrder.status");
|
||||||
|
process(json, "Observation.status");
|
||||||
|
process(json, "OperationOutcome.issue.severity");
|
||||||
|
process(json, "Organization.active");
|
||||||
|
process(json, "Patient.active");
|
||||||
|
process(json, "Patient.deceased[x]");
|
||||||
|
process(json, "Patient.animal");
|
||||||
|
process(json, "Patient.link");
|
||||||
|
process(json, "PaymentNotice.status");
|
||||||
|
process(json, "PaymentReconciliation.status");
|
||||||
|
process(json, "Person.active");
|
||||||
|
process(json, "Procedure.status");
|
||||||
|
process(json, "Procedure.notDone");
|
||||||
|
process(json, "ProcedureRequest.status");
|
||||||
|
process(json, "ProcedureRequest.intent");
|
||||||
|
process(json, "ProcedureRequest.doNotPerform");
|
||||||
|
process(json, "ProcessRequest.status");
|
||||||
|
process(json, "ProcessResponse.status");
|
||||||
|
process(json, "Questionnaire.item.enableWhen");
|
||||||
|
process(json, "QuestionnaireResponse.status");
|
||||||
|
process(json, "ReferralRequest.status");
|
||||||
|
process(json, "ReferralRequest.intent");
|
||||||
|
process(json, "RelatedPerson.active");
|
||||||
|
process(json, "RequestGroup.status");
|
||||||
|
process(json, "RequestGroup.intent");
|
||||||
|
process(json, "ResearchStudy.status");
|
||||||
|
process(json, "ResearchSubject.status");
|
||||||
|
process(json, "Schedule.active");
|
||||||
|
process(json, "Specimen.status");
|
||||||
|
process(json, "Subscription.status");
|
||||||
|
process(json, "SupplyDelivery.status");
|
||||||
|
process(json, "SupplyRequest.status");
|
||||||
|
process(json, "TestReport.status");
|
||||||
|
process(json, "ValueSet.compose.include.filter");
|
||||||
|
process(json, "VisionPrescription.status");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void process(JsonObject json, String name) {
|
||||||
|
JsonObject j = json.getAsJsonObject(name);
|
||||||
|
if (j == null) {
|
||||||
|
System.out.println("Can't find "+name);
|
||||||
|
} else {
|
||||||
|
j.addProperty("modifier", true);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,13 +6,19 @@ import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Map.Entry;
|
import java.util.Map.Entry;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import javax.naming.ldap.StartTlsRequest;
|
import javax.naming.ldap.StartTlsRequest;
|
||||||
import javax.xml.parsers.ParserConfigurationException;
|
import javax.xml.parsers.ParserConfigurationException;
|
||||||
|
|
||||||
|
import org.hl7.fhir.convertors.misc.XVerPackegeFixer.References;
|
||||||
|
import org.hl7.fhir.dstu2.model.ElementDefinition;
|
||||||
|
import org.hl7.fhir.dstu2.model.StructureDefinition;
|
||||||
|
import org.hl7.fhir.dstu2.model.StructureDefinition.StructureDefinitionSnapshotComponent;
|
||||||
import org.hl7.fhir.exceptions.FHIRFormatError;
|
import org.hl7.fhir.exceptions.FHIRFormatError;
|
||||||
import org.hl7.fhir.utilities.Utilities;
|
import org.hl7.fhir.utilities.Utilities;
|
||||||
import org.hl7.fhir.utilities.json.JsonTrackingParser;
|
import org.hl7.fhir.utilities.json.JsonTrackingParser;
|
||||||
|
@ -24,12 +30,35 @@ import com.google.gson.JsonObject;
|
||||||
|
|
||||||
public class XVerPackegeFixer {
|
public class XVerPackegeFixer {
|
||||||
|
|
||||||
private static final String R5_FOLDER = "C:\\work\\org.hl7.fhir\\packages\\hl7.fhir.rX\\hl7.fhir.r5.core\\package";
|
public static class References {
|
||||||
private static final String R4_FOLDER = "C:\\work\\org.hl7.fhir\\packages\\hl7.fhir.rX\\hl7.fhir.r4.core\\package";
|
private boolean modifier;
|
||||||
private static final String R3_FOLDER = "C:\\work\\org.hl7.fhir\\packages\\hl7.fhir.rX\\hl7.fhir.r3.core\\package";
|
private boolean inherited;
|
||||||
private static final String R2B_FOLDER = "C:\\work\\org.hl7.fhir\\packages\\hl7.fhir.rX\\hl7.fhir.r2b.core\\package";
|
|
||||||
private static final String R2_FOLDER = "C:\\work\\org.hl7.fhir\\packages\\hl7.fhir.rX\\hl7.fhir.r2.core\\package";
|
public boolean getModifier() {
|
||||||
private static int mod;
|
return modifier;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setModifier(boolean value) {
|
||||||
|
this.modifier = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean getInherited() {
|
||||||
|
return inherited;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setInherited(boolean inherited) {
|
||||||
|
this.inherited = inherited;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static final String R5_FOLDER = "/Users/grahamegrieve/work/packages/hl7.fhir.rX/hl7.fhir.r5.core/package";
|
||||||
|
private static final String R4_FOLDER = "/Users/grahamegrieve/work/packages/hl7.fhir.rX/hl7.fhir.r4.core/package";
|
||||||
|
private static final String R3_FOLDER = "/Users/grahamegrieve/work/packages/hl7.fhir.rX/hl7.fhir.r3.core/package";
|
||||||
|
private static final String R2B_FOLDER = "/Users/grahamegrieve/work/packages/hl7.fhir.rX/hl7.fhir.r2b.core/package";
|
||||||
|
private static final String R2_FOLDER = "/Users/grahamegrieve/work/packages/hl7.fhir.rX/hl7.fhir.r2.core/package";
|
||||||
|
private static int modCount;
|
||||||
|
|
||||||
private static Map<String, org.hl7.fhir.r5.model.StructureDefinition> map5 = new HashMap<>();
|
private static Map<String, org.hl7.fhir.r5.model.StructureDefinition> map5 = new HashMap<>();
|
||||||
private static Map<String, org.hl7.fhir.r4.model.StructureDefinition> map4 = new HashMap<>();
|
private static Map<String, org.hl7.fhir.r4.model.StructureDefinition> map4 = new HashMap<>();
|
||||||
|
@ -38,21 +67,23 @@ public class XVerPackegeFixer {
|
||||||
private static Map<String, org.hl7.fhir.dstu2016may.model.StructureDefinition> map2b = new HashMap<>();
|
private static Map<String, org.hl7.fhir.dstu2016may.model.StructureDefinition> map2b = new HashMap<>();
|
||||||
|
|
||||||
public static void main(String[] args) throws FileNotFoundException, ParserConfigurationException, SAXException, IOException {
|
public static void main(String[] args) throws FileNotFoundException, ParserConfigurationException, SAXException, IOException {
|
||||||
mod = 0;
|
modCount = 0;
|
||||||
for (File f : new File(args[0]).listFiles()) {
|
for (File f : new File(args[0]).listFiles()) {
|
||||||
if (f.getName().startsWith("xver-")) {
|
if (f.getName().startsWith("xver-") && f.getName().contains("1.0")) {
|
||||||
JsonObject j = JsonTrackingParser.parseJson(f);
|
JsonObject j = JsonTrackingParser.parseJson(f);
|
||||||
fixUp(j, f.getName());
|
fixUp(j, f.getName());
|
||||||
JsonTrackingParser.write(j, f, true);
|
JsonTrackingParser.write(j, f, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
System.out.println("all done: "+mod+" modifiers");
|
System.out.println("all done: "+modCount+" modifiers");
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void fixUp(JsonObject j, String name) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static void fixUp(JsonObject j, String name) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
name = name.replace(".json", "");
|
name = name.replace(".json", "");
|
||||||
System.out.println("Process "+name);
|
System.out.println("Process "+name);
|
||||||
String version = name.substring(name.lastIndexOf("-")+1);
|
String version = name.substring(name.lastIndexOf("-")+1);
|
||||||
|
Set<String> pr = new HashSet<>();
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for (Entry<String, JsonElement> e : j.entrySet()) {
|
for (Entry<String, JsonElement> e : j.entrySet()) {
|
||||||
if (i == 50) {
|
if (i == 50) {
|
||||||
|
@ -62,18 +93,31 @@ public class XVerPackegeFixer {
|
||||||
i++;
|
i++;
|
||||||
String n = e.getKey();
|
String n = e.getKey();
|
||||||
JsonObject o = ((JsonObject) e.getValue());
|
JsonObject o = ((JsonObject) e.getValue());
|
||||||
boolean ok = (o.has("types") && o.getAsJsonArray("types").size() > 0) || (o.has("elements") && o.getAsJsonArray("elements").size() > 0);
|
// boolean ok = (o.has("types") && o.getAsJsonArray("types").size() > 0) || (o.has("elements") && o.getAsJsonArray("elements").size() > 0);
|
||||||
if (!ok) {
|
// if (!ok) {
|
||||||
|
if (o.has("types")) {
|
||||||
|
o.remove("types");
|
||||||
|
}
|
||||||
|
if (o.has("elements")) {
|
||||||
|
o.remove("elements");
|
||||||
|
}
|
||||||
|
if (o.has("modifier")) {
|
||||||
|
o.remove("modifier");
|
||||||
|
}
|
||||||
List<String> types = new ArrayList<>();
|
List<String> types = new ArrayList<>();
|
||||||
List<String> elements = new ArrayList<>();
|
List<String> elements = new ArrayList<>();
|
||||||
getElementInfo(version, n, types, elements);
|
References mod = new References();
|
||||||
|
getElementInfo(version, n, types, elements, mod);
|
||||||
|
if (mod.getInherited()) {
|
||||||
|
pr.add(n);
|
||||||
|
}
|
||||||
if (elements.size() > 0) {
|
if (elements.size() > 0) {
|
||||||
JsonArray arr = o.getAsJsonArray("elements");
|
JsonArray arr = o.getAsJsonArray("elements");
|
||||||
if (arr == null) {
|
if (arr == null) {
|
||||||
arr = new JsonArray();
|
arr = new JsonArray();
|
||||||
o.add("elements", arr);
|
o.add("elements", arr);
|
||||||
}
|
}
|
||||||
for (String s : types) {
|
for (String s : elements) {
|
||||||
arr.add(s);
|
arr.add(s);
|
||||||
}
|
}
|
||||||
} else if (types.size() > 0) {
|
} else if (types.size() > 0) {
|
||||||
|
@ -86,30 +130,37 @@ public class XVerPackegeFixer {
|
||||||
arr.add(s);
|
arr.add(s);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (mod.getModifier()) {
|
||||||
|
o.addProperty("modifier", true);
|
||||||
|
modCount++;
|
||||||
}
|
}
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
for (String s : pr) {
|
||||||
|
j.remove(s);
|
||||||
}
|
}
|
||||||
System.out.println("done");
|
System.out.println("done");
|
||||||
}
|
}
|
||||||
|
|
||||||
private static boolean getElementInfo(String version, String n, List<String> types, List<String> elements) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static boolean getElementInfo(String version, String n, List<String> types, List<String> elements, References mod) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
if ("contained".equals(n.substring(n.indexOf(".")+1))) {
|
// if ("contained".equals(n.substring(n.indexOf(".")+1))) {
|
||||||
return false;
|
// return false;
|
||||||
}
|
// }
|
||||||
switch (version) {
|
switch (version) {
|
||||||
case "4.6": return getElementInfoR5(n, types, elements);
|
case "4.6": return getElementInfoR5(n, types, elements, mod);
|
||||||
case "4.0": return getElementInfoR4(n, types, elements);
|
case "4.0": return getElementInfoR4(n, types, elements, mod);
|
||||||
case "3.0": return getElementInfoR3(n, types, elements);
|
case "3.0": return getElementInfoR3(n, types, elements, mod);
|
||||||
case "1.4": return getElementInfoR2B(n, types, elements);
|
case "1.4": return getElementInfoR2B(n, types, elements, mod);
|
||||||
case "1.0": return getElementInfoR2(n, types, elements);
|
case "1.0": return getElementInfoR2(n, types, elements, mod);
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Object tail(String value) {
|
private static String tail(String value) {
|
||||||
return value.contains("/") ? value.substring(value.lastIndexOf("/")+1) : value;
|
return value.contains("/") ? value.substring(value.lastIndexOf("/")+1) : value;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static boolean getElementInfoR5(String n, List<String> types, List<String> elements) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static boolean getElementInfoR5(String n, List<String> types, List<String> elements, References mod) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
String tn = n.substring(0, n.indexOf("."));
|
String tn = n.substring(0, n.indexOf("."));
|
||||||
org.hl7.fhir.r5.model.StructureDefinition sd = null;
|
org.hl7.fhir.r5.model.StructureDefinition sd = null;
|
||||||
if (map5.containsKey(tn)) {
|
if (map5.containsKey(tn)) {
|
||||||
|
@ -120,6 +171,7 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
for (org.hl7.fhir.r5.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
for (org.hl7.fhir.r5.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
||||||
if (ed.getPath().equals(n)) {
|
if (ed.getPath().equals(n)) {
|
||||||
|
mod.setModifier(ed.getIsModifier());
|
||||||
List<org.hl7.fhir.r5.model.ElementDefinition> children = listChildrenR5(sd.getSnapshot().getElement(), ed);
|
List<org.hl7.fhir.r5.model.ElementDefinition> children = listChildrenR5(sd.getSnapshot().getElement(), ed);
|
||||||
if (children.size() > 0) {
|
if (children.size() > 0) {
|
||||||
for (org.hl7.fhir.r5.model.ElementDefinition c : children) {
|
for (org.hl7.fhir.r5.model.ElementDefinition c : children) {
|
||||||
|
@ -164,7 +216,7 @@ public class XVerPackegeFixer {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static boolean getElementInfoR4(String n, List<String> types, List<String> elements) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static boolean getElementInfoR4(String n, List<String> types, List<String> elements, References mod) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
String tn = n.substring(0, n.indexOf("."));
|
String tn = n.substring(0, n.indexOf("."));
|
||||||
org.hl7.fhir.r4.model.StructureDefinition sd = null;
|
org.hl7.fhir.r4.model.StructureDefinition sd = null;
|
||||||
if (map4.containsKey(tn)) {
|
if (map4.containsKey(tn)) {
|
||||||
|
@ -175,6 +227,7 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
for (org.hl7.fhir.r4.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
for (org.hl7.fhir.r4.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
||||||
if (ed.getPath().equals(n)) {
|
if (ed.getPath().equals(n)) {
|
||||||
|
mod.setModifier(ed.getIsModifier());
|
||||||
List<org.hl7.fhir.r4.model.ElementDefinition> children = listChildrenR4(sd.getSnapshot().getElement(), ed);
|
List<org.hl7.fhir.r4.model.ElementDefinition> children = listChildrenR4(sd.getSnapshot().getElement(), ed);
|
||||||
if (children.size() > 0) {
|
if (children.size() > 0) {
|
||||||
for (org.hl7.fhir.r4.model.ElementDefinition c : children) {
|
for (org.hl7.fhir.r4.model.ElementDefinition c : children) {
|
||||||
|
@ -220,7 +273,7 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static boolean getElementInfoR3(String n, List<String> types, List<String> elements) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static boolean getElementInfoR3(String n, List<String> types, List<String> elements, References mod) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
String tn = n.substring(0, n.indexOf("."));
|
String tn = n.substring(0, n.indexOf("."));
|
||||||
org.hl7.fhir.dstu3.model.StructureDefinition sd = null;
|
org.hl7.fhir.dstu3.model.StructureDefinition sd = null;
|
||||||
if (map3.containsKey(tn)) {
|
if (map3.containsKey(tn)) {
|
||||||
|
@ -231,6 +284,7 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
for (org.hl7.fhir.dstu3.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
for (org.hl7.fhir.dstu3.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
||||||
if (ed.getPath().equals(n)) {
|
if (ed.getPath().equals(n)) {
|
||||||
|
mod.setModifier(ed.getIsModifier());
|
||||||
List<org.hl7.fhir.dstu3.model.ElementDefinition> children = listChildrenR3(sd.getSnapshot().getElement(), ed);
|
List<org.hl7.fhir.dstu3.model.ElementDefinition> children = listChildrenR3(sd.getSnapshot().getElement(), ed);
|
||||||
if (children.size() > 0) {
|
if (children.size() > 0) {
|
||||||
for (org.hl7.fhir.dstu3.model.ElementDefinition c : children) {
|
for (org.hl7.fhir.dstu3.model.ElementDefinition c : children) {
|
||||||
|
@ -272,17 +326,14 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static boolean getElementInfoR2(String n, List<String> types, List<String> elements) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static boolean getElementInfoR2(String n, List<String> types, List<String> elements, References mod) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
String tn = n.substring(0, n.indexOf("."));
|
String tn = n.substring(0, n.indexOf("."));
|
||||||
org.hl7.fhir.dstu2.model.StructureDefinition sd = null;
|
org.hl7.fhir.dstu2.model.StructureDefinition sd = null;
|
||||||
if (map2.containsKey(tn)) {
|
sd = loadType(tn);
|
||||||
sd = map2.get(tn);
|
mod.setInherited(isInherited(n, sd));
|
||||||
} else {
|
|
||||||
sd = (org.hl7.fhir.dstu2.model.StructureDefinition) new org.hl7.fhir.dstu2.formats.JsonParser().parse(new FileInputStream(Utilities.path(R2_FOLDER, "StructureDefinition-"+tn+".json")));
|
|
||||||
map2.put(tn, sd);
|
|
||||||
}
|
|
||||||
for (org.hl7.fhir.dstu2.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
for (org.hl7.fhir.dstu2.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
||||||
if (ed.getPath().equals(n)) {
|
if (ed.getPath().equals(n)) {
|
||||||
|
mod.setModifier(ed.getIsModifier());
|
||||||
List<org.hl7.fhir.dstu2.model.ElementDefinition> children = listChildrenR2(sd.getSnapshot().getElement(), ed);
|
List<org.hl7.fhir.dstu2.model.ElementDefinition> children = listChildrenR2(sd.getSnapshot().getElement(), ed);
|
||||||
if (children.size() > 0) {
|
if (children.size() > 0) {
|
||||||
for (org.hl7.fhir.dstu2.model.ElementDefinition c : children) {
|
for (org.hl7.fhir.dstu2.model.ElementDefinition c : children) {
|
||||||
|
@ -290,8 +341,19 @@ public class XVerPackegeFixer {
|
||||||
elements.add(en);
|
elements.add(en);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
boolean isReference = false;
|
||||||
|
StringBuilder r = new StringBuilder();
|
||||||
|
r.append("Reference(");
|
||||||
for (org.hl7.fhir.dstu2.model.ElementDefinition.TypeRefComponent t : ed.getType()) {
|
for (org.hl7.fhir.dstu2.model.ElementDefinition.TypeRefComponent t : ed.getType()) {
|
||||||
if (t.hasProfile()) {
|
if (t.hasProfile()) {
|
||||||
|
if ("Reference".equals(t.getCode())) {
|
||||||
|
for (org.hl7.fhir.dstu2.model.UriType u : t.getProfile()) {
|
||||||
|
if (isReference)
|
||||||
|
r.append("|");
|
||||||
|
r.append(tail(u.getValue()));
|
||||||
|
isReference = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
StringBuilder b = new StringBuilder();
|
StringBuilder b = new StringBuilder();
|
||||||
b.append(t.getCode());
|
b.append(t.getCode());
|
||||||
b.append("(");
|
b.append("(");
|
||||||
|
@ -302,16 +364,53 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
b.append(")");
|
b.append(")");
|
||||||
types.add(b.toString());
|
types.add(b.toString());
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
types.add(t.getCode());
|
types.add(t.getCode());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (isReference) {
|
||||||
|
r.append(")");
|
||||||
|
types.add(r.toString());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static org.hl7.fhir.dstu2.model.StructureDefinition loadType(String tn)
|
||||||
|
throws IOException, FileNotFoundException {
|
||||||
|
org.hl7.fhir.dstu2.model.StructureDefinition sd;
|
||||||
|
if (map2.containsKey(tn)) {
|
||||||
|
sd = map2.get(tn);
|
||||||
|
} else {
|
||||||
|
sd = (org.hl7.fhir.dstu2.model.StructureDefinition) new org.hl7.fhir.dstu2.formats.JsonParser().parse(new FileInputStream(Utilities.path(R2_FOLDER, "StructureDefinition-"+tn+".json")));
|
||||||
|
map2.put(tn, sd);
|
||||||
|
}
|
||||||
|
return sd;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static boolean isInherited(String n, StructureDefinition sd) throws FileNotFoundException, IOException {
|
||||||
|
String tail = n.substring(n.indexOf(".")+1);
|
||||||
|
while (sd != null) {
|
||||||
|
sd = sd.hasBase() ? loadType(tail(sd.getBase())) : null;
|
||||||
|
if (sd != null && hasPath(sd.getSnapshot(), sd.getName()+"."+tail)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean hasPath(StructureDefinitionSnapshotComponent snapshot, String path) {
|
||||||
|
for (ElementDefinition ed : snapshot.getElement()) {
|
||||||
|
if (ed.getPath().equals(path)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
private static List<org.hl7.fhir.dstu2.model.ElementDefinition> listChildrenR2(List<org.hl7.fhir.dstu2.model.ElementDefinition> list, org.hl7.fhir.dstu2.model.ElementDefinition ed) {
|
private static List<org.hl7.fhir.dstu2.model.ElementDefinition> listChildrenR2(List<org.hl7.fhir.dstu2.model.ElementDefinition> list, org.hl7.fhir.dstu2.model.ElementDefinition ed) {
|
||||||
List<org.hl7.fhir.dstu2.model.ElementDefinition> res = new ArrayList<>();
|
List<org.hl7.fhir.dstu2.model.ElementDefinition> res = new ArrayList<>();
|
||||||
|
@ -328,7 +427,7 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static boolean getElementInfoR2B(String n, List<String> types, List<String> elements) throws FHIRFormatError, FileNotFoundException, IOException {
|
private static boolean getElementInfoR2B(String n, List<String> types, List<String> elements, References mod) throws FHIRFormatError, FileNotFoundException, IOException {
|
||||||
String tn = n.substring(0, n.indexOf("."));
|
String tn = n.substring(0, n.indexOf("."));
|
||||||
org.hl7.fhir.dstu2016may.model.StructureDefinition sd = null;
|
org.hl7.fhir.dstu2016may.model.StructureDefinition sd = null;
|
||||||
if (map2b.containsKey(tn)) {
|
if (map2b.containsKey(tn)) {
|
||||||
|
@ -339,6 +438,7 @@ public class XVerPackegeFixer {
|
||||||
}
|
}
|
||||||
for (org.hl7.fhir.dstu2016may.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
for (org.hl7.fhir.dstu2016may.model.ElementDefinition ed : sd.getSnapshot().getElement()) {
|
||||||
if (ed.getPath().equals(n)) {
|
if (ed.getPath().equals(n)) {
|
||||||
|
mod.setModifier(ed.getIsModifier());
|
||||||
List<org.hl7.fhir.dstu2016may.model.ElementDefinition> children = listChildrenR2B(sd.getSnapshot().getElement(), ed);
|
List<org.hl7.fhir.dstu2016may.model.ElementDefinition> children = listChildrenR2B(sd.getSnapshot().getElement(), ed);
|
||||||
if (children.size() > 0) {
|
if (children.size() > 0) {
|
||||||
for (org.hl7.fhir.dstu2016may.model.ElementDefinition c : children) {
|
for (org.hl7.fhir.dstu2016may.model.ElementDefinition c : children) {
|
||||||
|
|
|
@ -45,14 +45,14 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
|
|
||||||
public class ProfileComparison extends CanonicalResourceComparison<StructureDefinition> {
|
public class ProfileComparison extends CanonicalResourceComparison<StructureDefinition> {
|
||||||
|
|
||||||
private StructuralMatch<ElementDefinition> combined;
|
private StructuralMatch<ElementDefinitionNode> combined;
|
||||||
|
|
||||||
public ProfileComparison(StructureDefinition left, StructureDefinition right) {
|
public ProfileComparison(StructureDefinition left, StructureDefinition right) {
|
||||||
super(left, right);
|
super(left, right);
|
||||||
combined = new StructuralMatch<ElementDefinition>(); // base
|
combined = new StructuralMatch<ElementDefinitionNode>(); // base
|
||||||
}
|
}
|
||||||
|
|
||||||
public StructuralMatch<ElementDefinition> getCombined() {
|
public StructuralMatch<ElementDefinitionNode> getCombined() {
|
||||||
return combined;
|
return combined;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -79,7 +79,21 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private class ElementDefinitionNode {
|
||||||
|
private ElementDefinition def;
|
||||||
|
private StructureDefinition src;
|
||||||
|
private ElementDefinitionNode(StructureDefinition src, ElementDefinition def) {
|
||||||
|
super();
|
||||||
|
this.src = src;
|
||||||
|
this.def = def;
|
||||||
|
}
|
||||||
|
public ElementDefinition getDef() {
|
||||||
|
return def;
|
||||||
|
}
|
||||||
|
public StructureDefinition getSrc() {
|
||||||
|
return src;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private ProfileUtilities utilsLeft;
|
private ProfileUtilities utilsLeft;
|
||||||
private ProfileUtilities utilsRight;
|
private ProfileUtilities utilsRight;
|
||||||
|
@ -127,7 +141,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
if (left.getType().equals(right.getType())) {
|
if (left.getType().equals(right.getType())) {
|
||||||
DefinitionNavigator ln = new DefinitionNavigator(session.getContextLeft(), left);
|
DefinitionNavigator ln = new DefinitionNavigator(session.getContextLeft(), left);
|
||||||
DefinitionNavigator rn = new DefinitionNavigator(session.getContextRight(), right);
|
DefinitionNavigator rn = new DefinitionNavigator(session.getContextRight(), right);
|
||||||
StructuralMatch<ElementDefinition> sm = new StructuralMatch<ElementDefinition>(ln.current(), rn.current());
|
StructuralMatch<ElementDefinitionNode> sm = new StructuralMatch<ElementDefinitionNode>(new ElementDefinitionNode(left, ln.current()), new ElementDefinitionNode(right, rn.current()));
|
||||||
compareElements(res, sm, ln.path(), null, ln, rn);
|
compareElements(res, sm, ln.path(), null, ln, rn);
|
||||||
res.combined = sm;
|
res.combined = sm;
|
||||||
}
|
}
|
||||||
|
@ -147,7 +161,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
throw new DefinitionException("StructureDefinition snapshot is empty ("+name+": "+sd.getName()+")");
|
throw new DefinitionException("StructureDefinition snapshot is empty ("+name+": "+sd.getName()+")");
|
||||||
}
|
}
|
||||||
|
|
||||||
private void compareElements(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, String sliceName, DefinitionNavigator left, DefinitionNavigator right) throws DefinitionException, FHIRFormatError, IOException {
|
private void compareElements(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, String sliceName, DefinitionNavigator left, DefinitionNavigator right) throws DefinitionException, FHIRFormatError, IOException {
|
||||||
assert(path != null);
|
assert(path != null);
|
||||||
assert(left != null);
|
assert(left != null);
|
||||||
assert(right != null);
|
assert(right != null);
|
||||||
|
@ -190,7 +204,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
subset.setExample(left.current().hasExample() ? left.current().getExample() : right.current().getExample());
|
subset.setExample(left.current().hasExample() ? left.current().getExample() : right.current().getExample());
|
||||||
|
|
||||||
if (left.current().getMustSupport() != right.current().getMustSupport()) {
|
if (left.current().getMustSupport() != right.current().getMustSupport()) {
|
||||||
vm(IssueSeverity.ERROR, "Elements differ in definition for mustSupport:\r\n \""+left.current().getMustSupport()+"\"\r\n \""+right.current().getMustSupport()+"\"", path, comp.getMessages(), res.getMessages());
|
vm(IssueSeverity.WARNING, "Elements differ in definition for mustSupport: '"+left.current().getMustSupport()+"' vs '"+right.current().getMustSupport()+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
|
||||||
}
|
}
|
||||||
subset.setMustSupport(left.current().getMustSupport() || right.current().getMustSupport());
|
subset.setMustSupport(left.current().getMustSupport() || right.current().getMustSupport());
|
||||||
|
@ -198,15 +212,20 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
|
|
||||||
|
|
||||||
// compare and intersect
|
// compare and intersect
|
||||||
superset.setMin(unionMin(left.current().getMin(), right.current().getMin()));
|
int leftMin = left.current().getMin();
|
||||||
superset.setMax(unionMax(left.current().getMax(), right.current().getMax()));
|
int rightMin = right.current().getMin();
|
||||||
subset.setMin(intersectMin(left.current().getMin(), right.current().getMin()));
|
int leftMax = "*".equals(left.current().getMax()) ? Integer.MAX_VALUE : Integer.parseInt(left.current().getMax());
|
||||||
subset.setMax(intersectMax(left.current().getMax(), right.current().getMax()));
|
int rightMax = "*".equals(right.current().getMax()) ? Integer.MAX_VALUE : Integer.parseInt(right.current().getMax());
|
||||||
rule(comp, res, subset.getMax().equals("*") || Integer.parseInt(subset.getMax()) >= subset.getMin(), path, "Cardinality Mismatch: "+card(left)+"/"+card(right));
|
|
||||||
|
checkMinMax(comp, res, path, leftMin, rightMin, leftMax, rightMax);
|
||||||
|
superset.setMin(unionMin(leftMin, rightMin));
|
||||||
|
superset.setMax(unionMax(leftMax, rightMax, left.current().getMax(), right.current().getMax()));
|
||||||
|
subset.setMin(intersectMin(leftMin, rightMin));
|
||||||
|
subset.setMax(intersectMax(leftMax, rightMax, left.current().getMax(), right.current().getMax()));
|
||||||
|
|
||||||
superset.getType().addAll(unionTypes(comp, res, path, left.current().getType(), right.current().getType()));
|
superset.getType().addAll(unionTypes(comp, res, path, left.current().getType(), right.current().getType()));
|
||||||
subset.getType().addAll(intersectTypes(comp, res, subset, path, left.current().getType(), right.current().getType()));
|
subset.getType().addAll(intersectTypes(comp, res, subset, path, left.current().getType(), right.current().getType()));
|
||||||
rule(comp, res, !subset.getType().isEmpty() || (!left.current().hasType() && !right.current().hasType()), path, "Type Mismatch:\r\n "+typeCode(left)+"\r\n "+typeCode(right));
|
rule(comp, res, !subset.getType().isEmpty() || (!left.current().hasType() && !right.current().hasType()), path, "Type Mismatch: "+typeCode(left)+" vs "+typeCode(right));
|
||||||
// <fixed[x]><!-- ?? 0..1 * Value must be exactly this --></fixed[x]>
|
// <fixed[x]><!-- ?? 0..1 * Value must be exactly this --></fixed[x]>
|
||||||
// <pattern[x]><!-- ?? 0..1 * Value must have at least these property values --></pattern[x]>
|
// <pattern[x]><!-- ?? 0..1 * Value must have at least these property values --></pattern[x]>
|
||||||
superset.setMaxLengthElement(unionMaxLength(left.current().getMaxLength(), right.current().getMaxLength()));
|
superset.setMaxLengthElement(unionMaxLength(left.current().getMaxLength(), right.current().getMaxLength()));
|
||||||
|
@ -283,7 +302,8 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
// return null;
|
// return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void compareChildren(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, DefinitionNavigator left, DefinitionNavigator right) throws DefinitionException, IOException, FHIRFormatError {
|
|
||||||
|
private void compareChildren(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, DefinitionNavigator left, DefinitionNavigator right) throws DefinitionException, IOException, FHIRFormatError {
|
||||||
List<DefinitionNavigator> lc = left.children();
|
List<DefinitionNavigator> lc = left.children();
|
||||||
List<DefinitionNavigator> rc = right.children();
|
List<DefinitionNavigator> rc = right.children();
|
||||||
// it's possible that one of these profiles walks into a data type and the other doesn't
|
// it's possible that one of these profiles walks into a data type and the other doesn't
|
||||||
|
@ -299,10 +319,10 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
DefinitionNavigator r = findInList(rc, l);
|
DefinitionNavigator r = findInList(rc, l);
|
||||||
if (r == null) {
|
if (r == null) {
|
||||||
comp.getUnion().getSnapshot().getElement().add(l.current().copy());
|
comp.getUnion().getSnapshot().getElement().add(l.current().copy());
|
||||||
res.getChildren().add(new StructuralMatch<ElementDefinition>(l.current(), vmI(IssueSeverity.INFORMATION, "Removed this element", path)));
|
res.getChildren().add(new StructuralMatch<ElementDefinitionNode>(new ElementDefinitionNode(l.getStructure(), l.current()), vmI(IssueSeverity.INFORMATION, "Removed this element", path)));
|
||||||
} else {
|
} else {
|
||||||
matchR.add(r);
|
matchR.add(r);
|
||||||
StructuralMatch<ElementDefinition> sm = new StructuralMatch<ElementDefinition>(l.current(), r.current());
|
StructuralMatch<ElementDefinitionNode> sm = new StructuralMatch<ElementDefinitionNode>(new ElementDefinitionNode(l.getStructure(), l.current()), new ElementDefinitionNode(r.getStructure(), r.current()));
|
||||||
res.getChildren().add(sm);
|
res.getChildren().add(sm);
|
||||||
compareElements(comp, sm, l.path(), null, l, r);
|
compareElements(comp, sm, l.path(), null, l, r);
|
||||||
}
|
}
|
||||||
|
@ -310,7 +330,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
for (DefinitionNavigator r : rc) {
|
for (DefinitionNavigator r : rc) {
|
||||||
if (!matchR.contains(r)) {
|
if (!matchR.contains(r)) {
|
||||||
comp.getUnion().getSnapshot().getElement().add(r.current().copy());
|
comp.getUnion().getSnapshot().getElement().add(r.current().copy());
|
||||||
res.getChildren().add(new StructuralMatch<ElementDefinition>(vmI(IssueSeverity.INFORMATION, "Added this element", path), r.current()));
|
res.getChildren().add(new StructuralMatch<ElementDefinitionNode>(vmI(IssueSeverity.INFORMATION, "Added this element", path), new ElementDefinitionNode(r.getStructure(), r.current())));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -324,7 +344,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ruleEqual(ProfileComparison comp, StructuralMatch<ElementDefinition> res, DataType vLeft, DataType vRight, String name, String path) throws IOException {
|
private void ruleEqual(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, DataType vLeft, DataType vRight, String name, String path) throws IOException {
|
||||||
if (vLeft == null && vRight == null) {
|
if (vLeft == null && vRight == null) {
|
||||||
// nothing
|
// nothing
|
||||||
} else if (vLeft == null) {
|
} else if (vLeft == null) {
|
||||||
|
@ -338,7 +358,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
|
|
||||||
private String toString(DataType val, boolean left) throws IOException {
|
private String toString(DataType val, boolean left) throws IOException {
|
||||||
if (val instanceof PrimitiveType)
|
if (val instanceof PrimitiveType)
|
||||||
return "\"" + ((PrimitiveType) val).getValueAsString()+"\"";
|
return "'" + ((PrimitiveType) val).getValueAsString()+"'";
|
||||||
|
|
||||||
IParser jp = left ? session.getContextLeft().newJsonParser() : session.getContextRight().newJsonParser();
|
IParser jp = left ? session.getContextLeft().newJsonParser() : session.getContextRight().newJsonParser();
|
||||||
return jp.composeString(val, "value");
|
return jp.composeString(val, "value");
|
||||||
|
@ -356,14 +376,14 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean rule(ProfileComparison comp, StructuralMatch<ElementDefinition> res, boolean test, String path, String message) {
|
private boolean rule(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, boolean test, String path, String message) {
|
||||||
if (!test) {
|
if (!test) {
|
||||||
vm(IssueSeverity.ERROR, message, path, comp.getMessages(), res.getMessages());
|
vm(IssueSeverity.ERROR, message, path, comp.getMessages(), res.getMessages());
|
||||||
}
|
}
|
||||||
return test;
|
return test;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String mergeText(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, String name, String left, String right, boolean isError) {
|
private String mergeText(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, String name, String left, String right, boolean isError) {
|
||||||
if (left == null && right == null)
|
if (left == null && right == null)
|
||||||
return null;
|
return null;
|
||||||
if (left == null)
|
if (left == null)
|
||||||
|
@ -375,7 +395,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
if (left.equalsIgnoreCase(right))
|
if (left.equalsIgnoreCase(right))
|
||||||
return left;
|
return left;
|
||||||
if (path != null) {
|
if (path != null) {
|
||||||
vm(isError ? IssueSeverity.ERROR : IssueSeverity.WARNING, "Elements differ in "+name+":\r\n \""+left+"\"\r\n \""+right+"\"", path, comp.getMessages(), res.getMessages());
|
vm(isError ? IssueSeverity.ERROR : IssueSeverity.WARNING, "Elements differ in "+name+": '"+left+"' vs '"+right+"'", path, comp.getMessages(), res.getMessages());
|
||||||
}
|
}
|
||||||
return "left: "+left+"; right: "+right;
|
return "left: "+left+"; right: "+right;
|
||||||
}
|
}
|
||||||
|
@ -429,6 +449,36 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return right;
|
return right;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void checkMinMax(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, int leftMin, int rightMin, int leftMax, int rightMax) {
|
||||||
|
if (leftMin != rightMin) {
|
||||||
|
if (leftMin == 0) {
|
||||||
|
vm(IssueSeverity.INFORMATION, "Element minimum cardinalities differ: '"+leftMin+"' vs '"+rightMin+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
} else if (rightMin == 0) {
|
||||||
|
vm(IssueSeverity.INFORMATION, "Element minimum cardinalities differ: '"+leftMin+"' vs '"+rightMin+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
} else {
|
||||||
|
vm(IssueSeverity.INFORMATION, "Element minimum cardinalities differ: '"+leftMin+"' vs '"+rightMin+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (leftMax != rightMax) {
|
||||||
|
if (leftMax == Integer.MAX_VALUE) {
|
||||||
|
vm(IssueSeverity.INFORMATION, "Element maximum cardinalities differ: '"+leftMax+"' vs '"+rightMax+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
} else if (rightMax == Integer.MAX_VALUE) {
|
||||||
|
vm(IssueSeverity.INFORMATION, "Element maximum cardinalities differ: '"+leftMax+"' vs '"+rightMax+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
} else {
|
||||||
|
vm(IssueSeverity.INFORMATION, "Element maximum cardinalities differ: '"+leftMax+"' vs '"+rightMax+"'", path, comp.getMessages(), res.getMessages());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// rule(comp, res, subset.getMax().equals("*") || Integer.parseInt(subset.getMax()) >= subset.getMin(), path, "Cardinality Mismatch: "+card(left)+"/"+card(right));
|
||||||
|
|
||||||
|
// cross comparison - if max > min in either direction, there can be no instances that are valid against both
|
||||||
|
if (leftMax < rightMin) {
|
||||||
|
vm(IssueSeverity.ERROR, "Element minimum cardinalities conflict: '"+leftMin+".."+leftMax+"' vs '"+rightMin+".."+rightMax+"': No instances can be valid against both profiles", path, comp.getMessages(), res.getMessages());
|
||||||
|
}
|
||||||
|
if (rightMax < leftMin) {
|
||||||
|
vm(IssueSeverity.ERROR, "Element minimum cardinalities conflict: '"+leftMin+".."+leftMax+"' vs '"+rightMin+".."+rightMax+"': No instances can be valid against both profiles", path, comp.getMessages(), res.getMessages());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private int unionMin(int left, int right) {
|
private int unionMin(int left, int right) {
|
||||||
if (left > right)
|
if (left > right)
|
||||||
return right;
|
return right;
|
||||||
|
@ -436,18 +486,14 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return left;
|
return left;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String intersectMax(String left, String right) {
|
private String intersectMax(int l, int r, String left, String right) {
|
||||||
int l = "*".equals(left) ? Integer.MAX_VALUE : Integer.parseInt(left);
|
|
||||||
int r = "*".equals(right) ? Integer.MAX_VALUE : Integer.parseInt(right);
|
|
||||||
if (l < r)
|
if (l < r)
|
||||||
return left;
|
return left;
|
||||||
else
|
else
|
||||||
return right;
|
return right;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String unionMax(String left, String right) {
|
private String unionMax(int l, int r, String left, String right) {
|
||||||
int l = "*".equals(left) ? Integer.MAX_VALUE : Integer.parseInt(left);
|
|
||||||
int r = "*".equals(right) ? Integer.MAX_VALUE : Integer.parseInt(right);
|
|
||||||
if (l < r)
|
if (l < r)
|
||||||
return right;
|
return right;
|
||||||
else
|
else
|
||||||
|
@ -480,7 +526,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return Integer.toString(defn.current().getMin())+".."+defn.current().getMax();
|
return Integer.toString(defn.current().getMin())+".."+defn.current().getMax();
|
||||||
}
|
}
|
||||||
|
|
||||||
private Collection<? extends TypeRefComponent> unionTypes(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, List<TypeRefComponent> left, List<TypeRefComponent> right) throws DefinitionException, IOException, FHIRFormatError {
|
private Collection<? extends TypeRefComponent> unionTypes(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, List<TypeRefComponent> left, List<TypeRefComponent> right) throws DefinitionException, IOException, FHIRFormatError {
|
||||||
List<TypeRefComponent> result = new ArrayList<TypeRefComponent>();
|
List<TypeRefComponent> result = new ArrayList<TypeRefComponent>();
|
||||||
for (TypeRefComponent l : left)
|
for (TypeRefComponent l : left)
|
||||||
checkAddTypeUnion(comp, res, path, result, l, session.getContextLeft());
|
checkAddTypeUnion(comp, res, path, result, l, session.getContextLeft());
|
||||||
|
@ -489,7 +535,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void checkAddTypeUnion(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, List<TypeRefComponent> results, TypeRefComponent nw, IWorkerContext ctxt) throws DefinitionException, IOException, FHIRFormatError {
|
private void checkAddTypeUnion(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, List<TypeRefComponent> results, TypeRefComponent nw, IWorkerContext ctxt) throws DefinitionException, IOException, FHIRFormatError {
|
||||||
boolean pfound = false;
|
boolean pfound = false;
|
||||||
boolean tfound = false;
|
boolean tfound = false;
|
||||||
nw = nw.copy();
|
nw = nw.copy();
|
||||||
|
@ -586,7 +632,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Collection<? extends TypeRefComponent> intersectTypes(ProfileComparison comp, StructuralMatch<ElementDefinition> res, ElementDefinition ed, String path, List<TypeRefComponent> left, List<TypeRefComponent> right) throws DefinitionException, IOException, FHIRFormatError {
|
private Collection<? extends TypeRefComponent> intersectTypes(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, ElementDefinition ed, String path, List<TypeRefComponent> left, List<TypeRefComponent> right) throws DefinitionException, IOException, FHIRFormatError {
|
||||||
List<TypeRefComponent> result = new ArrayList<TypeRefComponent>();
|
List<TypeRefComponent> result = new ArrayList<TypeRefComponent>();
|
||||||
for (TypeRefComponent l : left) {
|
for (TypeRefComponent l : left) {
|
||||||
if (l.hasAggregation())
|
if (l.hasAggregation())
|
||||||
|
@ -665,7 +711,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return b.toString();
|
return b.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean compareBindings(ProfileComparison comp, StructuralMatch<ElementDefinition> res, ElementDefinition subset, ElementDefinition superset, String path, ElementDefinition lDef, ElementDefinition rDef) throws FHIRFormatError, DefinitionException, IOException {
|
private boolean compareBindings(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, ElementDefinition subset, ElementDefinition superset, String path, ElementDefinition lDef, ElementDefinition rDef) throws FHIRFormatError, DefinitionException, IOException {
|
||||||
assert(lDef.hasBinding() || rDef.hasBinding());
|
assert(lDef.hasBinding() || rDef.hasBinding());
|
||||||
if (!lDef.hasBinding()) {
|
if (!lDef.hasBinding()) {
|
||||||
subset.setBinding(rDef.getBinding());
|
subset.setBinding(rDef.getBinding());
|
||||||
|
@ -801,7 +847,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
}
|
}
|
||||||
|
|
||||||
// we can't really know about constraints. We create warnings, and collate them
|
// we can't really know about constraints. We create warnings, and collate them
|
||||||
private List<ElementDefinitionConstraintComponent> unionConstraints(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, List<ElementDefinitionConstraintComponent> left, List<ElementDefinitionConstraintComponent> right) {
|
private List<ElementDefinitionConstraintComponent> unionConstraints(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, List<ElementDefinitionConstraintComponent> left, List<ElementDefinitionConstraintComponent> right) {
|
||||||
List<ElementDefinitionConstraintComponent> result = new ArrayList<ElementDefinitionConstraintComponent>();
|
List<ElementDefinitionConstraintComponent> result = new ArrayList<ElementDefinitionConstraintComponent>();
|
||||||
for (ElementDefinitionConstraintComponent l : left) {
|
for (ElementDefinitionConstraintComponent l : left) {
|
||||||
boolean found = false;
|
boolean found = false;
|
||||||
|
@ -829,7 +875,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
private StructureDefinition resolveProfile(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, String url, String name, IWorkerContext ctxt) {
|
private StructureDefinition resolveProfile(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, String url, String name, IWorkerContext ctxt) {
|
||||||
StructureDefinition sd = ctxt.fetchResource(StructureDefinition.class, url);
|
StructureDefinition sd = ctxt.fetchResource(StructureDefinition.class, url);
|
||||||
if (sd == null) {
|
if (sd == null) {
|
||||||
ValidationMessage vm = vmI(IssueSeverity.WARNING, "Unable to resolve profile "+url+" in profile "+name, path);
|
ValidationMessage vm = vmI(IssueSeverity.WARNING, "Unable to resolve profile "+url+" in profile "+name, path);
|
||||||
|
@ -841,7 +887,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return binding.getStrength() == BindingStrength.EXAMPLE || binding.getStrength() == BindingStrength.PREFERRED;
|
return binding.getStrength() == BindingStrength.EXAMPLE || binding.getStrength() == BindingStrength.PREFERRED;
|
||||||
}
|
}
|
||||||
|
|
||||||
private ElementDefinitionBindingComponent unionBindings(ProfileComparison comp, StructuralMatch<ElementDefinition> res, String path, ElementDefinitionBindingComponent left, ElementDefinitionBindingComponent right) throws FHIRFormatError, DefinitionException, IOException {
|
private ElementDefinitionBindingComponent unionBindings(ProfileComparison comp, StructuralMatch<ElementDefinitionNode> res, String path, ElementDefinitionBindingComponent left, ElementDefinitionBindingComponent right) throws FHIRFormatError, DefinitionException, IOException {
|
||||||
ElementDefinitionBindingComponent union = new ElementDefinitionBindingComponent();
|
ElementDefinitionBindingComponent union = new ElementDefinitionBindingComponent();
|
||||||
if (left.getStrength().compareTo(right.getStrength()) < 0)
|
if (left.getStrength().compareTo(right.getStrength()) < 0)
|
||||||
union.setStrength(left.getStrength());
|
union.setStrength(left.getStrength());
|
||||||
|
@ -881,17 +927,17 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return gen.generate(model, prefix, 0, null);
|
return gen.generate(model, prefix, 0, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void genElementComp(String defPath, HierarchicalTableGenerator gen, List<Row> rows, StructuralMatch<ElementDefinition> combined, String corePath, String prefix, Row slicingRow, boolean root) throws IOException {
|
private void genElementComp(String defPath, HierarchicalTableGenerator gen, List<Row> rows, StructuralMatch<ElementDefinitionNode> combined, String corePath, String prefix, Row slicingRow, boolean root) throws IOException {
|
||||||
Row originalRow = slicingRow;
|
Row originalRow = slicingRow;
|
||||||
Row typesRow = null;
|
Row typesRow = null;
|
||||||
|
|
||||||
List<StructuralMatch<ElementDefinition>> children = combined.getChildren();
|
List<StructuralMatch<ElementDefinitionNode>> children = combined.getChildren();
|
||||||
|
|
||||||
Row row = gen.new Row();
|
Row row = gen.new Row();
|
||||||
rows.add(row);
|
rows.add(row);
|
||||||
String path = combined.either().getPath();
|
String path = combined.either().getDef().getPath();
|
||||||
row.setAnchor(path);
|
row.setAnchor(path);
|
||||||
row.setColor(utilsRight.getRowColor(combined.either(), false));
|
row.setColor(utilsRight.getRowColor(combined.either().getDef(), false));
|
||||||
if (eitherHasSlicing(combined))
|
if (eitherHasSlicing(combined))
|
||||||
row.setLineColor(1);
|
row.setLineColor(1);
|
||||||
else if (eitherHasSliceName(combined))
|
else if (eitherHasSliceName(combined))
|
||||||
|
@ -917,7 +963,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
row.setIcon("icon_choice.gif", HierarchicalTableGenerator.TEXT_ICON_CHOICE);
|
row.setIcon("icon_choice.gif", HierarchicalTableGenerator.TEXT_ICON_CHOICE);
|
||||||
typesRow = row;
|
typesRow = row;
|
||||||
}
|
}
|
||||||
} else if (combined.either().hasContentReference())
|
} else if (combined.either().getDef().hasContentReference())
|
||||||
row.setIcon("icon_reuse.png", HierarchicalTableGenerator.TEXT_ICON_REUSE);
|
row.setIcon("icon_reuse.png", HierarchicalTableGenerator.TEXT_ICON_REUSE);
|
||||||
else if (isPrimitive(combined))
|
else if (isPrimitive(combined))
|
||||||
row.setIcon("icon_primitive.png", HierarchicalTableGenerator.TEXT_ICON_PRIMITIVE);
|
row.setIcon("icon_primitive.png", HierarchicalTableGenerator.TEXT_ICON_PRIMITIVE);
|
||||||
|
@ -927,7 +973,7 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
row.setIcon("icon_datatype.gif", HierarchicalTableGenerator.TEXT_ICON_DATATYPE);
|
row.setIcon("icon_datatype.gif", HierarchicalTableGenerator.TEXT_ICON_DATATYPE);
|
||||||
else
|
else
|
||||||
row.setIcon("icon_resource.png", HierarchicalTableGenerator.TEXT_ICON_RESOURCE);
|
row.setIcon("icon_resource.png", HierarchicalTableGenerator.TEXT_ICON_RESOURCE);
|
||||||
String ref = defPath == null ? null : defPath + combined.either().getId();
|
String ref = defPath == null ? null : defPath + combined.either().getDef().getId();
|
||||||
String sName = tail(path);
|
String sName = tail(path);
|
||||||
String sn = getSliceName(combined);
|
String sn = getSliceName(combined);
|
||||||
if (sn != null)
|
if (sn != null)
|
||||||
|
@ -937,23 +983,23 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
String leftColor = !combined.hasLeft() ? COLOR_NO_ROW_LEFT : combined.hasErrors() ? COLOR_DIFFERENT : null;
|
String leftColor = !combined.hasLeft() ? COLOR_NO_ROW_LEFT : combined.hasErrors() ? COLOR_DIFFERENT : null;
|
||||||
String rightColor = !combined.hasRight() ? COLOR_NO_ROW_LEFT : combined.hasErrors() ? COLOR_DIFFERENT : null;
|
String rightColor = !combined.hasRight() ? COLOR_NO_ROW_LEFT : combined.hasErrors() ? COLOR_DIFFERENT : null;
|
||||||
if (combined.hasLeft()) {
|
if (combined.hasLeft()) {
|
||||||
nc = utilsRight.genElementNameCell(gen, combined.getLeft(), "??", true, corePath, prefix, root, false, false, null, typesRow, row, false, ext, used , ref, sName);
|
nc = utilsRight.genElementNameCell(gen, combined.getLeft().getDef(), "??", true, corePath, prefix, root, false, false, combined.getLeft().getSrc(), typesRow, row, false, ext, used , ref, sName);
|
||||||
} else {
|
} else {
|
||||||
nc = utilsRight.genElementNameCell(gen, combined.getRight(), "??", true, corePath, prefix, root, false, false, null, typesRow, row, false, ext, used , ref, sName);
|
nc = utilsRight.genElementNameCell(gen, combined.getRight().getDef(), "??", true, corePath, prefix, root, false, false, combined.getRight().getSrc(), typesRow, row, false, ext, used , ref, sName);
|
||||||
}
|
}
|
||||||
if (combined.hasLeft()) {
|
if (combined.hasLeft()) {
|
||||||
frame(utilsRight.genElementCells(gen, combined.getLeft(), "??", true, corePath, prefix, root, false, false, null, typesRow, row, false, ext, used , ref, sName, nc, false), leftColor);
|
frame(utilsRight.genElementCells(gen, combined.getLeft().getDef(), "??", true, corePath, prefix, root, false, false, combined.getLeft().getSrc(), typesRow, row, true, ext, used , ref, sName, nc, false, false), leftColor);
|
||||||
} else {
|
} else {
|
||||||
frame(spacers(row, 4, gen), leftColor);
|
frame(spacers(row, 4, gen), leftColor);
|
||||||
}
|
}
|
||||||
if (combined.hasRight()) {
|
if (combined.hasRight()) {
|
||||||
frame(utilsRight.genElementCells(gen, combined.getRight(), "??", true, corePath, prefix, root, false, false, null, typesRow, row, false, ext, used, ref, sName, nc, false), rightColor);
|
frame(utilsRight.genElementCells(gen, combined.getRight().getDef(), "??", true, corePath, prefix, root, false, false, combined.getRight().getSrc(), typesRow, row, true, ext, used, ref, sName, nc, false, false), rightColor);
|
||||||
} else {
|
} else {
|
||||||
frame(spacers(row, 4, gen), rightColor);
|
frame(spacers(row, 4, gen), rightColor);
|
||||||
}
|
}
|
||||||
row.getCells().add(cellForMessages(gen, combined.getMessages()));
|
row.getCells().add(cellForMessages(gen, combined.getMessages()));
|
||||||
|
|
||||||
for (StructuralMatch<ElementDefinition> child : children) {
|
for (StructuralMatch<ElementDefinitionNode> child : children) {
|
||||||
genElementComp(defPath, gen, row.getSubRows(), child, corePath, prefix, originalRow, false);
|
genElementComp(defPath, gen, row.getSubRows(), child, corePath, prefix, originalRow, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -978,47 +1024,47 @@ public class ProfileComparer extends CanonicalResourceComparer {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String getSliceName(StructuralMatch<ElementDefinition> combined) {
|
private String getSliceName(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isDataType(StructuralMatch<ElementDefinition> combined) {
|
private boolean isDataType(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean hasTarget(StructuralMatch<ElementDefinition> combined) {
|
private boolean hasTarget(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isPrimitive(StructuralMatch<ElementDefinition> combined) {
|
private boolean isPrimitive(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean allAreReference(StructuralMatch<ElementDefinition> combined) {
|
private boolean allAreReference(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean hasChoice(StructuralMatch<ElementDefinition> combined) {
|
private boolean hasChoice(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean elementIsComplex(StructuralMatch<ElementDefinition> combined) {
|
private boolean elementIsComplex(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub velement.hasType() && element.getType().get(0).hasProfile() && extensionIsComplex(element.getType().get(0).getProfile().get(0).getValue()
|
// TODO Auto-generated method stub velement.hasType() && element.getType().get(0).hasProfile() && extensionIsComplex(element.getType().get(0).getProfile().get(0).getValue()
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean eitherHasSliceName(StructuralMatch<ElementDefinition> combined) {
|
private boolean eitherHasSliceName(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean eitherHasSlicing(StructuralMatch<ElementDefinition> combined) {
|
private boolean eitherHasSlicing(StructuralMatch<ElementDefinitionNode> combined) {
|
||||||
// TODO Auto-generated method stub
|
// TODO Auto-generated method stub
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -549,8 +549,18 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<ElementDefinition> getChildList(StructureDefinition structure, ElementDefinition element) {
|
public List<ElementDefinition> getChildList(StructureDefinition structure, ElementDefinition element) {
|
||||||
|
if (element.hasContentReference()) {
|
||||||
|
ElementDefinition target = element;
|
||||||
|
for (ElementDefinition t : structure.getSnapshot().getElement()) {
|
||||||
|
if (t.getId().equals(element.getContentReference().substring(1))) {
|
||||||
|
target = t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return getChildList(structure, target.getPath(), target.getId(), false);
|
||||||
|
} else {
|
||||||
return getChildList(structure, element.getPath(), element.getId(), false);
|
return getChildList(structure, element.getPath(), element.getId(), false);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void updateMaps(StructureDefinition base, StructureDefinition derived) throws DefinitionException {
|
public void updateMaps(StructureDefinition base, StructureDefinition derived) throws DefinitionException {
|
||||||
if (base == null)
|
if (base == null)
|
||||||
|
@ -613,6 +623,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
derived.setUserData("profileutils.snapshot.generating", true);
|
derived.setUserData("profileutils.snapshot.generating", true);
|
||||||
snapshotStack.add(derived.getUrl());
|
snapshotStack.add(derived.getUrl());
|
||||||
|
try {
|
||||||
|
|
||||||
if (!Utilities.noString(webUrl) && !webUrl.endsWith("/"))
|
if (!Utilities.noString(webUrl) && !webUrl.endsWith("/"))
|
||||||
webUrl = webUrl + '/';
|
webUrl = webUrl + '/';
|
||||||
|
@ -780,7 +791,10 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
derived.clearUserData("profileutils.snapshot.generating");
|
derived.clearUserData("profileutils.snapshot.generating");
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
derived.clearUserData("profileutils.snapshot.generating");
|
derived.clearUserData("profileutils.snapshot.generating");
|
||||||
|
snapshotStack.remove(derived.getUrl());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void checkDifferentialBaseType(StructureDefinition derived) throws Error {
|
public void checkDifferentialBaseType(StructureDefinition derived) throws Error {
|
||||||
|
@ -1077,6 +1091,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy());
|
ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy());
|
||||||
outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc));
|
outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc));
|
||||||
updateFromBase(outcome, currentBase);
|
updateFromBase(outcome, currentBase);
|
||||||
|
updateConstraintSources(outcome, srcSD.getUrl());
|
||||||
markDerived(outcome);
|
markDerived(outcome);
|
||||||
if (resultPathBase == null)
|
if (resultPathBase == null)
|
||||||
resultPathBase = outcome.getPath();
|
resultPathBase = outcome.getPath();
|
||||||
|
@ -1132,7 +1147,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
processPaths(indent+" ", result, base, differential, nbc, start, nbl-1, diffCursor-1, url, webUrl, profileName, tgt.getElement().getPath(), outcome.getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
processPaths(indent+" ", result, base, differential, nbc, start, nbl-1, diffCursor-1, url, webUrl, profileName, tgt.getElement().getPath(), outcome.getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = outcome.getType().size() > 1 ? context.fetchTypeDefinition("Element") : getProfileForDataType(outcome.getType().get(0));
|
StructureDefinition dt = outcome.getType().size() > 1 ? context.fetchTypeDefinition("Element") : getProfileForDataType(outcome.getType().get(0), webUrl);
|
||||||
if (dt == null) {
|
if (dt == null) {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), cpath));
|
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), cpath));
|
||||||
}
|
}
|
||||||
|
@ -1285,7 +1300,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
processPaths(indent+" ", result, base, differential, nbc, start - 1, nbl-1, diffCursor - 1, url, webUrl, profileName, tgt.getElement().getPath(), diffMatches.get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
processPaths(indent+" ", result, base, differential, nbc, start - 1, nbl-1, diffCursor - 1, url, webUrl, profileName, tgt.getElement().getPath(), diffMatches.get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = outcome.getType().size() == 1 ? getProfileForDataType(outcome.getType().get(0)) : getProfileForDataType("Element");
|
StructureDefinition dt = outcome.getType().size() == 1 ? getProfileForDataType(outcome.getType().get(0), webUrl) : getProfileForDataType("Element");
|
||||||
if (dt == null)
|
if (dt == null)
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__FOR_TYPE__IN_PROFILE__BUT_CANT_FIND_TYPE, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__FOR_TYPE__IN_PROFILE__BUT_CANT_FIND_TYPE, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
||||||
contextName = dt.getUrl();
|
contextName = dt.getUrl();
|
||||||
|
@ -1496,7 +1511,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (baseHasChildren(base, currentBase)) { // not a new type here
|
if (baseHasChildren(base, currentBase)) { // not a new type here
|
||||||
throw new Error("This situation is not yet handled (constrain slicing to 1..1 and fix base slice for inline structure - please report issue to grahame@fhir.org along with a test case that reproduces this error (@ "+cpath+" | "+currentBase.getPath()+")");
|
throw new Error("This situation is not yet handled (constrain slicing to 1..1 and fix base slice for inline structure - please report issue to grahame@fhir.org along with a test case that reproduces this error (@ "+cpath+" | "+currentBase.getPath()+")");
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome);
|
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome, webUrl);
|
||||||
contextName = dt.getUrl();
|
contextName = dt.getUrl();
|
||||||
diffCursor++;
|
diffCursor++;
|
||||||
start = diffCursor;
|
start = diffCursor;
|
||||||
|
@ -1562,7 +1577,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
processPaths(indent+" ", result, base, differential, baseCursor+1, diffCursor, baseLimit, diffLimit, url, webUrl, profileName, contextPathSrc, contextPathDst, trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
processPaths(indent+" ", result, base, differential, baseCursor+1, diffCursor, baseLimit, diffLimit, url, webUrl, profileName, contextPathSrc, contextPathDst, trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
||||||
baseCursor = indexOfFirstNonChild(base, currentBase, baseCursor, baseLimit);
|
baseCursor = indexOfFirstNonChild(base, currentBase, baseCursor, baseLimit);
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome);
|
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome, webUrl);
|
||||||
contextName = dt.getUrl();
|
contextName = dt.getUrl();
|
||||||
int start = diffCursor;
|
int start = diffCursor;
|
||||||
while (differential.getElement().size() > diffCursor && pathStartsWith(differential.getElement().get(diffCursor).getPath(), cpath+"."))
|
while (differential.getElement().size() > diffCursor && pathStartsWith(differential.getElement().get(diffCursor).getPath(), cpath+"."))
|
||||||
|
@ -1763,7 +1778,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (base.getElement().get(baseCursor).getType().size() != 1) {
|
if (base.getElement().get(baseCursor).getType().size() != 1) {
|
||||||
throw new Error(context.formatMessage(I18nConstants.DIFFERENTIAL_WALKS_INTO____BUT_THE_BASE_DOES_NOT_AND_THERE_IS_NOT_A_SINGLE_FIXED_TYPE_THE_TYPE_IS__THIS_IS_NOT_HANDLED_YET, cpath, diffMatches.get(0).toString(), base.getElement().get(baseCursor).typeSummary()));
|
throw new Error(context.formatMessage(I18nConstants.DIFFERENTIAL_WALKS_INTO____BUT_THE_BASE_DOES_NOT_AND_THERE_IS_NOT_A_SINGLE_FIXED_TYPE_THE_TYPE_IS__THIS_IS_NOT_HANDLED_YET, cpath, diffMatches.get(0).toString(), base.getElement().get(baseCursor).typeSummary()));
|
||||||
}
|
}
|
||||||
StructureDefinition dt = getProfileForDataType(base.getElement().get(baseCursor).getType().get(0));
|
StructureDefinition dt = getProfileForDataType(base.getElement().get(baseCursor).getType().get(0), webUrl);
|
||||||
if (dt == null) {
|
if (dt == null) {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
||||||
}
|
}
|
||||||
|
@ -1879,7 +1894,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
diffCursor - 1, url, webUrl, profileName+pathTail(diffMatches, 0), base.getElement().get(0).getPath(), base.getElement().get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
diffCursor - 1, url, webUrl, profileName+pathTail(diffMatches, 0), base.getElement().get(0).getPath(), base.getElement().get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0));
|
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0), webUrl);
|
||||||
// if (t.getCode().equals("Extension") && t.hasProfile() && !t.getProfile().contains(":")) {
|
// if (t.getCode().equals("Extension") && t.hasProfile() && !t.getProfile().contains(":")) {
|
||||||
// lloydfix dt =
|
// lloydfix dt =
|
||||||
// }
|
// }
|
||||||
|
@ -1913,6 +1928,15 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void updateConstraintSources(ElementDefinition ed, String url) {
|
||||||
|
for (ElementDefinitionConstraintComponent c : ed.getConstraint()) {
|
||||||
|
if (!c.hasSource()) {
|
||||||
|
c.setSource(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
private Set<String> getListOfTypes(ElementDefinition e) {
|
private Set<String> getListOfTypes(ElementDefinition e) {
|
||||||
Set<String> result = new HashSet<>();
|
Set<String> result = new HashSet<>();
|
||||||
for (TypeRefComponent t : e.getType()) {
|
for (TypeRefComponent t : e.getType()) {
|
||||||
|
@ -1922,7 +1946,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
|
|
||||||
public StructureDefinition getTypeForElement(StructureDefinitionDifferentialComponent differential, int diffCursor, String profileName,
|
public StructureDefinition getTypeForElement(StructureDefinitionDifferentialComponent differential, int diffCursor, String profileName,
|
||||||
List<ElementDefinition> diffMatches, ElementDefinition outcome) {
|
List<ElementDefinition> diffMatches, ElementDefinition outcome, String webUrl) {
|
||||||
if (outcome.getType().size() == 0) {
|
if (outcome.getType().size() == 0) {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_NO_CHILDREN__AND_NO_TYPES_IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), profileName));
|
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_NO_CHILDREN__AND_NO_TYPES_IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), profileName));
|
||||||
}
|
}
|
||||||
|
@ -1932,7 +1956,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__AND_MULTIPLE_TYPES__IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__AND_MULTIPLE_TYPES__IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0));
|
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0), webUrl);
|
||||||
if (dt == null)
|
if (dt == null)
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
||||||
return dt;
|
return dt;
|
||||||
|
@ -2087,6 +2111,8 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
|
|
||||||
private void removeStatusExtensions(ElementDefinition outcome) {
|
private void removeStatusExtensions(ElementDefinition outcome) {
|
||||||
outcome.removeExtension(ToolingExtensions.EXT_FMM_LEVEL);
|
outcome.removeExtension(ToolingExtensions.EXT_FMM_LEVEL);
|
||||||
|
outcome.removeExtension(ToolingExtensions.EXT_FMM_SUPPORT);
|
||||||
|
outcome.removeExtension(ToolingExtensions.EXT_FMM_DERIVED);
|
||||||
outcome.removeExtension(ToolingExtensions.EXT_STANDARDS_STATUS);
|
outcome.removeExtension(ToolingExtensions.EXT_STANDARDS_STATUS);
|
||||||
outcome.removeExtension(ToolingExtensions.EXT_NORMATIVE_VERSION);
|
outcome.removeExtension(ToolingExtensions.EXT_NORMATIVE_VERSION);
|
||||||
outcome.removeExtension(ToolingExtensions.EXT_WORKGROUP);
|
outcome.removeExtension(ToolingExtensions.EXT_WORKGROUP);
|
||||||
|
@ -2402,10 +2428,16 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
private StructureDefinition getProfileForDataType(TypeRefComponent type) {
|
private StructureDefinition getProfileForDataType(TypeRefComponent type, String webUrl) {
|
||||||
StructureDefinition sd = null;
|
StructureDefinition sd = null;
|
||||||
if (type.hasProfile()) {
|
if (type.hasProfile()) {
|
||||||
sd = context.fetchResource(StructureDefinition.class, type.getProfile().get(0).getValue());
|
sd = context.fetchResource(StructureDefinition.class, type.getProfile().get(0).getValue());
|
||||||
|
if (sd == null) {
|
||||||
|
if (xver != null && xver.matchingUrl(type.getProfile().get(0).getValue()) && xver.status(type.getProfile().get(0).getValue()) == XVerExtensionStatus.Valid) {
|
||||||
|
sd = xver.makeDefinition(type.getProfile().get(0).getValue());
|
||||||
|
generateSnapshot(context.fetchTypeDefinition("Extension"), sd, sd.getUrl(), webUrl, sd.getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
if (sd == null)
|
if (sd == null)
|
||||||
System.out.println("Failed to find referenced profile: " + type.getProfile());
|
System.out.println("Failed to find referenced profile: " + type.getProfile());
|
||||||
}
|
}
|
||||||
|
@ -2475,19 +2507,19 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (webUrl != null) {
|
if (webUrl != null) {
|
||||||
// also, must touch up the markdown
|
// also, must touch up the markdown
|
||||||
if (element.hasDefinition())
|
if (element.hasDefinition())
|
||||||
element.setDefinition(processRelativeUrls(element.getDefinition(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames));
|
element.setDefinition(processRelativeUrls(element.getDefinition(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames, true));
|
||||||
if (element.hasComment())
|
if (element.hasComment())
|
||||||
element.setComment(processRelativeUrls(element.getComment(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames));
|
element.setComment(processRelativeUrls(element.getComment(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames, true));
|
||||||
if (element.hasRequirements())
|
if (element.hasRequirements())
|
||||||
element.setRequirements(processRelativeUrls(element.getRequirements(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames));
|
element.setRequirements(processRelativeUrls(element.getRequirements(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames, true));
|
||||||
if (element.hasMeaningWhenMissing())
|
if (element.hasMeaningWhenMissing())
|
||||||
element.setMeaningWhenMissing(processRelativeUrls(element.getMeaningWhenMissing(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames));
|
element.setMeaningWhenMissing(processRelativeUrls(element.getMeaningWhenMissing(), webUrl, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames, true));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return element;
|
return element;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static String processRelativeUrls(String markdown, String webUrl, String basePath, List<String> resourceNames, Set<String> filenames) {
|
public static String processRelativeUrls(String markdown, String webUrl, String basePath, List<String> resourceNames, Set<String> filenames, boolean processRelatives) {
|
||||||
StringBuilder b = new StringBuilder();
|
StringBuilder b = new StringBuilder();
|
||||||
int i = 0;
|
int i = 0;
|
||||||
while (i < markdown.length()) {
|
while (i < markdown.length()) {
|
||||||
|
@ -2515,7 +2547,14 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
} else {
|
} else {
|
||||||
b.append("](");
|
b.append("](");
|
||||||
// disabled 7-Dec 2021 GDG - we don't want to fool with relative URLs at all?
|
// disabled 7-Dec 2021 GDG - we don't want to fool with relative URLs at all?
|
||||||
// b.append(webUrl);
|
// re-enabled 11-Feb 2022 GDG - we do want to do this. At least, $assemble in davinci-dtr, where the markdown comes from the SDC IG, and an SDC local reference must be changed to point to SDC. in this case, it's called when generating snapshots
|
||||||
|
// added processRelatives parameter to deal with this (well, to try)
|
||||||
|
if (processRelatives && webUrl != null) {
|
||||||
|
// System.out.println("Making "+url+" relative to '"+webUrl+"'");
|
||||||
|
b.append(webUrl);
|
||||||
|
} else {
|
||||||
|
// System.out.println("Not making "+url+" relative to '"+webUrl+"'");
|
||||||
|
}
|
||||||
i = i + 1;
|
i = i + 1;
|
||||||
}
|
}
|
||||||
} else
|
} else
|
||||||
|
@ -2773,7 +2812,9 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
profile = source.getType().size() == 1 && source.getTypeFirstRep().hasProfile() ? context.fetchResource(StructureDefinition.class, source.getTypeFirstRep().getProfile().get(0).getValue()) : null;
|
profile = source.getType().size() == 1 && source.getTypeFirstRep().hasProfile() ? context.fetchResource(StructureDefinition.class, source.getTypeFirstRep().getProfile().get(0).getValue()) : null;
|
||||||
if (profile != null) {
|
if (profile != null) {
|
||||||
ElementDefinition e = profile.getSnapshot().getElement().get(0);
|
ElementDefinition e = profile.getSnapshot().getElement().get(0);
|
||||||
base.setDefinition(e.getDefinition());
|
String webroot = profile.getUserString("webroot");
|
||||||
|
|
||||||
|
base.setDefinition(processRelativeUrls(e.getDefinition(), webroot, baseSpecUrl(), context.getResourceNames(), masterSourceFileNames, true));
|
||||||
base.setShort(e.getShort());
|
base.setShort(e.getShort());
|
||||||
if (e.hasCommentElement())
|
if (e.hasCommentElement())
|
||||||
base.setCommentElement(e.getCommentElement());
|
base.setCommentElement(e.getCommentElement());
|
||||||
|
@ -3502,7 +3543,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
List<TypeRefComponent> types = e.getType();
|
List<TypeRefComponent> types = e.getType();
|
||||||
if (!e.hasType()) {
|
if (!e.hasType()) {
|
||||||
if (root) { // we'll use base instead of types then
|
if (root) { // we'll use base instead of types then
|
||||||
StructureDefinition bsd = context.fetchResource(StructureDefinition.class, profile.getBaseDefinition());
|
StructureDefinition bsd = profile == null ? null : context.fetchResource(StructureDefinition.class, profile.getBaseDefinition());
|
||||||
if (bsd != null) {
|
if (bsd != null) {
|
||||||
if (bsd.hasUserData("path")) {
|
if (bsd.hasUserData("path")) {
|
||||||
c.getPieces().add(gen.new Piece(Utilities.isAbsoluteUrl(bsd.getUserString("path")) ? bsd.getUserString("path") : imagePath +bsd.getUserString("path"), bsd.getName(), null));
|
c.getPieces().add(gen.new Piece(Utilities.isAbsoluteUrl(bsd.getUserString("path")) ? bsd.getUserString("path") : imagePath +bsd.getUserString("path"), bsd.getName(), null));
|
||||||
|
@ -3584,7 +3625,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
c.addPiece(checkForNoChange(tl, gen.new Piece(null,", ", null)));
|
c.addPiece(checkForNoChange(tl, gen.new Piece(null,", ", null)));
|
||||||
}
|
}
|
||||||
|
|
||||||
ref = pkp.getLinkForProfile(profile, p.getValue());
|
ref = pkp == null ? null : pkp.getLinkForProfile(profile, p.getValue());
|
||||||
if (ref != null) {
|
if (ref != null) {
|
||||||
String[] parts = ref.split("\\|");
|
String[] parts = ref.split("\\|");
|
||||||
if (parts[0].startsWith("http:") || parts[0].startsWith("https:")) {
|
if (parts[0].startsWith("http:") || parts[0].startsWith("https:")) {
|
||||||
|
@ -3665,7 +3706,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private String codeForAggregation(AggregationMode a) {
|
public static String codeForAggregation(AggregationMode a) {
|
||||||
switch (a) {
|
switch (a) {
|
||||||
case BUNDLED : return "b";
|
case BUNDLED : return "b";
|
||||||
case CONTAINED : return "c";
|
case CONTAINED : return "c";
|
||||||
|
@ -3674,7 +3715,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private String hintForAggregation(AggregationMode a) {
|
public static String hintForAggregation(AggregationMode a) {
|
||||||
if (a != null)
|
if (a != null)
|
||||||
return a.getDefinition();
|
return a.getDefinition();
|
||||||
else
|
else
|
||||||
|
@ -3683,7 +3724,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
|
|
||||||
|
|
||||||
private String checkPrepend(String corePath, String path) {
|
private String checkPrepend(String corePath, String path) {
|
||||||
if (pkp.prependLinks() && !(path.startsWith("http:") || path.startsWith("https:")))
|
if (pkp != null && pkp.prependLinks() && !(path.startsWith("http:") || path.startsWith("https:")))
|
||||||
return corePath+path;
|
return corePath+path;
|
||||||
else
|
else
|
||||||
return path;
|
return path;
|
||||||
|
@ -4055,7 +4096,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (logicalModel && element.hasRepresentation(PropertyRepresentation.XMLATTR))
|
if (logicalModel && element.hasRepresentation(PropertyRepresentation.XMLATTR))
|
||||||
sName = "@"+sName;
|
sName = "@"+sName;
|
||||||
Cell nc = genElementNameCell(gen, element, profileBaseFileName, snapshot, corePath, imagePath, root, logicalModel, allInvariants, profile, typesRow, row, hasDef, ext, used, ref, sName);
|
Cell nc = genElementNameCell(gen, element, profileBaseFileName, snapshot, corePath, imagePath, root, logicalModel, allInvariants, profile, typesRow, row, hasDef, ext, used, ref, sName);
|
||||||
genElementCells(gen, element, profileBaseFileName, snapshot, corePath, imagePath, root, logicalModel, allInvariants, profile, typesRow, row, hasDef, ext, used, ref, sName, nc, mustSupport);
|
genElementCells(gen, element, profileBaseFileName, snapshot, corePath, imagePath, root, logicalModel, allInvariants, profile, typesRow, row, hasDef, ext, used, ref, sName, nc, mustSupport, true);
|
||||||
if (element.hasSlicing()) {
|
if (element.hasSlicing()) {
|
||||||
if (standardExtensionSlicing(element)) {
|
if (standardExtensionSlicing(element)) {
|
||||||
used.used = true; // doesn't matter whether we have a type, we're used if we're setting up slicing ... element.hasType() && element.getType().get(0).hasProfile();
|
used.used = true; // doesn't matter whether we have a type, we're used if we're setting up slicing ... element.hasType() && element.getType().get(0).hasProfile();
|
||||||
|
@ -4183,7 +4224,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
|
|
||||||
public List<Cell> genElementCells(HierarchicalTableGenerator gen, ElementDefinition element, String profileBaseFileName, boolean snapshot, String corePath,
|
public List<Cell> genElementCells(HierarchicalTableGenerator gen, ElementDefinition element, String profileBaseFileName, boolean snapshot, String corePath,
|
||||||
String imagePath, boolean root, boolean logicalModel, boolean allInvariants, StructureDefinition profile, Row typesRow, Row row, boolean hasDef,
|
String imagePath, boolean root, boolean logicalModel, boolean allInvariants, StructureDefinition profile, Row typesRow, Row row, boolean hasDef,
|
||||||
boolean ext, UnusedTracker used, String ref, String sName, Cell nameCell, boolean mustSupport) throws IOException {
|
boolean ext, UnusedTracker used, String ref, String sName, Cell nameCell, boolean mustSupport, boolean allowSubRows) throws IOException {
|
||||||
List<Cell> res = new ArrayList<>();
|
List<Cell> res = new ArrayList<>();
|
||||||
Cell gc = gen.new Cell();
|
Cell gc = gen.new Cell();
|
||||||
row.getCells().add(gc);
|
row.getCells().add(gc);
|
||||||
|
@ -4209,7 +4250,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (extDefn == null) {
|
if (extDefn == null) {
|
||||||
res.add(genCardinality(gen, element, row, hasDef, used, null));
|
res.add(genCardinality(gen, element, row, hasDef, used, null));
|
||||||
res.add(addCell(row, gen.new Cell(null, null, "?gen-e1? "+element.getType().get(0).getProfile(), null, null)));
|
res.add(addCell(row, gen.new Cell(null, null, "?gen-e1? "+element.getType().get(0).getProfile(), null, null)));
|
||||||
res.add(generateDescription(gen, row, element, (ElementDefinition) element.getUserData(DERIVATION_POINTER), used.used, profile == null ? "" : profile.getUrl(), eurl, profile, corePath, imagePath, root, logicalModel, allInvariants, snapshot, mustSupport));
|
res.add(generateDescription(gen, row, element, (ElementDefinition) element.getUserData(DERIVATION_POINTER), used.used, profile == null ? "" : profile.getUrl(), eurl, profile, corePath, imagePath, root, logicalModel, allInvariants, snapshot, mustSupport, allowSubRows));
|
||||||
} else {
|
} else {
|
||||||
String name = urltail(eurl);
|
String name = urltail(eurl);
|
||||||
nameCell.getPieces().get(0).setText(name);
|
nameCell.getPieces().get(0).setText(name);
|
||||||
|
@ -4222,7 +4263,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
else // if it's complex, we just call it nothing
|
else // if it's complex, we just call it nothing
|
||||||
// genTypes(gen, row, extDefn.getSnapshot().getElement().get(0), profileBaseFileName, profile);
|
// genTypes(gen, row, extDefn.getSnapshot().getElement().get(0), profileBaseFileName, profile);
|
||||||
res.add(addCell(row, gen.new Cell(null, null, "("+translate("sd.table", "Complex")+")", null, null)));
|
res.add(addCell(row, gen.new Cell(null, null, "("+translate("sd.table", "Complex")+")", null, null)));
|
||||||
res.add(generateDescription(gen, row, element, extDefn.getElement(), used.used, null, extDefn.getUrl(), profile, corePath, imagePath, root, logicalModel, allInvariants, valueDefn, snapshot, mustSupport));
|
res.add(generateDescription(gen, row, element, extDefn.getElement(), used.used, null, extDefn.getUrl(), profile, corePath, imagePath, root, logicalModel, allInvariants, valueDefn, snapshot, mustSupport, allowSubRows));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
res.add(genCardinality(gen, element, row, hasDef, used, null));
|
res.add(genCardinality(gen, element, row, hasDef, used, null));
|
||||||
|
@ -4230,7 +4271,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
res.add(addCell(row, gen.new Cell()));
|
res.add(addCell(row, gen.new Cell()));
|
||||||
else
|
else
|
||||||
res.add(genTypes(gen, row, element, profileBaseFileName, profile, corePath, imagePath, root, mustSupport));
|
res.add(genTypes(gen, row, element, profileBaseFileName, profile, corePath, imagePath, root, mustSupport));
|
||||||
res.add(generateDescription(gen, row, element, (ElementDefinition) element.getUserData(DERIVATION_POINTER), used.used, null, null, profile, corePath, imagePath, root, logicalModel, allInvariants, snapshot, mustSupport));
|
res.add(generateDescription(gen, row, element, (ElementDefinition) element.getUserData(DERIVATION_POINTER), used.used, null, null, profile, corePath, imagePath, root, logicalModel, allInvariants, snapshot, mustSupport, allowSubRows));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
res.add(genCardinality(gen, element, row, hasDef, used, null));
|
res.add(genCardinality(gen, element, row, hasDef, used, null));
|
||||||
|
@ -4238,7 +4279,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
res.add(genTypes(gen, row, element, profileBaseFileName, profile, corePath, imagePath, root, mustSupport));
|
res.add(genTypes(gen, row, element, profileBaseFileName, profile, corePath, imagePath, root, mustSupport));
|
||||||
else
|
else
|
||||||
res.add(addCell(row, gen.new Cell()));
|
res.add(addCell(row, gen.new Cell()));
|
||||||
res.add(generateDescription(gen, row, element, (ElementDefinition) element.getUserData(DERIVATION_POINTER), used.used, null, null, profile, corePath, imagePath, root, logicalModel, allInvariants, snapshot, mustSupport));
|
res.add(generateDescription(gen, row, element, (ElementDefinition) element.getUserData(DERIVATION_POINTER), used.used, null, null, profile, corePath, imagePath, root, logicalModel, allInvariants, snapshot, mustSupport, allowSubRows));
|
||||||
}
|
}
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
@ -4289,12 +4330,12 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
|
|
||||||
private boolean isBaseCondition(IdType c) {
|
private boolean isBaseCondition(IdType c) {
|
||||||
String key = c.asStringValue();
|
String key = c.asStringValue();
|
||||||
return key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
return key != null && key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isBaseConstraint(ElementDefinitionConstraintComponent con) {
|
private boolean isBaseConstraint(ElementDefinitionConstraintComponent con) {
|
||||||
String key = con.getKey();
|
String key = con.getKey();
|
||||||
return key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
return key != null && key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
||||||
}
|
}
|
||||||
|
|
||||||
private void makeChoiceRows(List<Row> subRows, ElementDefinition element, HierarchicalTableGenerator gen, String corePath, String profileBaseFileName, boolean mustSupportMode) {
|
private void makeChoiceRows(List<Row> subRows, ElementDefinition element, HierarchicalTableGenerator gen, String corePath, String profileBaseFileName, boolean mustSupportMode) {
|
||||||
|
@ -4544,11 +4585,11 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
&& element.getSlicing().getRules() != SlicingRules.CLOSED && element.getSlicing().getDiscriminator().size() == 1 && element.getSlicing().getDiscriminator().get(0).getPath().equals("url") && element.getSlicing().getDiscriminator().get(0).getType().equals(DiscriminatorType.VALUE);
|
&& element.getSlicing().getRules() != SlicingRules.CLOSED && element.getSlicing().getDiscriminator().size() == 1 && element.getSlicing().getDiscriminator().get(0).getPath().equals("url") && element.getSlicing().getDiscriminator().get(0).getType().equals(DiscriminatorType.VALUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
private Cell generateDescription(HierarchicalTableGenerator gen, Row row, ElementDefinition definition, ElementDefinition fallback, boolean used, String baseURL, String url, StructureDefinition profile, String corePath, String imagePath, boolean root, boolean logicalModel, boolean allInvariants, boolean snapshot, boolean mustSupportOnly) throws IOException, FHIRException {
|
private Cell generateDescription(HierarchicalTableGenerator gen, Row row, ElementDefinition definition, ElementDefinition fallback, boolean used, String baseURL, String url, StructureDefinition profile, String corePath, String imagePath, boolean root, boolean logicalModel, boolean allInvariants, boolean snapshot, boolean mustSupportOnly, boolean allowSubRows) throws IOException, FHIRException {
|
||||||
return generateDescription(gen, row, definition, fallback, used, baseURL, url, profile, corePath, imagePath, root, logicalModel, allInvariants, null, snapshot, mustSupportOnly);
|
return generateDescription(gen, row, definition, fallback, used, baseURL, url, profile, corePath, imagePath, root, logicalModel, allInvariants, null, snapshot, mustSupportOnly, allowSubRows);
|
||||||
}
|
}
|
||||||
|
|
||||||
private Cell generateDescription(HierarchicalTableGenerator gen, Row row, ElementDefinition definition, ElementDefinition fallback, boolean used, String baseURL, String url, StructureDefinition profile, String corePath, String imagePath, boolean root, boolean logicalModel, boolean allInvariants, ElementDefinition valueDefn, boolean snapshot, boolean mustSupportOnly) throws IOException, FHIRException {
|
private Cell generateDescription(HierarchicalTableGenerator gen, Row row, ElementDefinition definition, ElementDefinition fallback, boolean used, String baseURL, String url, StructureDefinition profile, String corePath, String imagePath, boolean root, boolean logicalModel, boolean allInvariants, ElementDefinition valueDefn, boolean snapshot, boolean mustSupportOnly, boolean allowSubRows) throws IOException, FHIRException {
|
||||||
Cell c = gen.new Cell();
|
Cell c = gen.new Cell();
|
||||||
row.getCells().add(c);
|
row.getCells().add(c);
|
||||||
|
|
||||||
|
@ -4564,7 +4605,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (root) {
|
if (root) {
|
||||||
if (profile.getAbstract()) {
|
if (profile != null && profile.getAbstract()) {
|
||||||
if (!c.getPieces().isEmpty()) { c.addPiece(gen.new Piece("br")); }
|
if (!c.getPieces().isEmpty()) { c.addPiece(gen.new Piece("br")); }
|
||||||
c.addPiece(gen.new Piece(null, "This is an abstract profile", null));
|
c.addPiece(gen.new Piece(null, "This is an abstract profile", null));
|
||||||
}
|
}
|
||||||
|
@ -4659,7 +4700,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (binding!=null && !binding.isEmpty()) {
|
if (binding!=null && !binding.isEmpty()) {
|
||||||
if (!c.getPieces().isEmpty())
|
if (!c.getPieces().isEmpty())
|
||||||
c.addPiece(gen.new Piece("br"));
|
c.addPiece(gen.new Piece("br"));
|
||||||
BindingResolution br = pkp.resolveBinding(profile, binding, definition.getPath());
|
BindingResolution br = pkp == null ? makeNullBr(binding) : pkp.resolveBinding(profile, binding, definition.getPath());
|
||||||
c.getPieces().add(checkForNoChange(binding, gen.new Piece(null, translate("sd.table", "Binding")+": ", null).addStyle("font-weight:bold")));
|
c.getPieces().add(checkForNoChange(binding, gen.new Piece(null, translate("sd.table", "Binding")+": ", null).addStyle("font-weight:bold")));
|
||||||
c.getPieces().add(checkForNoChange(binding.getValueSetElement(), gen.new Piece(br.url == null ? null : Utilities.isAbsoluteUrl(br.url) || !pkp.prependLinks() ? br.url : corePath+br.url, br.display, null)));
|
c.getPieces().add(checkForNoChange(binding.getValueSetElement(), gen.new Piece(br.url == null ? null : Utilities.isAbsoluteUrl(br.url) || !pkp.prependLinks() ? br.url : corePath+br.url, br.display, null)));
|
||||||
if (binding.hasStrength()) {
|
if (binding.hasStrength()) {
|
||||||
|
@ -4668,7 +4709,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
c.getPieces().add(checkForNoChange(binding.getStrengthElement(), gen.new Piece(null, ")", null)));
|
c.getPieces().add(checkForNoChange(binding.getStrengthElement(), gen.new Piece(null, ")", null)));
|
||||||
}
|
}
|
||||||
if (binding.hasExtension(ToolingExtensions.EXT_MAX_VALUESET)) {
|
if (binding.hasExtension(ToolingExtensions.EXT_MAX_VALUESET)) {
|
||||||
br = pkp.resolveBinding(profile, ToolingExtensions.readStringExtension(binding, ToolingExtensions.EXT_MAX_VALUESET), definition.getPath());
|
br = pkp == null ? makeNullBr(binding) : pkp.resolveBinding(profile, ToolingExtensions.readStringExtension(binding, ToolingExtensions.EXT_MAX_VALUESET), definition.getPath());
|
||||||
c.addPiece(gen.new Piece("br"));
|
c.addPiece(gen.new Piece("br"));
|
||||||
c.getPieces().add(checkForNoChange(binding, gen.new Piece(corePath+"extension-elementdefinition-maxvalueset.html", translate("sd.table", "Max Binding")+": ", "Max Value Set Extension").addStyle("font-weight:bold")));
|
c.getPieces().add(checkForNoChange(binding, gen.new Piece(corePath+"extension-elementdefinition-maxvalueset.html", translate("sd.table", "Max Binding")+": ", "Max Value Set Extension").addStyle("font-weight:bold")));
|
||||||
c.getPieces().add(checkForNoChange(binding, gen.new Piece(br.url == null ? null : Utilities.isAbsoluteUrl(br.url) || !pkp.prependLinks() ? br.url : corePath+br.url, br.display, null)));
|
c.getPieces().add(checkForNoChange(binding, gen.new Piece(br.url == null ? null : Utilities.isAbsoluteUrl(br.url) || !pkp.prependLinks() ? br.url : corePath+br.url, br.display, null)));
|
||||||
|
@ -4683,6 +4724,13 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
c.getPieces().add(gen.new Piece(null, ": ", null));
|
c.getPieces().add(gen.new Piece(null, ": ", null));
|
||||||
c.addMarkdownNoPara(PublicationHacker.fixBindingDescriptions(context, binding.getDescriptionElement()).asStringValue(), checkForNoChange(PublicationHacker.fixBindingDescriptions(context, binding.getDescriptionElement())));
|
c.addMarkdownNoPara(PublicationHacker.fixBindingDescriptions(context, binding.getDescriptionElement()).asStringValue(), checkForNoChange(PublicationHacker.fixBindingDescriptions(context, binding.getDescriptionElement())));
|
||||||
}
|
}
|
||||||
|
if (binding.hasExtension(ToolingExtensions.EXT_BINDING_ADDITIONAL)) {
|
||||||
|
c.addPiece(gen.new Piece("br"));
|
||||||
|
c.getPieces().add(checkForNoChange(binding, gen.new Piece(null, translate("sd.table", "Additional Bindings")+": ", null).addStyle("font-weight:bold")));
|
||||||
|
for (Extension ext : binding.getExtensionsByUrl(ToolingExtensions.EXT_BINDING_ADDITIONAL)) {
|
||||||
|
renderAdditionalBinding(gen, c, ext);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for (ElementDefinitionConstraintComponent inv : definition.getConstraint()) {
|
for (ElementDefinitionConstraintComponent inv : definition.getConstraint()) {
|
||||||
if (!inv.hasSource() || profile == null || inv.getSource().equals(profile.getUrl()) || allInvariants) {
|
if (!inv.hasSource() || profile == null || inv.getSource().equals(profile.getUrl()) || allInvariants) {
|
||||||
|
@ -4704,7 +4752,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (definition.hasFixed()) {
|
if (definition.hasFixed()) {
|
||||||
if (!c.getPieces().isEmpty()) { c.addPiece(gen.new Piece("br")); }
|
if (!c.getPieces().isEmpty()) { c.addPiece(gen.new Piece("br")); }
|
||||||
c.getPieces().add(checkForNoChange(definition.getFixed(), gen.new Piece(null, translate("sd.table", "Fixed Value")+": ", null).addStyle("font-weight:bold")));
|
c.getPieces().add(checkForNoChange(definition.getFixed(), gen.new Piece(null, translate("sd.table", "Fixed Value")+": ", null).addStyle("font-weight:bold")));
|
||||||
if (!useTableForFixedValues || definition.getFixed().isPrimitive()) {
|
if (!useTableForFixedValues || !allowSubRows || definition.getFixed().isPrimitive()) {
|
||||||
String s = buildJson(definition.getFixed());
|
String s = buildJson(definition.getFixed());
|
||||||
String link = null;
|
String link = null;
|
||||||
if (Utilities.isAbsoluteUrl(s))
|
if (Utilities.isAbsoluteUrl(s))
|
||||||
|
@ -4722,7 +4770,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
} else if (definition.hasPattern()) {
|
} else if (definition.hasPattern()) {
|
||||||
if (!c.getPieces().isEmpty()) { c.addPiece(gen.new Piece("br")); }
|
if (!c.getPieces().isEmpty()) { c.addPiece(gen.new Piece("br")); }
|
||||||
c.getPieces().add(checkForNoChange(definition.getPattern(), gen.new Piece(null, translate("sd.table", "Required Pattern")+": ", null).addStyle("font-weight:bold")));
|
c.getPieces().add(checkForNoChange(definition.getPattern(), gen.new Piece(null, translate("sd.table", "Required Pattern")+": ", null).addStyle("font-weight:bold")));
|
||||||
if (!useTableForFixedValues || definition.getPattern().isPrimitive())
|
if (!useTableForFixedValues || !allowSubRows || definition.getPattern().isPrimitive())
|
||||||
c.getPieces().add(checkForNoChange(definition.getPattern(), gen.new Piece(null, buildJson(definition.getPattern()), null).addStyle("color: darkgreen")));
|
c.getPieces().add(checkForNoChange(definition.getPattern(), gen.new Piece(null, buildJson(definition.getPattern()), null).addStyle("color: darkgreen")));
|
||||||
else {
|
else {
|
||||||
c.getPieces().add(checkForNoChange(definition.getPattern(), gen.new Piece(null, "At least the following", null).addStyle("color: darkgreen")));
|
c.getPieces().add(checkForNoChange(definition.getPattern(), gen.new Piece(null, "At least the following", null).addStyle("color: darkgreen")));
|
||||||
|
@ -4762,6 +4810,30 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
return c;
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void renderAdditionalBinding(HierarchicalTableGenerator gen, Cell c, Extension ext) {
|
||||||
|
// <nsbp>2 <sp> purpose <sp> value-set-link ([context]) {documentation}
|
||||||
|
String purpose = ext.getExtensionString("purpose");
|
||||||
|
String valueSet = ext.getExtensionString("valueSet");
|
||||||
|
String doco = ext.getExtensionString("documentation");
|
||||||
|
//UsageContext usage = (ext.hasExtension("usage")) ? ext.getExtensionByUrl("usage").getValueUsageContext() : null;
|
||||||
|
//
|
||||||
|
// purpose: code - defines how the binding is used
|
||||||
|
// usage : UsageContext - defines the contexts in which this binding is used for it's purpose
|
||||||
|
// valueSet : canonical(ValueSet)
|
||||||
|
// documentation : markdown
|
||||||
|
// !!
|
||||||
|
// c.getPieces().add(checkForNoChange(inv, gen.new Piece(null, inv.getKey()+": ", null).addStyle("font-weight:bold")));
|
||||||
|
// c.getPieces().add(checkForNoChange(inv, gen.new Piece(null, gt(inv.getHumanElement()), null)));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private BindingResolution makeNullBr(ElementDefinitionBindingComponent binding) {
|
||||||
|
BindingResolution br = new BindingResolution();
|
||||||
|
br.url = "http://none.none/none";
|
||||||
|
br.display = "todo";
|
||||||
|
return br;
|
||||||
|
}
|
||||||
|
|
||||||
private ElementDefinitionBindingComponent makeUnifiedBinding(ElementDefinitionBindingComponent binding, ElementDefinition element) {
|
private ElementDefinitionBindingComponent makeUnifiedBinding(ElementDefinitionBindingComponent binding, ElementDefinition element) {
|
||||||
if (!element.hasUserData(DERIVATION_POINTER)) {
|
if (!element.hasUserData(DERIVATION_POINTER)) {
|
||||||
return binding;
|
return binding;
|
||||||
|
|
|
@ -198,6 +198,9 @@ public class ToolingExtensions {
|
||||||
public static final String EXT_TARGET_ID = "http://hl7.org/fhir/StructureDefinition/targetElement";
|
public static final String EXT_TARGET_ID = "http://hl7.org/fhir/StructureDefinition/targetElement";
|
||||||
public static final String EXT_TARGET_PATH = "http://hl7.org/fhir/StructureDefinition/targetPath";
|
public static final String EXT_TARGET_PATH = "http://hl7.org/fhir/StructureDefinition/targetPath";
|
||||||
public static final String EXT_VALUESET_SYSTEM = "http://hl7.org/fhir/StructureDefinition/valueset-system";
|
public static final String EXT_VALUESET_SYSTEM = "http://hl7.org/fhir/StructureDefinition/valueset-system";
|
||||||
|
public static final String EXT_EXPAND_RULES = "http://hl7.org/fhir/StructureDefinition/valueset-expand-rules";
|
||||||
|
public static final String EXT_EXPAND_GROUP = "http://hl7.org/fhir/StructureDefinition/valueset-expand-group";
|
||||||
|
public static final String EXT_BINDING_ADDITIONAL = "http://hl7.org/fhir/tools/StructureDefinition/additional-binding";
|
||||||
|
|
||||||
// specific extension helpers
|
// specific extension helpers
|
||||||
|
|
||||||
|
@ -511,7 +514,7 @@ public class ToolingExtensions {
|
||||||
* @return The extension, if on this element, else null
|
* @return The extension, if on this element, else null
|
||||||
*/
|
*/
|
||||||
public static Extension getExtension(DomainResource resource, String name) {
|
public static Extension getExtension(DomainResource resource, String name) {
|
||||||
if (name == null)
|
if (resource == null || name == null)
|
||||||
return null;
|
return null;
|
||||||
if (!resource.hasExtension())
|
if (!resource.hasExtension())
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -50,9 +50,6 @@ public class XVerExtensionManager {
|
||||||
return XVerExtensionStatus.Invalid;
|
return XVerExtensionStatus.Invalid;
|
||||||
}
|
}
|
||||||
String v = url.substring(20, 23);
|
String v = url.substring(20, 23);
|
||||||
if ("5.0".equals(v)) {
|
|
||||||
v = "4.6"; // for now
|
|
||||||
}
|
|
||||||
String e = url.substring(54);
|
String e = url.substring(54);
|
||||||
if (!lists.containsKey(v)) {
|
if (!lists.containsKey(v)) {
|
||||||
if (context.getBinaries().containsKey("xver-paths-"+v+".json")) {
|
if (context.getBinaries().containsKey("xver-paths-"+v+".json")) {
|
||||||
|
@ -67,6 +64,9 @@ public class XVerExtensionManager {
|
||||||
}
|
}
|
||||||
JsonObject root = lists.get(v);
|
JsonObject root = lists.get(v);
|
||||||
JsonObject path = root.getAsJsonObject(e);
|
JsonObject path = root.getAsJsonObject(e);
|
||||||
|
if (path == null) {
|
||||||
|
path = root.getAsJsonObject(e+"[x]");
|
||||||
|
}
|
||||||
if (path == null) {
|
if (path == null) {
|
||||||
return XVerExtensionStatus.Unknown;
|
return XVerExtensionStatus.Unknown;
|
||||||
}
|
}
|
||||||
|
@ -83,13 +83,13 @@ public class XVerExtensionManager {
|
||||||
|
|
||||||
public StructureDefinition makeDefinition(String url) {
|
public StructureDefinition makeDefinition(String url) {
|
||||||
String verSource = url.substring(20, 23);
|
String verSource = url.substring(20, 23);
|
||||||
if ("5.0".equals(verSource)) {
|
|
||||||
verSource = "4.6"; // for now
|
|
||||||
}
|
|
||||||
String verTarget = VersionUtilities.getMajMin(context.getVersion());
|
String verTarget = VersionUtilities.getMajMin(context.getVersion());
|
||||||
String e = url.substring(54);
|
String e = url.substring(54);
|
||||||
JsonObject root = lists.get(verSource);
|
JsonObject root = lists.get(verSource);
|
||||||
JsonObject path = root.getAsJsonObject(e);
|
JsonObject path = root.getAsJsonObject(e);
|
||||||
|
if (path == null) {
|
||||||
|
path = root.getAsJsonObject(e+"[x]");
|
||||||
|
}
|
||||||
|
|
||||||
StructureDefinition sd = new StructureDefinition();
|
StructureDefinition sd = new StructureDefinition();
|
||||||
sd.setUserData(XVER_EXT_MARKER, "true");
|
sd.setUserData(XVER_EXT_MARKER, "true");
|
||||||
|
@ -117,17 +117,19 @@ public class XVerExtensionManager {
|
||||||
populateTypes(path, val, verSource, verTarget);
|
populateTypes(path, val, verSource, verTarget);
|
||||||
} else if (path.has("elements")) {
|
} else if (path.has("elements")) {
|
||||||
for (JsonElement i : path.getAsJsonArray("elements")) {
|
for (JsonElement i : path.getAsJsonArray("elements")) {
|
||||||
String s = i.getAsString();
|
JsonObject elt = root.getAsJsonObject(e+"."+i.getAsString());
|
||||||
|
if (elt != null) {
|
||||||
|
String s = i.getAsString().replace("[x]", "");
|
||||||
sd.getDifferential().addElement().setPath("Extension.extension").setSliceName(s);
|
sd.getDifferential().addElement().setPath("Extension.extension").setSliceName(s);
|
||||||
sd.getDifferential().addElement().setPath("Extension.extension.extension").setMax("0");
|
sd.getDifferential().addElement().setPath("Extension.extension.extension").setMax("0");
|
||||||
sd.getDifferential().addElement().setPath("Extension.extension.url").setFixed(new UriType(s));
|
sd.getDifferential().addElement().setPath("Extension.extension.url").setFixed(new UriType(s));
|
||||||
ElementDefinition val = sd.getDifferential().addElement().setPath("Extension.extension.value[x]").setMin(1);
|
ElementDefinition val = sd.getDifferential().addElement().setPath("Extension.extension.value[x]").setMin(1);
|
||||||
JsonObject elt = root.getAsJsonObject(e+"."+s);
|
|
||||||
if (!elt.has("types")) {
|
if (!elt.has("types")) {
|
||||||
throw new FHIRException("Internal error - nested elements not supported yet");
|
throw new FHIRException("Internal error - nested elements not supported yet");
|
||||||
}
|
}
|
||||||
populateTypes(elt, val, verSource, verTarget);
|
populateTypes(elt, val, verSource, verTarget);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
sd.getDifferential().addElement().setPath("Extension.url").setFixed(new UriType(url));
|
sd.getDifferential().addElement().setPath("Extension.url").setFixed(new UriType(url));
|
||||||
sd.getDifferential().addElement().setPath("Extension.value[x]").setMax("0");
|
sd.getDifferential().addElement().setPath("Extension.value[x]").setMax("0");
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -4,6 +4,7 @@ import org.hl7.fhir.exceptions.FHIRException;
|
||||||
import org.hl7.fhir.r4b.conformance.ProfileUtilities;
|
import org.hl7.fhir.r4b.conformance.ProfileUtilities;
|
||||||
import org.hl7.fhir.r4b.model.Base;
|
import org.hl7.fhir.r4b.model.Base;
|
||||||
import org.hl7.fhir.r4b.model.ElementDefinition;
|
import org.hl7.fhir.r4b.model.ElementDefinition;
|
||||||
|
import org.hl7.fhir.r4b.model.ElementDefinition.ElementDefinitionConstraintComponent;
|
||||||
import org.hl7.fhir.r4b.model.IntegerType;
|
import org.hl7.fhir.r4b.model.IntegerType;
|
||||||
import org.hl7.fhir.r4b.model.StructureDefinition;
|
import org.hl7.fhir.r4b.model.StructureDefinition;
|
||||||
import org.hl7.fhir.r4b.model.StructureDefinition.TypeDerivationRule;
|
import org.hl7.fhir.r4b.model.StructureDefinition.TypeDerivationRule;
|
||||||
|
@ -51,6 +52,12 @@ public class ProfileUtilitiesTests {
|
||||||
f.setBase(null);
|
f.setBase(null);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
ok = Base.compareDeep(b, f, true);
|
ok = Base.compareDeep(b, f, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -83,6 +90,12 @@ public class ProfileUtilitiesTests {
|
||||||
if (ok) {
|
if (ok) {
|
||||||
ElementDefinition b = base.getSnapshot().getElement().get(i);
|
ElementDefinition b = base.getSnapshot().getElement().get(i);
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath()))
|
if (!f.hasBase() || !b.getPath().equals(f.getPath()))
|
||||||
ok = false;
|
ok = false;
|
||||||
else {
|
else {
|
||||||
|
@ -128,6 +141,12 @@ public class ProfileUtilitiesTests {
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
@ -174,6 +193,12 @@ public class ProfileUtilitiesTests {
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
@ -221,6 +246,12 @@ public class ProfileUtilitiesTests {
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -49,6 +49,7 @@ import org.hl7.fhir.utilities.npm.ToolsVersion;
|
||||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||||
import org.hl7.fhir.utilities.validation.ValidationMessage.IssueSeverity;
|
import org.hl7.fhir.utilities.validation.ValidationMessage.IssueSeverity;
|
||||||
import org.hl7.fhir.utilities.xml.XMLUtil;
|
import org.hl7.fhir.utilities.xml.XMLUtil;
|
||||||
|
import org.junit.Assert;
|
||||||
import org.junit.jupiter.api.Assertions;
|
import org.junit.jupiter.api.Assertions;
|
||||||
import org.junit.jupiter.api.BeforeAll;
|
import org.junit.jupiter.api.BeforeAll;
|
||||||
import org.junit.jupiter.params.ParameterizedTest;
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
|
@ -137,6 +138,7 @@ public class SnapShotGenerationTests {
|
||||||
private boolean fail;
|
private boolean fail;
|
||||||
private boolean newSliceProcessing;
|
private boolean newSliceProcessing;
|
||||||
private boolean debug;
|
private boolean debug;
|
||||||
|
private boolean noR4b;
|
||||||
|
|
||||||
private List<Rule> rules = new ArrayList<>();
|
private List<Rule> rules = new ArrayList<>();
|
||||||
private StructureDefinition source;
|
private StructureDefinition source;
|
||||||
|
@ -161,6 +163,7 @@ public class SnapShotGenerationTests {
|
||||||
rules.add(new Rule(rule));
|
rules.add(new Rule(rule));
|
||||||
rule = XMLUtil.getNextSibling(rule);
|
rule = XMLUtil.getNextSibling(rule);
|
||||||
}
|
}
|
||||||
|
noR4b = test.hasAttribute("r4b") && "false".equals(test.getAttribute("r4b"));
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getId() {
|
public String getId() {
|
||||||
|
@ -466,6 +469,9 @@ public class SnapShotGenerationTests {
|
||||||
fp.setHostServices(context);
|
fp.setHostServices(context);
|
||||||
messages = new ArrayList<ValidationMessage>();
|
messages = new ArrayList<ValidationMessage>();
|
||||||
|
|
||||||
|
if (test.noR4b) {
|
||||||
|
Assert.assertTrue(true);
|
||||||
|
} else {
|
||||||
System.out.println("---- "+id+" -----------------------------------------");
|
System.out.println("---- "+id+" -----------------------------------------");
|
||||||
if (test.isFail()) {
|
if (test.isFail()) {
|
||||||
boolean failed = true;
|
boolean failed = true;
|
||||||
|
@ -495,6 +501,7 @@ public class SnapShotGenerationTests {
|
||||||
Assertions.assertTrue(ok, r.description);
|
Assertions.assertTrue(ok, r.description);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void testSort(TestDetails test, SnapShotGenerationTestsContext context) throws DefinitionException, FHIRException, IOException {
|
private void testSort(TestDetails test, SnapShotGenerationTestsContext context) throws DefinitionException, FHIRException, IOException {
|
||||||
StructureDefinition base = getSD(test.getSource().getBaseDefinition(), context);
|
StructureDefinition base = getSD(test.getSource().getBaseDefinition(), context);
|
||||||
|
|
|
@ -627,6 +627,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
derived.setUserData("profileutils.snapshot.generating", true);
|
derived.setUserData("profileutils.snapshot.generating", true);
|
||||||
snapshotStack.add(derived.getUrl());
|
snapshotStack.add(derived.getUrl());
|
||||||
|
try {
|
||||||
|
|
||||||
if (!Utilities.noString(webUrl) && !webUrl.endsWith("/"))
|
if (!Utilities.noString(webUrl) && !webUrl.endsWith("/"))
|
||||||
webUrl = webUrl + '/';
|
webUrl = webUrl + '/';
|
||||||
|
@ -794,7 +795,10 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
derived.clearUserData("profileutils.snapshot.generating");
|
derived.clearUserData("profileutils.snapshot.generating");
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
derived.clearUserData("profileutils.snapshot.generating");
|
derived.clearUserData("profileutils.snapshot.generating");
|
||||||
|
snapshotStack.remove(derived.getUrl());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void checkDifferentialBaseType(StructureDefinition derived) throws Error {
|
public void checkDifferentialBaseType(StructureDefinition derived) throws Error {
|
||||||
|
@ -1091,6 +1095,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy());
|
ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy());
|
||||||
outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc));
|
outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc));
|
||||||
updateFromBase(outcome, currentBase);
|
updateFromBase(outcome, currentBase);
|
||||||
|
updateConstraintSources(outcome, srcSD.getUrl());
|
||||||
markDerived(outcome);
|
markDerived(outcome);
|
||||||
if (resultPathBase == null)
|
if (resultPathBase == null)
|
||||||
resultPathBase = outcome.getPath();
|
resultPathBase = outcome.getPath();
|
||||||
|
@ -1146,7 +1151,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
processPaths(indent+" ", result, base, differential, nbc, start, nbl-1, diffCursor-1, url, webUrl, profileName, tgt.getElement().getPath(), outcome.getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
processPaths(indent+" ", result, base, differential, nbc, start, nbl-1, diffCursor-1, url, webUrl, profileName, tgt.getElement().getPath(), outcome.getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = outcome.getType().size() > 1 ? context.fetchTypeDefinition("Element") : getProfileForDataType(outcome.getType().get(0));
|
StructureDefinition dt = outcome.getType().size() > 1 ? context.fetchTypeDefinition("Element") : getProfileForDataType(outcome.getType().get(0), webUrl);
|
||||||
if (dt == null) {
|
if (dt == null) {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), cpath));
|
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), cpath));
|
||||||
}
|
}
|
||||||
|
@ -1299,7 +1304,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
processPaths(indent+" ", result, base, differential, nbc, start - 1, nbl-1, diffCursor - 1, url, webUrl, profileName, tgt.getElement().getPath(), diffMatches.get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
processPaths(indent+" ", result, base, differential, nbc, start - 1, nbl-1, diffCursor - 1, url, webUrl, profileName, tgt.getElement().getPath(), diffMatches.get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirectorStack(redirector, outcome, cpath), srcSD);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = outcome.getType().size() == 1 ? getProfileForDataType(outcome.getType().get(0)) : getProfileForDataType("Element");
|
StructureDefinition dt = outcome.getType().size() == 1 ? getProfileForDataType(outcome.getType().get(0), webUrl) : getProfileForDataType("Element");
|
||||||
if (dt == null)
|
if (dt == null)
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__FOR_TYPE__IN_PROFILE__BUT_CANT_FIND_TYPE, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__FOR_TYPE__IN_PROFILE__BUT_CANT_FIND_TYPE, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
||||||
contextName = dt.getUrl();
|
contextName = dt.getUrl();
|
||||||
|
@ -1510,7 +1515,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (baseHasChildren(base, currentBase)) { // not a new type here
|
if (baseHasChildren(base, currentBase)) { // not a new type here
|
||||||
throw new Error("This situation is not yet handled (constrain slicing to 1..1 and fix base slice for inline structure - please report issue to grahame@fhir.org along with a test case that reproduces this error (@ "+cpath+" | "+currentBase.getPath()+")");
|
throw new Error("This situation is not yet handled (constrain slicing to 1..1 and fix base slice for inline structure - please report issue to grahame@fhir.org along with a test case that reproduces this error (@ "+cpath+" | "+currentBase.getPath()+")");
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome);
|
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome, webUrl);
|
||||||
contextName = dt.getUrl();
|
contextName = dt.getUrl();
|
||||||
diffCursor++;
|
diffCursor++;
|
||||||
start = diffCursor;
|
start = diffCursor;
|
||||||
|
@ -1576,7 +1581,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
processPaths(indent+" ", result, base, differential, baseCursor+1, diffCursor, baseLimit, diffLimit, url, webUrl, profileName, contextPathSrc, contextPathDst, trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
processPaths(indent+" ", result, base, differential, baseCursor+1, diffCursor, baseLimit, diffLimit, url, webUrl, profileName, contextPathSrc, contextPathDst, trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
||||||
baseCursor = indexOfFirstNonChild(base, currentBase, baseCursor, baseLimit);
|
baseCursor = indexOfFirstNonChild(base, currentBase, baseCursor, baseLimit);
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome);
|
StructureDefinition dt = getTypeForElement(differential, diffCursor, profileName, diffMatches, outcome, webUrl);
|
||||||
contextName = dt.getUrl();
|
contextName = dt.getUrl();
|
||||||
int start = diffCursor;
|
int start = diffCursor;
|
||||||
while (differential.getElement().size() > diffCursor && pathStartsWith(differential.getElement().get(diffCursor).getPath(), cpath+"."))
|
while (differential.getElement().size() > diffCursor && pathStartsWith(differential.getElement().get(diffCursor).getPath(), cpath+"."))
|
||||||
|
@ -1777,7 +1782,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
if (base.getElement().get(baseCursor).getType().size() != 1) {
|
if (base.getElement().get(baseCursor).getType().size() != 1) {
|
||||||
throw new Error(context.formatMessage(I18nConstants.DIFFERENTIAL_WALKS_INTO____BUT_THE_BASE_DOES_NOT_AND_THERE_IS_NOT_A_SINGLE_FIXED_TYPE_THE_TYPE_IS__THIS_IS_NOT_HANDLED_YET, cpath, diffMatches.get(0).toString(), base.getElement().get(baseCursor).typeSummary()));
|
throw new Error(context.formatMessage(I18nConstants.DIFFERENTIAL_WALKS_INTO____BUT_THE_BASE_DOES_NOT_AND_THERE_IS_NOT_A_SINGLE_FIXED_TYPE_THE_TYPE_IS__THIS_IS_NOT_HANDLED_YET, cpath, diffMatches.get(0).toString(), base.getElement().get(baseCursor).typeSummary()));
|
||||||
}
|
}
|
||||||
StructureDefinition dt = getProfileForDataType(base.getElement().get(baseCursor).getType().get(0));
|
StructureDefinition dt = getProfileForDataType(base.getElement().get(baseCursor).getType().get(0), webUrl);
|
||||||
if (dt == null) {
|
if (dt == null) {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
||||||
}
|
}
|
||||||
|
@ -1893,7 +1898,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
diffCursor - 1, url, webUrl, profileName+pathTail(diffMatches, 0), base.getElement().get(0).getPath(), base.getElement().get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
diffCursor - 1, url, webUrl, profileName+pathTail(diffMatches, 0), base.getElement().get(0).getPath(), base.getElement().get(0).getPath(), trimDifferential, contextName, resultPathBase, false, null, null, redirector, srcSD);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0));
|
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0), webUrl);
|
||||||
// if (t.getCode().equals("Extension") && t.hasProfile() && !t.getProfile().contains(":")) {
|
// if (t.getCode().equals("Extension") && t.hasProfile() && !t.getProfile().contains(":")) {
|
||||||
// lloydfix dt =
|
// lloydfix dt =
|
||||||
// }
|
// }
|
||||||
|
@ -1927,6 +1932,15 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void updateConstraintSources(ElementDefinition ed, String url) {
|
||||||
|
for (ElementDefinitionConstraintComponent c : ed.getConstraint()) {
|
||||||
|
if (!c.hasSource()) {
|
||||||
|
c.setSource(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
private Set<String> getListOfTypes(ElementDefinition e) {
|
private Set<String> getListOfTypes(ElementDefinition e) {
|
||||||
Set<String> result = new HashSet<>();
|
Set<String> result = new HashSet<>();
|
||||||
for (TypeRefComponent t : e.getType()) {
|
for (TypeRefComponent t : e.getType()) {
|
||||||
|
@ -1936,7 +1950,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
}
|
}
|
||||||
|
|
||||||
public StructureDefinition getTypeForElement(StructureDefinitionDifferentialComponent differential, int diffCursor, String profileName,
|
public StructureDefinition getTypeForElement(StructureDefinitionDifferentialComponent differential, int diffCursor, String profileName,
|
||||||
List<ElementDefinition> diffMatches, ElementDefinition outcome) {
|
List<ElementDefinition> diffMatches, ElementDefinition outcome, String webUrl) {
|
||||||
if (outcome.getType().size() == 0) {
|
if (outcome.getType().size() == 0) {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_NO_CHILDREN__AND_NO_TYPES_IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), profileName));
|
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_NO_CHILDREN__AND_NO_TYPES_IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), profileName));
|
||||||
}
|
}
|
||||||
|
@ -1946,7 +1960,7 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__AND_MULTIPLE_TYPES__IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
throw new DefinitionException(context.formatMessage(I18nConstants._HAS_CHILDREN__AND_MULTIPLE_TYPES__IN_PROFILE_, diffMatches.get(0).getPath(), differential.getElement().get(diffCursor).getPath(), typeCode(outcome.getType()), profileName));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0));
|
StructureDefinition dt = getProfileForDataType(outcome.getType().get(0), webUrl);
|
||||||
if (dt == null)
|
if (dt == null)
|
||||||
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
throw new DefinitionException(context.formatMessage(I18nConstants.UNKNOWN_TYPE__AT_, outcome.getType().get(0), diffMatches.get(0).getPath()));
|
||||||
return dt;
|
return dt;
|
||||||
|
@ -2418,10 +2432,16 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
private StructureDefinition getProfileForDataType(TypeRefComponent type) {
|
private StructureDefinition getProfileForDataType(TypeRefComponent type, String webUrl) {
|
||||||
StructureDefinition sd = null;
|
StructureDefinition sd = null;
|
||||||
if (type.hasProfile()) {
|
if (type.hasProfile()) {
|
||||||
sd = context.fetchResource(StructureDefinition.class, type.getProfile().get(0).getValue());
|
sd = context.fetchResource(StructureDefinition.class, type.getProfile().get(0).getValue());
|
||||||
|
if (sd == null) {
|
||||||
|
if (xver != null && xver.matchingUrl(type.getProfile().get(0).getValue()) && xver.status(type.getProfile().get(0).getValue()) == XVerExtensionStatus.Valid) {
|
||||||
|
sd = xver.makeDefinition(type.getProfile().get(0).getValue());
|
||||||
|
generateSnapshot(context.fetchTypeDefinition("Extension"), sd, sd.getUrl(), webUrl, sd.getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
if (sd == null)
|
if (sd == null)
|
||||||
System.out.println("Failed to find referenced profile: " + type.getProfile());
|
System.out.println("Failed to find referenced profile: " + type.getProfile());
|
||||||
}
|
}
|
||||||
|
@ -4314,12 +4334,12 @@ public class ProfileUtilities extends TranslatingUtilities {
|
||||||
|
|
||||||
private boolean isBaseCondition(IdType c) {
|
private boolean isBaseCondition(IdType c) {
|
||||||
String key = c.asStringValue();
|
String key = c.asStringValue();
|
||||||
return key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
return key != null && key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isBaseConstraint(ElementDefinitionConstraintComponent con) {
|
private boolean isBaseConstraint(ElementDefinitionConstraintComponent con) {
|
||||||
String key = con.getKey();
|
String key = con.getKey();
|
||||||
return key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
return key != null && key.startsWith("ele-") || key.startsWith("res-") || key.startsWith("ext-") || key.startsWith("dom-") || key.startsWith("dr-");
|
||||||
}
|
}
|
||||||
|
|
||||||
private void makeChoiceRows(List<Row> subRows, ElementDefinition element, HierarchicalTableGenerator gen, String corePath, String profileBaseFileName, boolean mustSupportMode) {
|
private void makeChoiceRows(List<Row> subRows, ElementDefinition element, HierarchicalTableGenerator gen, String corePath, String profileBaseFileName, boolean mustSupportMode) {
|
||||||
|
|
|
@ -50,9 +50,6 @@ public class XVerExtensionManager {
|
||||||
return XVerExtensionStatus.Invalid;
|
return XVerExtensionStatus.Invalid;
|
||||||
}
|
}
|
||||||
String v = url.substring(20, 23);
|
String v = url.substring(20, 23);
|
||||||
if ("5.0".equals(v)) {
|
|
||||||
v = "4.6"; // for now
|
|
||||||
}
|
|
||||||
String e = url.substring(54);
|
String e = url.substring(54);
|
||||||
if (!lists.containsKey(v)) {
|
if (!lists.containsKey(v)) {
|
||||||
if (context.getBinaries().containsKey("xver-paths-"+v+".json")) {
|
if (context.getBinaries().containsKey("xver-paths-"+v+".json")) {
|
||||||
|
@ -86,9 +83,6 @@ public class XVerExtensionManager {
|
||||||
|
|
||||||
public StructureDefinition makeDefinition(String url) {
|
public StructureDefinition makeDefinition(String url) {
|
||||||
String verSource = url.substring(20, 23);
|
String verSource = url.substring(20, 23);
|
||||||
if ("5.0".equals(verSource)) {
|
|
||||||
verSource = "4.6"; // for now
|
|
||||||
}
|
|
||||||
String verTarget = VersionUtilities.getMajMin(context.getVersion());
|
String verTarget = VersionUtilities.getMajMin(context.getVersion());
|
||||||
String e = url.substring(54);
|
String e = url.substring(54);
|
||||||
JsonObject root = lists.get(verSource);
|
JsonObject root = lists.get(verSource);
|
||||||
|
@ -123,17 +117,19 @@ public class XVerExtensionManager {
|
||||||
populateTypes(path, val, verSource, verTarget);
|
populateTypes(path, val, verSource, verTarget);
|
||||||
} else if (path.has("elements")) {
|
} else if (path.has("elements")) {
|
||||||
for (JsonElement i : path.getAsJsonArray("elements")) {
|
for (JsonElement i : path.getAsJsonArray("elements")) {
|
||||||
|
JsonObject elt = root.getAsJsonObject(e+"."+i.getAsString());
|
||||||
|
if (elt != null) {
|
||||||
String s = i.getAsString().replace("[x]", "");
|
String s = i.getAsString().replace("[x]", "");
|
||||||
sd.getDifferential().addElement().setPath("Extension.extension").setSliceName(s);
|
sd.getDifferential().addElement().setPath("Extension.extension").setSliceName(s);
|
||||||
sd.getDifferential().addElement().setPath("Extension.extension.extension").setMax("0");
|
sd.getDifferential().addElement().setPath("Extension.extension.extension").setMax("0");
|
||||||
sd.getDifferential().addElement().setPath("Extension.extension.url").setFixed(new UriType(s));
|
sd.getDifferential().addElement().setPath("Extension.extension.url").setFixed(new UriType(s));
|
||||||
ElementDefinition val = sd.getDifferential().addElement().setPath("Extension.extension.value[x]").setMin(1);
|
ElementDefinition val = sd.getDifferential().addElement().setPath("Extension.extension.value[x]").setMin(1);
|
||||||
JsonObject elt = root.getAsJsonObject(e+"."+i.getAsString());
|
|
||||||
if (!elt.has("types")) {
|
if (!elt.has("types")) {
|
||||||
throw new FHIRException("Internal error - nested elements not supported yet");
|
throw new FHIRException("Internal error - nested elements not supported yet");
|
||||||
}
|
}
|
||||||
populateTypes(elt, val, verSource, verTarget);
|
populateTypes(elt, val, verSource, verTarget);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
sd.getDifferential().addElement().setPath("Extension.url").setFixed(new UriType(url));
|
sd.getDifferential().addElement().setPath("Extension.url").setFixed(new UriType(url));
|
||||||
sd.getDifferential().addElement().setPath("Extension.value[x]").setMax("0");
|
sd.getDifferential().addElement().setPath("Extension.value[x]").setMax("0");
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -7,6 +7,7 @@ import org.hl7.fhir.exceptions.FHIRException;
|
||||||
import org.hl7.fhir.r5.conformance.ProfileUtilities;
|
import org.hl7.fhir.r5.conformance.ProfileUtilities;
|
||||||
import org.hl7.fhir.r5.model.Base;
|
import org.hl7.fhir.r5.model.Base;
|
||||||
import org.hl7.fhir.r5.model.ElementDefinition;
|
import org.hl7.fhir.r5.model.ElementDefinition;
|
||||||
|
import org.hl7.fhir.r5.model.ElementDefinition.ElementDefinitionConstraintComponent;
|
||||||
import org.hl7.fhir.r5.model.IntegerType;
|
import org.hl7.fhir.r5.model.IntegerType;
|
||||||
import org.hl7.fhir.r5.model.StructureDefinition;
|
import org.hl7.fhir.r5.model.StructureDefinition;
|
||||||
import org.hl7.fhir.r5.model.StructureDefinition.TypeDerivationRule;
|
import org.hl7.fhir.r5.model.StructureDefinition.TypeDerivationRule;
|
||||||
|
@ -51,6 +52,12 @@ public class ProfileUtilitiesTests {
|
||||||
f.setBase(null);
|
f.setBase(null);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
ok = Base.compareDeep(b, f, true);
|
ok = Base.compareDeep(b, f, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -83,6 +90,12 @@ public class ProfileUtilitiesTests {
|
||||||
if (ok) {
|
if (ok) {
|
||||||
ElementDefinition b = base.getSnapshot().getElement().get(i);
|
ElementDefinition b = base.getSnapshot().getElement().get(i);
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath()))
|
if (!f.hasBase() || !b.getPath().equals(f.getPath()))
|
||||||
ok = false;
|
ok = false;
|
||||||
else {
|
else {
|
||||||
|
@ -128,6 +141,12 @@ public class ProfileUtilitiesTests {
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
@ -174,6 +193,12 @@ public class ProfileUtilitiesTests {
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
@ -221,6 +246,12 @@ public class ProfileUtilitiesTests {
|
||||||
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
ElementDefinition f = focus.getSnapshot().getElement().get(i);
|
||||||
b.setRequirements(null);
|
b.setRequirements(null);
|
||||||
f.setRequirements(null);
|
f.setRequirements(null);
|
||||||
|
for (ElementDefinitionConstraintComponent c : b.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
|
for (ElementDefinitionConstraintComponent c : f.getConstraint()) {
|
||||||
|
c.setSource(null);
|
||||||
|
}
|
||||||
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
if (!f.hasBase() || !b.getPath().equals(f.getPath())) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -366,6 +366,8 @@ public class I18nConstants {
|
||||||
public static final String SD_ED_TYPE_PROFILE_UNKNOWN = "SD_ED_TYPE_PROFILE_UNKNOWN";
|
public static final String SD_ED_TYPE_PROFILE_UNKNOWN = "SD_ED_TYPE_PROFILE_UNKNOWN";
|
||||||
public static final String SD_ED_TYPE_PROFILE_NOTYPE = "SD_ED_TYPE_PROFILE_NOTYPE";
|
public static final String SD_ED_TYPE_PROFILE_NOTYPE = "SD_ED_TYPE_PROFILE_NOTYPE";
|
||||||
public static final String SD_ED_TYPE_PROFILE_WRONG = "SD_ED_TYPE_PROFILE_WRONG";
|
public static final String SD_ED_TYPE_PROFILE_WRONG = "SD_ED_TYPE_PROFILE_WRONG";
|
||||||
|
public static final String SD_ED_TYPE_PROFILE_IS_MODIFIER = "SD_ED_TYPE_PROFILE_IS_MODIFIER";
|
||||||
|
public static final String SD_ED_TYPE_PROFILE_NOT_MODIFIER = "SD_ED_TYPE_PROFILE_NOT_MODIFIER";
|
||||||
public static final String SD_ED_TYPE_PROFILE_WRONG_TARGET = "SD_ED_TYPE_PROFILE_WRONG_TARGET";
|
public static final String SD_ED_TYPE_PROFILE_WRONG_TARGET = "SD_ED_TYPE_PROFILE_WRONG_TARGET";
|
||||||
public static final String SD_ED_TYPE_NO_TARGET_PROFILE = "SD_ED_TYPE_NO_TARGET_PROFILE";
|
public static final String SD_ED_TYPE_NO_TARGET_PROFILE = "SD_ED_TYPE_NO_TARGET_PROFILE";
|
||||||
public static final String SD_ED_SHOULD_BIND = "SD_ED_SHOULD_BIND";
|
public static final String SD_ED_SHOULD_BIND = "SD_ED_SHOULD_BIND";
|
||||||
|
|
|
@ -3,7 +3,7 @@ package org.hl7.fhir.utilities.npm;
|
||||||
public class CommonPackages {
|
public class CommonPackages {
|
||||||
|
|
||||||
public static final String ID_XVER = "hl7.fhir.xver-extensions";
|
public static final String ID_XVER = "hl7.fhir.xver-extensions";
|
||||||
public static final String VER_XVER = "0.0.8";
|
public static final String VER_XVER = "0.0.11";
|
||||||
|
|
||||||
public static final String ID_PUBPACK = "hl7.fhir.pubpack";
|
public static final String ID_PUBPACK = "hl7.fhir.pubpack";
|
||||||
public static final String VER_PUBPACK = "0.0.9";
|
public static final String VER_PUBPACK = "0.0.9";
|
||||||
|
|
|
@ -629,6 +629,8 @@ Unable_to_connect_to_terminology_server = Unable to connect to terminology serve
|
||||||
SD_ED_TYPE_PROFILE_UNKNOWN = Unable to resolve profile {0}
|
SD_ED_TYPE_PROFILE_UNKNOWN = Unable to resolve profile {0}
|
||||||
SD_ED_TYPE_PROFILE_NOTYPE = Found profile {0}, but unable to determine the type it applies to
|
SD_ED_TYPE_PROFILE_NOTYPE = Found profile {0}, but unable to determine the type it applies to
|
||||||
SD_ED_TYPE_PROFILE_WRONG = Profile {0} is for type {1}, but the {3} element has type {2}
|
SD_ED_TYPE_PROFILE_WRONG = Profile {0} is for type {1}, but the {3} element has type {2}
|
||||||
|
SD_ED_TYPE_PROFILE_IS_MODIFIER = Profile {0} is for not for a modifier extension, but the {3} element is a modifier
|
||||||
|
SD_ED_TYPE_PROFILE_NOT_MODIFIER = Profile {0} is for a modifier extension, but the {3} element is not a modifier
|
||||||
SD_ED_TYPE_PROFILE_WRONG_TARGET = Profile {0} is for type {1}, which is not a {4} (which is required because the {3} element has type {2})
|
SD_ED_TYPE_PROFILE_WRONG_TARGET = Profile {0} is for type {1}, which is not a {4} (which is required because the {3} element has type {2})
|
||||||
SD_ED_TYPE_NO_TARGET_PROFILE = Type {0} does not allow for target Profiles
|
SD_ED_TYPE_NO_TARGET_PROFILE = Type {0} does not allow for target Profiles
|
||||||
TERMINOLOGY_TX_NOSVC_BOUND_REQ = Could not confirm that the codes provided are from the required value set {0} because there is no terminology service
|
TERMINOLOGY_TX_NOSVC_BOUND_REQ = Could not confirm that the codes provided are from the required value set {0} because there is no terminology service
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
package org.hl7.fhir.utilities.tests;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Assertions;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
public class RegexTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPath1() throws IOException {
|
||||||
|
Assertions.assertFalse("http://fhir.org/guides/cqf/common/Library/FHIR-ModelInfo|4.0.1".matches("Library"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPath2() throws IOException {
|
||||||
|
Assertions.assertTrue("http://fhir.org/guides/cqf/common/Library/FHIR-ModelInfo|4.0.1".matches(".*Library.*"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPath3() throws IOException {
|
||||||
|
Assertions.assertTrue("http://fhir.org/guides/cqf/common/Library/FHIR-ModelInfo|4.0.1".matches("(?s).*Library.*"));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -294,6 +294,15 @@ public class StructureDefinitionValidator extends BaseValidator {
|
||||||
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.SD_ED_TYPE_PROFILE_NOTYPE, p);
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.SD_ED_TYPE_PROFILE_NOTYPE, p);
|
||||||
} else {
|
} else {
|
||||||
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), isInstanceOf(t, code), I18nConstants.SD_ED_TYPE_PROFILE_WRONG, p, t, code, path);
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), isInstanceOf(t, code), I18nConstants.SD_ED_TYPE_PROFILE_WRONG, p, t, code, path);
|
||||||
|
if (t.getType().equals("Extension")) {
|
||||||
|
boolean isModifierDefinition = checkIsModifierExtension(sd);
|
||||||
|
boolean isModifierContext = path.endsWith(".modifierExtension");
|
||||||
|
if (isModifierDefinition) {
|
||||||
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), isModifierContext, I18nConstants.SD_ED_TYPE_PROFILE_NOT_MODIFIER, p, t, code, path);
|
||||||
|
} else {
|
||||||
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), !isModifierContext, I18nConstants.SD_ED_TYPE_PROFILE_IS_MODIFIER, p, t, code, path);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -325,9 +334,23 @@ public class StructureDefinitionValidator extends BaseValidator {
|
||||||
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.SD_ED_TYPE_PROFILE_NOTYPE, p);
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.SD_ED_TYPE_PROFILE_NOTYPE, p);
|
||||||
} else if (!isInstanceOf(t, code)) {
|
} else if (!isInstanceOf(t, code)) {
|
||||||
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.SD_ED_TYPE_PROFILE_WRONG, p, t, code, path);
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.SD_ED_TYPE_PROFILE_WRONG, p, t, code, path);
|
||||||
|
} else {
|
||||||
|
if (t.getType().equals("Extension")) {
|
||||||
|
boolean isModifierDefinition = checkIsModifierExtension(sd);
|
||||||
|
boolean isModifierContext = path.endsWith(".modifierExtension");
|
||||||
|
if (isModifierDefinition) {
|
||||||
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), isModifierContext, I18nConstants.SD_ED_TYPE_PROFILE_NOT_MODIFIER, p, t, code, path);
|
||||||
|
} else {
|
||||||
|
rule(errors, IssueType.EXCEPTION, stack.getLiteralPath(), !isModifierContext, I18nConstants.SD_ED_TYPE_PROFILE_IS_MODIFIER, p, t, code, path);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean checkIsModifierExtension(StructureDefinition t) {
|
||||||
|
return t.getSnapshot().getElementFirstRep().getIsModifier();
|
||||||
|
}
|
||||||
|
|
||||||
private void validateTargetProfile(List<ValidationMessage> errors, Element profile, String code, NodeStack stack, String path) {
|
private void validateTargetProfile(List<ValidationMessage> errors, Element profile, String code, NodeStack stack, String path) {
|
||||||
String p = profile.primitiveValue();
|
String p = profile.primitiveValue();
|
||||||
|
|
2
pom.xml
2
pom.xml
|
@ -19,7 +19,7 @@
|
||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<hapi_fhir_version>5.4.0</hapi_fhir_version>
|
<hapi_fhir_version>5.4.0</hapi_fhir_version>
|
||||||
<validator_test_case_version>1.1.90</validator_test_case_version>
|
<validator_test_case_version>1.1.93-SNAPSHOT</validator_test_case_version>
|
||||||
<junit_jupiter_version>5.7.1</junit_jupiter_version>
|
<junit_jupiter_version>5.7.1</junit_jupiter_version>
|
||||||
<junit_platform_launcher_version>1.7.1</junit_platform_launcher_version>
|
<junit_platform_launcher_version>1.7.1</junit_platform_launcher_version>
|
||||||
<maven_surefire_version>3.0.0-M5</maven_surefire_version>
|
<maven_surefire_version>3.0.0-M5</maven_surefire_version>
|
||||||
|
|
Loading…
Reference in New Issue