fix failing tests.
This commit is contained in:
parent
a34c33c0c0
commit
791f16a05d
|
@ -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);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -627,174 +627,178 @@ 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());
|
||||||
|
|
||||||
if (!Utilities.noString(webUrl) && !webUrl.endsWith("/"))
|
|
||||||
webUrl = webUrl + '/';
|
|
||||||
|
|
||||||
if (defWebRoot == null)
|
|
||||||
defWebRoot = webUrl;
|
|
||||||
derived.setSnapshot(new StructureDefinitionSnapshotComponent());
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
checkDifferential(derived.getDifferential().getElement(), typeName(derived.getType()), derived.getUrl());
|
|
||||||
checkDifferentialBaseType(derived);
|
|
||||||
|
|
||||||
// so we have two lists - the base list, and the differential list
|
if (!Utilities.noString(webUrl) && !webUrl.endsWith("/"))
|
||||||
// the differential list is only allowed to include things that are in the base list, but
|
webUrl = webUrl + '/';
|
||||||
// is allowed to include them multiple times - thereby slicing them
|
|
||||||
|
|
||||||
// our approach is to walk through the base list, and see whether the differential
|
if (defWebRoot == null)
|
||||||
// says anything about them.
|
defWebRoot = webUrl;
|
||||||
int baseCursor = 0;
|
derived.setSnapshot(new StructureDefinitionSnapshotComponent());
|
||||||
int diffCursor = 0; // we need a diff cursor because we can only look ahead, in the bound scoped by longer paths
|
|
||||||
|
try {
|
||||||
|
checkDifferential(derived.getDifferential().getElement(), typeName(derived.getType()), derived.getUrl());
|
||||||
|
checkDifferentialBaseType(derived);
|
||||||
|
|
||||||
|
// so we have two lists - the base list, and the differential list
|
||||||
|
// the differential list is only allowed to include things that are in the base list, but
|
||||||
|
// is allowed to include them multiple times - thereby slicing them
|
||||||
|
|
||||||
|
// our approach is to walk through the base list, and see whether the differential
|
||||||
|
// says anything about them.
|
||||||
|
int baseCursor = 0;
|
||||||
|
int diffCursor = 0; // we need a diff cursor because we can only look ahead, in the bound scoped by longer paths
|
||||||
|
|
||||||
|
|
||||||
for (ElementDefinition e : derived.getDifferential().getElement())
|
for (ElementDefinition e : derived.getDifferential().getElement())
|
||||||
e.clearUserData(GENERATED_IN_SNAPSHOT);
|
e.clearUserData(GENERATED_IN_SNAPSHOT);
|
||||||
|
|
||||||
// we actually delegate the work to a subroutine so we can re-enter it with a different cursors
|
// we actually delegate the work to a subroutine so we can re-enter it with a different cursors
|
||||||
StructureDefinitionDifferentialComponent diff = cloneDiff(derived.getDifferential()); // we make a copy here because we're sometimes going to hack the differential while processing it. Have to migrate user data back afterwards
|
StructureDefinitionDifferentialComponent diff = cloneDiff(derived.getDifferential()); // we make a copy here because we're sometimes going to hack the differential while processing it. Have to migrate user data back afterwards
|
||||||
StructureDefinitionSnapshotComponent baseSnapshot = base.getSnapshot();
|
StructureDefinitionSnapshotComponent baseSnapshot = base.getSnapshot();
|
||||||
if (derived.getDerivation() == TypeDerivationRule.SPECIALIZATION) {
|
if (derived.getDerivation() == TypeDerivationRule.SPECIALIZATION) {
|
||||||
String derivedType = derived.getType();
|
String derivedType = derived.getType();
|
||||||
if (StructureDefinitionKind.LOGICAL.equals(derived.getKind()) && derived.getType().contains("/")) {
|
if (StructureDefinitionKind.LOGICAL.equals(derived.getKind()) && derived.getType().contains("/")) {
|
||||||
derivedType = derivedType.substring(derivedType.lastIndexOf("/")+1);
|
derivedType = derivedType.substring(derivedType.lastIndexOf("/")+1);
|
||||||
|
}
|
||||||
|
baseSnapshot = cloneSnapshot(baseSnapshot, base.getType(), derivedType);
|
||||||
}
|
}
|
||||||
baseSnapshot = cloneSnapshot(baseSnapshot, base.getType(), derivedType);
|
// if (derived.getId().equals("2.16.840.1.113883.10.20.22.2.1.1")) {
|
||||||
}
|
// debug = true;
|
||||||
// if (derived.getId().equals("2.16.840.1.113883.10.20.22.2.1.1")) {
|
// }
|
||||||
// debug = true;
|
processPaths("", derived.getSnapshot(), baseSnapshot, diff, baseCursor, diffCursor, baseSnapshot.getElement().size()-1,
|
||||||
// }
|
derived.getDifferential().hasElement() ? derived.getDifferential().getElement().size()-1 : -1, url, webUrl, derived.present(), null, null, false, base.getUrl(), null, false, null, null, new ArrayList<ElementRedirection>(), base);
|
||||||
processPaths("", derived.getSnapshot(), baseSnapshot, diff, baseCursor, diffCursor, baseSnapshot.getElement().size()-1,
|
checkGroupConstraints(derived);
|
||||||
derived.getDifferential().hasElement() ? derived.getDifferential().getElement().size()-1 : -1, url, webUrl, derived.present(), null, null, false, base.getUrl(), null, false, null, null, new ArrayList<ElementRedirection>(), base);
|
if (derived.getDerivation() == TypeDerivationRule.SPECIALIZATION) {
|
||||||
checkGroupConstraints(derived);
|
for (ElementDefinition e : diff.getElement()) {
|
||||||
if (derived.getDerivation() == TypeDerivationRule.SPECIALIZATION) {
|
if (!e.hasUserData(GENERATED_IN_SNAPSHOT)) {
|
||||||
for (ElementDefinition e : diff.getElement()) {
|
ElementDefinition outcome = updateURLs(url, webUrl, e.copy());
|
||||||
if (!e.hasUserData(GENERATED_IN_SNAPSHOT)) {
|
e.setUserData(GENERATED_IN_SNAPSHOT, outcome);
|
||||||
ElementDefinition outcome = updateURLs(url, webUrl, e.copy());
|
derived.getSnapshot().addElement(outcome);
|
||||||
e.setUserData(GENERATED_IN_SNAPSHOT, outcome);
|
}
|
||||||
derived.getSnapshot().addElement(outcome);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (derived.getKind() != StructureDefinitionKind.LOGICAL && !derived.getSnapshot().getElementFirstRep().getType().isEmpty())
|
if (derived.getKind() != StructureDefinitionKind.LOGICAL && !derived.getSnapshot().getElementFirstRep().getType().isEmpty())
|
||||||
throw new Error(context.formatMessage(I18nConstants.TYPE_ON_FIRST_SNAPSHOT_ELEMENT_FOR__IN__FROM_, derived.getSnapshot().getElementFirstRep().getPath(), derived.getUrl(), base.getUrl()));
|
throw new Error(context.formatMessage(I18nConstants.TYPE_ON_FIRST_SNAPSHOT_ELEMENT_FOR__IN__FROM_, derived.getSnapshot().getElementFirstRep().getPath(), derived.getUrl(), base.getUrl()));
|
||||||
updateMaps(base, derived);
|
updateMaps(base, derived);
|
||||||
|
|
||||||
setIds(derived, false);
|
setIds(derived, false);
|
||||||
if (debug) {
|
if (debug) {
|
||||||
System.out.println("Differential: ");
|
|
||||||
for (ElementDefinition ed : derived.getDifferential().getElement())
|
|
||||||
System.out.println(" "+ed.getId()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
|
||||||
System.out.println("Snapshot: ");
|
|
||||||
for (ElementDefinition ed : derived.getSnapshot().getElement())
|
|
||||||
System.out.println(" "+ed.getId()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
|
||||||
}
|
|
||||||
CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder();
|
|
||||||
//Check that all differential elements have a corresponding snapshot element
|
|
||||||
int ce = 0;
|
|
||||||
for (ElementDefinition e : diff.getElement()) {
|
|
||||||
if (!e.hasUserData("diff-source"))
|
|
||||||
throw new Error(context.formatMessage(I18nConstants.UNXPECTED_INTERNAL_CONDITION__NO_SOURCE_ON_DIFF_ELEMENT));
|
|
||||||
else {
|
|
||||||
if (e.hasUserData(DERIVATION_EQUALS))
|
|
||||||
((Base) e.getUserData("diff-source")).setUserData(DERIVATION_EQUALS, e.getUserData(DERIVATION_EQUALS));
|
|
||||||
if (e.hasUserData(DERIVATION_POINTER))
|
|
||||||
((Base) e.getUserData("diff-source")).setUserData(DERIVATION_POINTER, e.getUserData(DERIVATION_POINTER));
|
|
||||||
}
|
|
||||||
if (!e.hasUserData(GENERATED_IN_SNAPSHOT)) {
|
|
||||||
b.append(e.hasId() ? "id: "+e.getId() : "path: "+e.getPath());
|
|
||||||
ce++;
|
|
||||||
if (e.hasId()) {
|
|
||||||
String msg = "No match found in the generated snapshot: check that the path and definitions are legal in the differential (including order)";
|
|
||||||
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url+"#"+e.getId(), msg, ValidationMessage.IssueSeverity.ERROR));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!Utilities.noString(b.toString())) {
|
|
||||||
String msg = "The profile "+derived.getUrl()+" has "+ce+" "+Utilities.pluralize("element", ce)+" in the differential ("+b.toString()+") that don't have a matching element in the snapshot: check that the path and definitions are legal in the differential (including order)";
|
|
||||||
System.out.println("Error in snapshot generation: "+msg);
|
|
||||||
if (!debug) {
|
|
||||||
System.out.println("Differential: ");
|
System.out.println("Differential: ");
|
||||||
for (ElementDefinition ed : derived.getDifferential().getElement())
|
for (ElementDefinition ed : derived.getDifferential().getElement())
|
||||||
System.out.println(" "+ed.getId()+" = "+ed.getPath()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
System.out.println(" "+ed.getId()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
||||||
System.out.println("Snapshot: ");
|
System.out.println("Snapshot: ");
|
||||||
for (ElementDefinition ed : derived.getSnapshot().getElement())
|
for (ElementDefinition ed : derived.getSnapshot().getElement())
|
||||||
System.out.println(" "+ed.getId()+" = "+ed.getPath()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
System.out.println(" "+ed.getId()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
||||||
}
|
}
|
||||||
if (exception)
|
CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder();
|
||||||
throw new DefinitionException(msg);
|
//Check that all differential elements have a corresponding snapshot element
|
||||||
else
|
int ce = 0;
|
||||||
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url, msg, ValidationMessage.IssueSeverity.ERROR));
|
for (ElementDefinition e : diff.getElement()) {
|
||||||
}
|
if (!e.hasUserData("diff-source"))
|
||||||
// hack around a problem in R4 definitions (somewhere?)
|
throw new Error(context.formatMessage(I18nConstants.UNXPECTED_INTERNAL_CONDITION__NO_SOURCE_ON_DIFF_ELEMENT));
|
||||||
for (ElementDefinition ed : derived.getSnapshot().getElement()) {
|
else {
|
||||||
for (ElementDefinitionMappingComponent mm : ed.getMapping()) {
|
if (e.hasUserData(DERIVATION_EQUALS))
|
||||||
if (mm.hasMap()) {
|
((Base) e.getUserData("diff-source")).setUserData(DERIVATION_EQUALS, e.getUserData(DERIVATION_EQUALS));
|
||||||
mm.setMap(mm.getMap().trim());
|
if (e.hasUserData(DERIVATION_POINTER))
|
||||||
|
((Base) e.getUserData("diff-source")).setUserData(DERIVATION_POINTER, e.getUserData(DERIVATION_POINTER));
|
||||||
}
|
}
|
||||||
}
|
if (!e.hasUserData(GENERATED_IN_SNAPSHOT)) {
|
||||||
for (ElementDefinitionConstraintComponent s : ed.getConstraint()) {
|
b.append(e.hasId() ? "id: "+e.getId() : "path: "+e.getPath());
|
||||||
if (s.hasSource()) {
|
ce++;
|
||||||
String ref = s.getSource();
|
if (e.hasId()) {
|
||||||
if (!Utilities.isAbsoluteUrl(ref)) {
|
String msg = "No match found in the generated snapshot: check that the path and definitions are legal in the differential (including order)";
|
||||||
if (ref.contains(".")) {
|
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url+"#"+e.getId(), msg, ValidationMessage.IssueSeverity.ERROR));
|
||||||
s.setSource("http://hl7.org/fhir/StructureDefinition/"+ref.substring(0, ref.indexOf("."))+"#"+ref);
|
|
||||||
} else {
|
|
||||||
s.setSource("http://hl7.org/fhir/StructureDefinition/"+ref);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
if (!Utilities.noString(b.toString())) {
|
||||||
if (derived.getDerivation() == TypeDerivationRule.SPECIALIZATION) {
|
String msg = "The profile "+derived.getUrl()+" has "+ce+" "+Utilities.pluralize("element", ce)+" in the differential ("+b.toString()+") that don't have a matching element in the snapshot: check that the path and definitions are legal in the differential (including order)";
|
||||||
|
System.out.println("Error in snapshot generation: "+msg);
|
||||||
|
if (!debug) {
|
||||||
|
System.out.println("Differential: ");
|
||||||
|
for (ElementDefinition ed : derived.getDifferential().getElement())
|
||||||
|
System.out.println(" "+ed.getId()+" = "+ed.getPath()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
||||||
|
System.out.println("Snapshot: ");
|
||||||
|
for (ElementDefinition ed : derived.getSnapshot().getElement())
|
||||||
|
System.out.println(" "+ed.getId()+" = "+ed.getPath()+" : "+typeSummaryWithProfile(ed)+"["+ed.getMin()+".."+ed.getMax()+"]"+sliceSummary(ed)+" "+constraintSummary(ed));
|
||||||
|
}
|
||||||
|
if (exception)
|
||||||
|
throw new DefinitionException(msg);
|
||||||
|
else
|
||||||
|
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url, msg, ValidationMessage.IssueSeverity.ERROR));
|
||||||
|
}
|
||||||
|
// hack around a problem in R4 definitions (somewhere?)
|
||||||
for (ElementDefinition ed : derived.getSnapshot().getElement()) {
|
for (ElementDefinition ed : derived.getSnapshot().getElement()) {
|
||||||
if (!ed.hasBase()) {
|
for (ElementDefinitionMappingComponent mm : ed.getMapping()) {
|
||||||
ed.getBase().setPath(ed.getPath()).setMin(ed.getMin()).setMax(ed.getMax());
|
if (mm.hasMap()) {
|
||||||
}
|
mm.setMap(mm.getMap().trim());
|
||||||
}
|
|
||||||
}
|
|
||||||
// last, check for wrong profiles or target profiles
|
|
||||||
for (ElementDefinition ed : derived.getSnapshot().getElement()) {
|
|
||||||
for (TypeRefComponent t : ed.getType()) {
|
|
||||||
for (UriType u : t.getProfile()) {
|
|
||||||
StructureDefinition sd = context.fetchResource(StructureDefinition.class, u.getValue());
|
|
||||||
if (sd == null) {
|
|
||||||
if (xver != null && xver.matchingUrl(u.getValue()) && xver.status(u.getValue()) == XVerExtensionStatus.Valid) {
|
|
||||||
sd = xver.makeDefinition(u.getValue());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (sd == null) {
|
}
|
||||||
if (messages != null) {
|
for (ElementDefinitionConstraintComponent s : ed.getConstraint()) {
|
||||||
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url+"#"+ed.getId(), "The type of profile "+u.getValue()+" cannot be checked as the profile is not known", IssueSeverity.WARNING));
|
if (s.hasSource()) {
|
||||||
}
|
String ref = s.getSource();
|
||||||
} else {
|
if (!Utilities.isAbsoluteUrl(ref)) {
|
||||||
String wt = t.getWorkingCode();
|
if (ref.contains(".")) {
|
||||||
if (ed.getPath().equals("Bundle.entry.response.outcome")) {
|
s.setSource("http://hl7.org/fhir/StructureDefinition/"+ref.substring(0, ref.indexOf("."))+"#"+ref);
|
||||||
wt = "OperationOutcome";
|
} else {
|
||||||
}
|
s.setSource("http://hl7.org/fhir/StructureDefinition/"+ref);
|
||||||
if (!sd.getType().equals(wt)) {
|
|
||||||
boolean ok = isCompatibleType(wt, sd);
|
|
||||||
if (!ok) {
|
|
||||||
String smsg = "The profile "+u.getValue()+" has type "+sd.getType()+" which is not consistent with the stated type "+wt;
|
|
||||||
if (exception)
|
|
||||||
throw new DefinitionException(smsg);
|
|
||||||
else
|
|
||||||
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url+"#"+ed.getId(), smsg, IssueSeverity.ERROR));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (derived.getDerivation() == TypeDerivationRule.SPECIALIZATION) {
|
||||||
|
for (ElementDefinition ed : derived.getSnapshot().getElement()) {
|
||||||
|
if (!ed.hasBase()) {
|
||||||
|
ed.getBase().setPath(ed.getPath()).setMin(ed.getMin()).setMax(ed.getMax());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// last, check for wrong profiles or target profiles
|
||||||
|
for (ElementDefinition ed : derived.getSnapshot().getElement()) {
|
||||||
|
for (TypeRefComponent t : ed.getType()) {
|
||||||
|
for (UriType u : t.getProfile()) {
|
||||||
|
StructureDefinition sd = context.fetchResource(StructureDefinition.class, u.getValue());
|
||||||
|
if (sd == null) {
|
||||||
|
if (xver != null && xver.matchingUrl(u.getValue()) && xver.status(u.getValue()) == XVerExtensionStatus.Valid) {
|
||||||
|
sd = xver.makeDefinition(u.getValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (sd == null) {
|
||||||
|
if (messages != null) {
|
||||||
|
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url+"#"+ed.getId(), "The type of profile "+u.getValue()+" cannot be checked as the profile is not known", IssueSeverity.WARNING));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
String wt = t.getWorkingCode();
|
||||||
|
if (ed.getPath().equals("Bundle.entry.response.outcome")) {
|
||||||
|
wt = "OperationOutcome";
|
||||||
|
}
|
||||||
|
if (!sd.getType().equals(wt)) {
|
||||||
|
boolean ok = isCompatibleType(wt, sd);
|
||||||
|
if (!ok) {
|
||||||
|
String smsg = "The profile "+u.getValue()+" has type "+sd.getType()+" which is not consistent with the stated type "+wt;
|
||||||
|
if (exception)
|
||||||
|
throw new DefinitionException(smsg);
|
||||||
|
else
|
||||||
|
messages.add(new ValidationMessage(Source.ProfileValidator, ValidationMessage.IssueType.VALUE, url+"#"+ed.getId(), smsg, IssueSeverity.ERROR));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
// if we had an exception generating the snapshot, make sure we don't leave any half generated snapshot behind
|
||||||
|
derived.setSnapshot(null);
|
||||||
|
derived.clearUserData("profileutils.snapshot.generating");
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
} finally {
|
||||||
// if we had an exception generating the snapshot, make sure we don't leave any half generated snapshot behind
|
|
||||||
derived.setSnapshot(null);
|
|
||||||
derived.clearUserData("profileutils.snapshot.generating");
|
derived.clearUserData("profileutils.snapshot.generating");
|
||||||
throw e;
|
snapshotStack.remove(derived.getUrl());
|
||||||
}
|
}
|
||||||
derived.clearUserData("profileutils.snapshot.generating");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void checkDifferentialBaseType(StructureDefinition derived) throws Error {
|
public void checkDifferentialBaseType(StructureDefinition derived) throws Error {
|
||||||
|
@ -1147,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));
|
||||||
}
|
}
|
||||||
|
@ -1300,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();
|
||||||
|
@ -1511,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;
|
||||||
|
@ -1577,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+"."))
|
||||||
|
@ -1778,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()));
|
||||||
}
|
}
|
||||||
|
@ -1894,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 =
|
||||||
// }
|
// }
|
||||||
|
@ -1946,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));
|
||||||
}
|
}
|
||||||
|
@ -1956,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;
|
||||||
|
@ -2428,13 +2432,14 @@ 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 (sd == null) {
|
||||||
if (xver != null && xver.matchingUrl(type.getProfile().get(0).getValue()) && xver.status(type.getProfile().get(0).getValue()) == XVerExtensionStatus.Valid) {
|
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());
|
sd = xver.makeDefinition(type.getProfile().get(0).getValue());
|
||||||
|
generateSnapshot(context.fetchTypeDefinition("Extension"), sd, sd.getUrl(), webUrl, sd.getName());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (sd == null)
|
if (sd == null)
|
||||||
|
|
|
@ -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.*"));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
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.92</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