Merge pull request #1370 from hapifhir/2023-07-gg-extension-context-fixes

2023 07 gg extension context fixes
This commit is contained in:
Grahame Grieve 2023-08-02 06:05:10 +10:00 committed by GitHub
commit 96b5b6c520
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 59 additions and 23 deletions

View File

@ -949,6 +949,7 @@ public class I18nConstants {
public static final String ED_INVARIANT_KEY_ALREADY_USED = "ED_INVARIANT_KEY_ALREADY_USED"; public static final String ED_INVARIANT_KEY_ALREADY_USED = "ED_INVARIANT_KEY_ALREADY_USED";
public static final String FHIRPATH_OFTYPE_IMPOSSIBLE = "FHIRPATH_OFTYPE_IMPOSSIBLE"; public static final String FHIRPATH_OFTYPE_IMPOSSIBLE = "FHIRPATH_OFTYPE_IMPOSSIBLE";
public static final String ED_SEARCH_EXPRESSION_ERROR = "ED_SEARCH_EXPRESSION_ERROR"; public static final String ED_SEARCH_EXPRESSION_ERROR = "ED_SEARCH_EXPRESSION_ERROR";
public static final String SD_EXTENSION_URL_MISMATCH = "SD_EXTENSION_URL_MISMATCH";

View File

@ -1004,4 +1004,5 @@ ED_INVARIANT_DIFF_NO_SOURCE = The invariant {0} defined in the differential must
FHIRPATH_COLLECTION_STATUS_OPERATION_LEFT = The left side is inherently a collection, and so the expression ''{0}'' may fail or return false if there is more than one item in the content being evaluated FHIRPATH_COLLECTION_STATUS_OPERATION_LEFT = The left side is inherently a collection, and so the expression ''{0}'' may fail or return false if there is more than one item in the content being evaluated
FHIRPATH_COLLECTION_STATUS_OPERATION_RIGHT = The right side is inherently a collection, and so this expression ''{0}'' may fail or return false if there is more than one item in the content being evaluated FHIRPATH_COLLECTION_STATUS_OPERATION_RIGHT = The right side is inherently a collection, and so this expression ''{0}'' may fail or return false if there is more than one item in the content being evaluated
FHIRPATH_OFTYPE_IMPOSSIBLE = The type specified in ofType is {1} which is not a possible candidate for the existing types ({0}) in the expression {2}. Check the paths and types to be sure this is what is intended FHIRPATH_OFTYPE_IMPOSSIBLE = The type specified in ofType is {1} which is not a possible candidate for the existing types ({0}) in the expression {2}. Check the paths and types to be sure this is what is intended
ED_SEARCH_EXPRESSION_ERROR = Error in search expression ''{0}'': {1} ED_SEARCH_EXPRESSION_ERROR = Error in search expression ''{0}'': {1}
SD_EXTENSION_URL_MISMATCH = The fixed value for the extension URL is {1} which doesn''t match the canonical URL {0}

View File

@ -2038,36 +2038,45 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
if (en.contains("#")) { if (en.contains("#")) {
pu = en.substring(0, en.indexOf("#")); pu = en.substring(0, en.indexOf("#"));
en = en.substring(en.indexOf("#")+1); en = en.substring(en.indexOf("#")+1);
} else {
//pu = en;
} }
if (Utilities.existsInList(en, "Element", "Any")) { if (Utilities.existsInList(en, "Element", "Any")) {
ok = true; ok = true;
} else if (en.equals("Resource") && container.isResource()) { } else if (en.equals("Resource") && container.isResource()) {
ok = true; ok = true;
} else if (en.equals("CanonicalResource") && VersionUtilities.getCanonicalResourceNames(context.getVersion()).contains(stack.getLiteralPath())) {
ok = true;
} else if (plist.contains(en) && pu == null) {
ok = true;
} }
if (checkConformsToProfile(appContext, errors, resource, container, stack, extUrl, ctxt.getExpression(), pu)) {
for (String p : plist) { if (!ok) {
if (ok) { if (checkConformsToProfile(appContext, errors, resource, container, stack, extUrl, ctxt.getExpression(), pu)) {
break; for (String p : plist) {
} if (ok) {
if (p.equals(en)) { break;
ok = true;
} else {
String pn = p;
String pt = "";
if (p.contains(".")) {
pn = p.substring(0, p.indexOf("."));
pt = p.substring(p.indexOf("."));
} }
StructureDefinition sd = context.fetchTypeDefinition(pn); if (p.equals(en)) {
while (sd != null) { ok = true;
if ((sd.getType() + pt).equals(en)) { } else {
ok = true; String pn = p;
break; String pt = "";
if (p.contains(".")) {
pn = p.substring(0, p.indexOf("."));
pt = p.substring(p.indexOf("."));
} }
if (sd.getBaseDefinition() != null) { StructureDefinition sd = context.fetchTypeDefinition(pn);
sd = context.fetchResource(StructureDefinition.class, sd.getBaseDefinition(), sd); while (sd != null) {
} else { if ((sd.getType() + pt).equals(en)) {
sd = null; ok = true;
break;
}
if (sd.getBaseDefinition() != null) {
sd = context.fetchResource(StructureDefinition.class, sd.getBaseDefinition(), sd);
} else {
sd = null;
}
} }
} }
} }

View File

@ -78,6 +78,8 @@ public class StructureDefinitionValidator extends BaseValidator {
StructureDefinition sd = null; StructureDefinition sd = null;
String typeName = null; String typeName = null;
try { try {
String url = src.getNamedChildValue("url");
sd = loadAsSD(src); sd = loadAsSD(src);
checkExtensionContext(errors, src, stack); checkExtensionContext(errors, src, stack);
@ -160,6 +162,16 @@ public class StructureDefinitionValidator extends BaseValidator {
} }
c++; c++;
} }
// if this is defining an extension, make sure that the extension fixed value matches the URL
String type = src.getNamedChildValue("type");
if ("Extension".equals(type)) {
String baseD = src.getNamedChildValue("baseDefinition");
if ("http://hl7.org/fhir/StructureDefinition/Extension".equals(baseD) && url != null) {
String fixedUrl = getFixedValue(src);
ok = rule(errors, "2023-05-27", IssueType.INVALID, stack.getLiteralPath(), url.equals(fixedUrl), I18nConstants.SD_EXTENSION_URL_MISMATCH, url, fixedUrl) && ok;
}
}
} catch (Exception e) { } catch (Exception e) {
rule(errors, NO_RULE_DATE, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.ERROR_GENERATING_SNAPSHOT, e.getMessage()); rule(errors, NO_RULE_DATE, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.ERROR_GENERATING_SNAPSHOT, e.getMessage());
ok = false; ok = false;
@ -168,6 +180,19 @@ public class StructureDefinitionValidator extends BaseValidator {
} }
private String getFixedValue(Element src) {
Element diff = src.getNamedChild("differential");
if (diff != null) {
for (Element ed : diff.getChildrenByName("element")) {
String path = ed.getNamedChildValue("path");
if ("Extension.url".equals(path)) {
return ed.getNamedChildValue("fixed");
}
}
}
return null;
}
private boolean validateInheritsObligationProfile(List<ValidationMessage> errors, Element extension, NodeStack stack, Element src) { private boolean validateInheritsObligationProfile(List<ValidationMessage> errors, Element extension, NodeStack stack, Element src) {
String tgt = extension.getNamedChildValue("value"); String tgt = extension.getNamedChildValue("value");
if (rule(errors, "2023-05-27", IssueType.INVALID, stack.getLiteralPath(), tgt != null, if (rule(errors, "2023-05-27", IssueType.INVALID, stack.getLiteralPath(), tgt != null,