From a589f5fdf486cc9d60137a93198ad65c7feb049e Mon Sep 17 00:00:00 2001 From: Lloyd McKenzie Date: Tue, 27 Sep 2022 08:26:46 -0600 Subject: [PATCH 1/3] Added support for rendering with differences between either base StructureDfeinition or parent StructureDefinition --- .../AdditionalBindingsRenderer.java | 176 ++++++++++++++---- .../fhir/r5/conformance/ProfileUtilities.java | 64 +++++-- .../StructureDefinitionRenderer.java | 2 +- 3 files changed, 189 insertions(+), 53 deletions(-) diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java index ffdfb9d9b..df10ee082 100644 --- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java +++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java @@ -2,6 +2,7 @@ package org.hl7.fhir.r5.conformance; import java.io.IOException; import java.util.ArrayList; +import java.util.HashMap; import java.util.List; import org.hl7.fhir.exceptions.DefinitionException; @@ -33,8 +34,28 @@ public class AdditionalBindingsRenderer { private UsageContext usage; private boolean any; private boolean unchanged; + private boolean matched; + private AdditionalBindingDetail compare; + private int count = 1; + private String getKey() { + // Todo: Consider extending this with content from usageContext if purpose isn't sufficiently differentiating + return purpose; + } + private void incrementCount() { + count++; + } + private void setCompare(AdditionalBindingDetail match) { + compare = match; + match.matched = true; + } + private boolean alreadyMatched() { + return matched; + } } + private static String STYLE_UNCHANGED = "font-color: darkgray;"; + private static String STYLE_REMOVED = STYLE_UNCHANGED + "text-decoration: line-through;"; + private List bindings = new ArrayList<>(); private ProfileKnowledgeProvider pkp; private String corePath; @@ -52,34 +73,78 @@ public class AdditionalBindingsRenderer { this.md = md; } - public void seeMaxBinding(Extension ext) { + public void seeMaxBinding(Extension ext) { + seeMaxBinding(ext, null, false); + } + + public void seeMaxBinding(Extension ext, Extension compExt, boolean compare) { + seeBinding(ext, compExt, compare, "maximum"); + } + + protected void seeBinding(Extension ext, Extension compExt, boolean compare, String label) { AdditionalBindingDetail abr = new AdditionalBindingDetail(); - abr.purpose = "maximum"; + abr.purpose = label; abr.valueSet = ext.getValue().primitiveValue(); - abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); - bindings.add(abr); + if (compare) { + abr.unchanged = compExt!=null && ext.getValue().primitiveValue().equals(compExt.getValue().primitiveValue()); + + abr.compare = new AdditionalBindingDetail(); + abr.compare.valueSet = compExt==null ? null : compExt.getValue().primitiveValue(); + } else { + abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); + } + bindings.add(abr); } public void seeMinBinding(Extension ext) { - AdditionalBindingDetail abr = new AdditionalBindingDetail(); - abr.purpose = "minimum"; - abr.valueSet = ext.getValue().primitiveValue(); - abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); - bindings.add(abr); + seeMinBinding(ext, null, false); + } + + public void seeMinBinding(Extension ext, Extension compExt, boolean compare) { + seeBinding(ext, compExt, compare, "minimum"); } public void seeAdditionalBindings(List list) { - for (Extension ext : list) { - AdditionalBindingDetail abr = new AdditionalBindingDetail(); - abr.purpose = ext.getExtensionString("purpose"); - abr.valueSet = ext.getExtensionString("valueSet"); - abr.doco = ext.getExtensionString("documentation"); - abr.usage = (ext.hasExtension("usage")) && ext.getExtensionByUrl("usage").hasValueUsageContext() ? ext.getExtensionByUrl("usage").getValueUsageContext() : null; - abr.any = "any".equals(ext.getExtensionString("scope")); - abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); - bindings.add(abr); + seeAdditionalBindings(list, null, false); + } + + public void seeAdditionalBindings(List list, List compList, boolean compare) { + HashMap compBindings = new HashMap(); + if (compare && compList!=null) { + for (Extension ext : compList) { + AdditionalBindingDetail abr = additionalBinding(ext); + while (compBindings.containsKey(abr.getKey())) + abr.incrementCount(); + compBindings.put(abr.getKey(), abr); + } } - } + + for (Extension ext : list) { + AdditionalBindingDetail abr = additionalBinding(ext); + if (compare && compList!=null) { + AdditionalBindingDetail match = null; + do { + match = compBindings.get(abr.getKey()); + if (abr.alreadyMatched()) + abr.incrementCount(); + } while (match!=null && !match.alreadyMatched()); + if (match!=null) + abr.setCompare(match); + } else + bindings.add(abr); + } + } + + protected AdditionalBindingDetail additionalBinding(Extension ext) { + AdditionalBindingDetail abr = new AdditionalBindingDetail(); + abr.purpose = ext.getExtensionString("purpose"); + abr.valueSet = ext.getExtensionString("valueSet"); + abr.doco = ext.getExtensionString("documentation"); + abr.usage = (ext.hasExtension("usage")) && ext.getExtensionByUrl("usage").hasValueUsageContext() ? ext.getExtensionByUrl("usage").getValueUsageContext() : null; + abr.any = "any".equals(ext.getExtensionString("scope")); + abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); + return abr; + } public String render() throws IOException { if (bindings.isEmpty()) { @@ -107,12 +172,11 @@ public class AdditionalBindingsRenderer { boolean usage = false; boolean any = false; for (AdditionalBindingDetail binding : bindings) { - doco = doco || (doDoco && binding.doco != null); - usage = usage || binding.usage != null; - any = any || binding.any; + doco = doco || (doDoco && (binding.doco != null || (binding.compare!=null && binding.compare.doco!=null))); + usage = usage || binding.usage != null || (binding.compare!=null && binding.compare.usage!=null); + any = any || binding.any || (binding.compare!=null && binding.compare.any); } - XhtmlNode tr = new XhtmlNode(NodeType.Element, "tr"); children.add(tr); tr.td().style("font-size: 11px").b().tx("Additional Bindings"); @@ -133,31 +197,55 @@ public class AdditionalBindingsRenderer { } children.add(tr); BindingResolution br = pkp == null ? makeNullBr(binding) : pkp.resolveBinding(profile, binding.valueSet, path); + BindingResolution compBr = null; + if (binding.compare!=null && binding.compare.valueSet!=null) + compBr = pkp == null ? makeNullBr(binding.compare) : pkp.resolveBinding(profile, binding.compare.valueSet, path); + XhtmlNode valueset = tr.td().style("font-size: 11px"); + if (binding.compare!=null && binding.valueSet.equals(binding.compare.valueSet)) + valueset.style(STYLE_UNCHANGED); if (br.url != null) { - tr.td().style("font-size: 11px").ah(Utilities.isAbsoluteUrl(br.url) || !pkp.prependLinks() ? br.url : corePath+br.url, binding.valueSet).tx(br.display); + valueset.ah(determineUrl(br.url), binding.valueSet).tx(br.display); } else { - tr.td().style("font-size: 11px").span(null, binding.valueSet).tx(br.display); + valueset.span(null, binding.valueSet).tx(br.display); + } + if (binding.compare!=null && binding.compare.valueSet!=null && !binding.valueSet.equals(binding.compare.valueSet)) { + valueset.br(); + valueset = valueset.span(STYLE_REMOVED, null); + if (compBr.url != null) { + valueset.ah(determineUrl(compBr.url), binding.compare.valueSet).tx(compBr.display); + } else { + valueset.span(null, binding.compare.valueSet).tx(compBr.display); + } + } + + XhtmlNode purpose = tr.td().style("font-size: 11px"); + if (binding.compare!=null && binding.purpose.equals(binding.compare.purpose)) + purpose.style("font-color: darkgray"); + renderPurpose(purpose, binding.purpose); + if (binding.compare!=null && binding.compare.purpose!=null && !binding.purpose.equals(binding.compare.purpose)) { + purpose.br(); + purpose = purpose.span(STYLE_UNCHANGED, null); + renderPurpose(purpose, binding.compare.purpose); } - renderPurpose(tr.td().style("font-size: 11px"), binding.purpose); if (usage) { if (binding.usage != null) { + // TODO: This isn't rendered at all yet. Ideally, we want it to render with comparison... new DataRenderer(context).render(tr.td(), binding.usage); } else { tr.td(); } } if (any) { - if (binding.any) { - tr.td().style("font-size: 11px").tx("Any repeat"); - } else { - tr.td().style("font-size: 11px").tx("All repeats"); - } + String newRepeat = binding.any ? "Any repeats" : "All repeats"; + String oldRepeat = binding.compare!=null && binding.compare.any ? "Any repeats" : "All repeats"; + compareString(tr.td().style("font-size: 11px"), newRepeat, oldRepeat); } if (doco) { if (binding.doco != null) { - String d = md.processMarkdown("Binding.description", binding.doco); - tr.td().style("font-size: 11px").innerHTML(d); + String d = md.processMarkdown("Binding.description", binding.doco); + String oldD = binding.compare==null ? null : md.processMarkdown("Binding.description.compare", binding.compare.doco); + tr.td().style("font-size: 11px").innerHTML(compareHtml(d, oldD)); } else { tr.td().style("font-size: 11px"); } @@ -165,6 +253,28 @@ public class AdditionalBindingsRenderer { } } + private XhtmlNode compareString(XhtmlNode node, String newS, String oldS) { + if (oldS==null) + return node.tx(newS); + if (newS.equals(oldS)) + return node.style(STYLE_UNCHANGED).tx(newS); + node.tx(newS); + node.br(); + return node.span(STYLE_REMOVED,null).tx(oldS); + } + + private String compareHtml(String newS, String oldS) { + if (oldS==null) + return newS; + if (newS.equals(oldS)) + return "" + newS + ""; + return newS + "
" + oldS + ""; + } + + private String determineUrl(String url) { + return Utilities.isAbsoluteUrl(url) || !pkp.prependLinks() ? url : corePath + url; + } + private void renderPurpose(XhtmlNode td, String purpose) { switch (purpose) { case "maximum": diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java index 0bdff4535..ff6ee53f4 100644 --- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java +++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java @@ -325,7 +325,8 @@ public class ProfileUtilities extends TranslatingUtilities { public static final int STATUS_ERROR = 3; public static final int STATUS_FATAL = 4; - + public static final String BASE_MODEL = "base.model"; + public static final String BASE_PATH = "base.path"; public static final String DERIVATION_EQUALS = "derivation.equals"; public static final String DERIVATION_POINTER = "derived.pointer"; public static final String IS_DERIVED = "derived.fact"; @@ -354,6 +355,7 @@ public class ProfileUtilities extends TranslatingUtilities { private boolean wantFixDifferentialFirstElementType; private Set masterSourceFileNames; private Map> childMapCache = new HashMap<>(); + private Map> sdMapCache = new HashMap<>(); public ProfileUtilities(IWorkerContext context, List messages, ProfileKnowledgeProvider pkp, FHIRPathEngine fpe) { super(); @@ -423,7 +425,21 @@ public class ProfileUtilities extends TranslatingUtilities { String getLinkForUrl(String corePath, String s); } + public ElementDefinition getElementById(String structureCanonical, String id) { + Map sdCache = sdMapCache.get(structureCanonical); + if (sdCache == null) { + StructureDefinition sd = (StructureDefinition) context.fetchResource(StructureDefinition.class, structureCanonical); + if (sd==null) + throw new FHIRException("Unable to retrieve StructureDefinition with URL " + structureCanonical); + sdCache = new HashMap(); + sdMapCache.put(structureCanonical, sdCache); + for (ElementDefinition e : sd.getSnapshot().getElement()) { + sdCache.put(e.getId(), e); + } + } + return sdCache.get(id); + } public List getChildMap(StructureDefinition profile, ElementDefinition element) throws DefinitionException { if (childMapCache .containsKey(element)) { @@ -497,7 +513,7 @@ public class ProfileUtilities extends TranslatingUtilities { /** * Given a Structure, navigate to the element given by the path and return the direct children of that element * - * @param structure The structure to navigate into + * @param profile The structure to navigate into * @param path The path of the element within the structure to get the children for * @return A List containing the element children (all of them are Elements) */ @@ -608,10 +624,9 @@ public class ProfileUtilities extends TranslatingUtilities { * Given a base (snapshot) profile structure, and a differential profile, generate a new snapshot profile * * @param base - the base structure on which the differential will be applied - * @param differential - the differential to apply to the base + * @param derived - the differential to apply to the base * @param url - where the base has relative urls for profile references, these need to be converted to absolutes by prepending this URL (e.g. the canonical URL) * @param webUrl - where the base has relative urls in markdown, these need to be converted to absolutes by prepending this URL (this is not the same as the canonical URL) - * @param trimDifferential - if this is true, then the snap short generator will remove any material in the element definitions that is not different to the base * @return * @throws FHIRException * @throws DefinitionException @@ -1133,7 +1148,7 @@ public class ProfileUtilities extends TranslatingUtilities { // so we just copy it in ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy()); outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); updateConstraintSources(outcome, srcSD.getUrl()); markDerived(outcome); if (resultPathBase == null) @@ -1280,7 +1295,7 @@ public class ProfileUtilities extends TranslatingUtilities { outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); if (res == null) res = outcome; - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); if (diffMatches.get(0).hasSliceName()) { outcome.setSliceName(diffMatches.get(0).getSliceName()); if (!diffMatches.get(0).hasMin() && (diffMatches.size() > 1 || slicer == null || slicer.getSlicing().getRules() != SlicingRules.CLOSED) && !currentBase.hasSliceName()) { @@ -1543,7 +1558,7 @@ public class ProfileUtilities extends TranslatingUtilities { // we're just going to accept the differential slicing at face value ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy()); outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); if (!diffMatches.get(0).hasSlicing()) outcome.setSlicing(makeExtensionSlicing()); @@ -1618,7 +1633,7 @@ public class ProfileUtilities extends TranslatingUtilities { // so we just copy it in ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy()); outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); markDerived(outcome); if (resultPathBase == null) resultPathBase = outcome.getPath(); @@ -1655,6 +1670,8 @@ public class ProfileUtilities extends TranslatingUtilities { if (!outcome.getPath().startsWith(resultPathBase)) throw new DefinitionException(context.formatMessage(I18nConstants.ADDING_WRONG_PATH_IN_PROFILE___VS_, profileName, outcome.getPath(), resultPathBase)); result.getElement().add(outcome); // so we just copy it in + outcome.setUserData(BASE_MODEL, srcSD.getUrl()); + outcome.setUserData(BASE_PATH, resultPathBase); baseCursor++; } } @@ -1815,7 +1832,7 @@ public class ProfileUtilities extends TranslatingUtilities { } ElementDefinition outcome = updateURLs(url, webUrl, currentBase.copy()); outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); if (diffMatches.get(0).hasSlicing() || !diffMatches.get(0).hasSliceName()) { updateFromSlicing(outcome.getSlicing(), diffMatches.get(0).getSlicing()); updateFromDefinition(outcome, diffMatches.get(0), profileName, closed, url, srcSD); // if there's no slice, we don't want to update the unsliced description @@ -1867,7 +1884,7 @@ public class ProfileUtilities extends TranslatingUtilities { for (ElementDefinition baseItem : baseMatches) { baseCursor = base.getElement().indexOf(baseItem); outcome = updateURLs(url, webUrl, baseItem.copy()); - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); outcome.setSlicing(null); if (!outcome.getPath().startsWith(resultPathBase)) @@ -1894,6 +1911,8 @@ public class ProfileUtilities extends TranslatingUtilities { outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); if (!outcome.getPath().startsWith(resultPathBase)) throw new DefinitionException(context.formatMessage(I18nConstants.ADDING_WRONG_PATH)); + outcome.setUserData(BASE_PATH, outcome.getPath()); + outcome.setUserData(BASE_MODEL, srcSD.getUrl()); result.getElement().add(outcome); baseCursor++; } @@ -1923,7 +1942,7 @@ public class ProfileUtilities extends TranslatingUtilities { outcome = updateURLs(url, webUrl, currentBase.copy()); // outcome = updateURLs(url, diffItem.copy()); outcome.setPath(fixedPathDest(contextPathDst, outcome.getPath(), redirector, contextPathSrc)); - updateFromBase(outcome, currentBase); + updateFromBase(outcome, currentBase, srcSD.getUrl()); outcome.setSlicing(null); outcome.setMin(0); // we're in a slice, so it's only a mandatory if it's explicitly marked so if (!outcome.getPath().startsWith(resultPathBase)) @@ -2430,7 +2449,9 @@ public class ProfileUtilities extends TranslatingUtilities { } - private void updateFromBase(ElementDefinition derived, ElementDefinition base) { + private void updateFromBase(ElementDefinition derived, ElementDefinition base, String baseProfileUrl) { + derived.setUserData(BASE_MODEL, baseProfileUrl); + derived.setUserData(BASE_PATH, base.getPath()); if (base.hasBase()) { if (!derived.hasBase()) derived.setBase(new ElementDefinitionBaseComponent()); @@ -3522,7 +3543,7 @@ public class ProfileUtilities extends TranslatingUtilities { if (!child.getPath().endsWith(".id")) { List sdl = new ArrayList<>(); sdl.add(ed); - genElement(defFile == null ? "" : defFile+"-definitions.html#extension.", gen, r.getSubRows(), child, ed.getSnapshot().getElement(), sdl, true, defFile, true, full, corePath, imagePath, true, false, false, false, null, false, rc); + genElement(defFile == null ? "" : defFile+"-definitions.html#extension.", gen, r.getSubRows(), child, ed.getSnapshot().getElement(), sdl, true, defFile, true, full, corePath, imagePath, true, false, false, false, null, false, rc, ""); } } else if (deep) { List children = new ArrayList(); @@ -4003,8 +4024,13 @@ public class ProfileUtilities extends TranslatingUtilities { return piece; } - public XhtmlNode generateTable(String defFile, StructureDefinition profile, boolean diff, String imageFolder, boolean inlineGraphics, String profileBaseFileName, boolean snapshot, String corePath, String imagePath, - boolean logicalModel, boolean allInvariants, Set outputTracker, boolean active, boolean mustSupport, RenderingContext rc) throws IOException, FHIRException { + public XhtmlNode generateTable(String defFile, StructureDefinition profile, boolean diff, String imageFolder, boolean inlineGraphics, String profileBaseFileName, boolean snapshot, String corePath, String imagePath, + boolean logicalModel, boolean allInvariants, Set outputTracker, boolean active, boolean mustSupport, RenderingContext rc) throws IOException, FHIRException { + return generateTable(defFile, profile, diff, imageFolder, inlineGraphics, profileBaseFileName, snapshot, corePath, imagePath, logicalModel, allInvariants, outputTracker, active, mustSupport, rc, ""); + } + + public XhtmlNode generateTable(String defFile, StructureDefinition profile, boolean diff, String imageFolder, boolean inlineGraphics, String profileBaseFileName, boolean snapshot, String corePath, String imagePath, + boolean logicalModel, boolean allInvariants, Set outputTracker, boolean active, boolean mustSupport, RenderingContext rc, String anchorPrefix) throws IOException, FHIRException { assert(diff != snapshot);// check it's ok to get rid of one of these HierarchicalTableGenerator gen = new HierarchicalTableGenerator(imageFolder, inlineGraphics, true); gen.setTranslator(getTranslator()); @@ -4030,7 +4056,7 @@ public class ProfileUtilities extends TranslatingUtilities { if (diff) { insertMissingSparseElements(list); } - genElement(defFile == null ? null : defFile+"#", gen, model.getRows(), list.get(0), list, profiles, diff, profileBaseFileName, null, snapshot, corePath, imagePath, true, logicalModel, profile.getDerivation() == TypeDerivationRule.CONSTRAINT && usesMustSupport(list), allInvariants, null, mustSupport, rc); + genElement(defFile == null ? null : defFile+"#", gen, model.getRows(), list.get(0), list, profiles, diff, profileBaseFileName, null, snapshot, corePath, imagePath, true, logicalModel, profile.getDerivation() == TypeDerivationRule.CONSTRAINT && usesMustSupport(list), allInvariants, null, mustSupport, rc, anchorPrefix); try { return gen.generate(model, imagePath, 0, outputTracker); } catch (org.hl7.fhir.exceptions.FHIRException e) { @@ -4126,7 +4152,7 @@ public class ProfileUtilities extends TranslatingUtilities { private Row genElement(String defPath, HierarchicalTableGenerator gen, List rows, ElementDefinition element, List all, List profiles, boolean showMissing, String profileBaseFileName, Boolean extensions, - boolean snapshot, String corePath, String imagePath, boolean root, boolean logicalModel, boolean isConstraintMode, boolean allInvariants, Row slicingRow, boolean mustSupport, RenderingContext rc) throws IOException, FHIRException { + boolean snapshot, String corePath, String imagePath, boolean root, boolean logicalModel, boolean isConstraintMode, boolean allInvariants, Row slicingRow, boolean mustSupport, RenderingContext rc, String anchorPrefix) throws IOException, FHIRException { Row originalRow = slicingRow; StructureDefinition profile = profiles == null ? null : profiles.get(profiles.size()-1); Row typesRow = null; @@ -4188,7 +4214,7 @@ public class ProfileUtilities extends TranslatingUtilities { row.setOpacity("0.5"); } UnusedTracker used = new UnusedTracker(); - String ref = defPath == null ? null : defPath + element.getId(); + String ref = defPath == null ? null : defPath + anchorPrefix + element.getId(); String sName = tail(element.getPath()); if (element.hasSliceName()) sName = sName +":"+element.getSliceName(); @@ -4263,7 +4289,7 @@ public class ProfileUtilities extends TranslatingUtilities { Row childRow = chooseChildRowByGroup(gen, currRow, groups, child, element, isConstraintMode); if (logicalModel || !child.getPath().endsWith(".id") || (child.getPath().endsWith(".id") && (profile != null) && (profile.getDerivation() == TypeDerivationRule.CONSTRAINT))) { - currRow = genElement(defPath, gen, childRow.getSubRows(), child, all, profiles, showMissing, profileBaseFileName, isExtension, snapshot, corePath, imagePath, false, logicalModel, isConstraintMode, allInvariants, currRow, mustSupport, rc); + currRow = genElement(defPath, gen, childRow.getSubRows(), child, all, profiles, showMissing, profileBaseFileName, isExtension, snapshot, corePath, imagePath, false, logicalModel, isConstraintMode, allInvariants, currRow, mustSupport, rc, anchorPrefix); } } } diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/renderers/StructureDefinitionRenderer.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/renderers/StructureDefinitionRenderer.java index 480686c89..a5962056e 100644 --- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/renderers/StructureDefinitionRenderer.java +++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/renderers/StructureDefinitionRenderer.java @@ -28,7 +28,7 @@ public class StructureDefinitionRenderer extends ResourceRenderer { } public boolean render(XhtmlNode x, StructureDefinition sd) throws FHIRFormatError, DefinitionException, IOException { - x.getChildNodes().add(context.getProfileUtilities().generateTable(context.getDefinitionsTarget(), sd, true, context.getDestDir(), false, sd.getId(), false, context.getSpecificationLink(), "", false, false, null, false, false, context)); + x.getChildNodes().add(context.getProfileUtilities().generateTable(context.getDefinitionsTarget(), sd, true, context.getDestDir(), false, sd.getId(), false, context.getSpecificationLink(), "", false, false, null, false, false, context, "")); return true; } From c892d65a8db08427d1de497fc7f5519e28dcf642 Mon Sep 17 00:00:00 2001 From: Lloyd McKenzie Date: Tue, 27 Sep 2022 12:40:59 -0600 Subject: [PATCH 2/3] Clean-up after testing --- .../fhir/r5/conformance/ProfileUtilities.java | 35 +++++++++++-------- 1 file changed, 21 insertions(+), 14 deletions(-) diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java index 5d707260e..a4c79f0ae 100644 --- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java +++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java @@ -4042,19 +4042,9 @@ public class ProfileUtilities extends TranslatingUtilities { return generateTable(defFile, profile, diff, imageFolder, inlineGraphics, profileBaseFileName, snapshot, corePath, imagePath, logicalModel, allInvariants, outputTracker, active, mustSupport, rc, ""); } - public XhtmlNode generateTable(String defFile, StructureDefinition profile, boolean diff, String imageFolder, boolean inlineGraphics, String profileBaseFileName, boolean snapshot, String corePath, String imagePath, - boolean logicalModel, boolean allInvariants, Set outputTracker, boolean active, boolean mustSupport, RenderingContext rc, String anchorPrefix) throws IOException, FHIRException { - assert(diff != snapshot);// check it's ok to get rid of one of these - HierarchicalTableGenerator gen = new HierarchicalTableGenerator(imageFolder, inlineGraphics, true); - gen.setTranslator(getTranslator()); - TableModel model = gen.initNormalTable(corePath, false, true, profile.getId()+(diff ? "d" : "s"), active); + public List supplementMissingDiffElements(StructureDefinition profile) { List list = new ArrayList<>(); - if (diff) - list.addAll(profile.getDifferential().getElement()); - else - list.addAll(profile.getSnapshot().getElement()); - List profiles = new ArrayList(); - profiles.add(profile); + list.addAll(profile.getDifferential().getElement()); if (list.isEmpty()) { ElementDefinition root = new ElementDefinition().setPath(profile.getType()); root.setId(profile.getType()); @@ -4066,9 +4056,26 @@ public class ProfileUtilities extends TranslatingUtilities { list.add(0, root); } } - if (diff) { - insertMissingSparseElements(list); + insertMissingSparseElements(list); + return list; + } + + public XhtmlNode generateTable(String defFile, StructureDefinition profile, boolean diff, String imageFolder, boolean inlineGraphics, String profileBaseFileName, boolean snapshot, String corePath, String imagePath, + boolean logicalModel, boolean allInvariants, Set outputTracker, boolean active, boolean mustSupport, RenderingContext rc, String anchorPrefix) throws IOException, FHIRException { + assert(diff != snapshot);// check it's ok to get rid of one of these + HierarchicalTableGenerator gen = new HierarchicalTableGenerator(imageFolder, inlineGraphics, true); + gen.setTranslator(getTranslator()); + TableModel model = gen.initNormalTable(corePath, false, true, profile.getId()+(diff ? "d" : "s"), active); + List list; + if (diff) + list = supplementMissingDiffElements(profile); + else { + list = new ArrayList<>(); + list.addAll(profile.getSnapshot().getElement()); } + List profiles = new ArrayList(); + profiles.add(profile); + genElement(defFile == null ? null : defFile+"#", gen, model.getRows(), list.get(0), list, profiles, diff, profileBaseFileName, null, snapshot, corePath, imagePath, true, logicalModel, profile.getDerivation() == TypeDerivationRule.CONSTRAINT && usesMustSupport(list), allInvariants, null, mustSupport, rc, anchorPrefix); try { return gen.generate(model, imagePath, 0, outputTracker); From 1bb4c27651ac6040f037e9d302cdda5a3187ba84 Mon Sep 17 00:00:00 2001 From: Lloyd McKenzie Date: Thu, 6 Oct 2022 11:36:06 -0600 Subject: [PATCH 3/3] Corrections to binding rendering; don't cache SD elements in HAPI --- .../AdditionalBindingsRenderer.java | 47 ++++++++++++++----- .../fhir/r5/conformance/ProfileUtilities.java | 17 ------- 2 files changed, 35 insertions(+), 29 deletions(-) diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java index df10ee082..65ac7a124 100644 --- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java +++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/AdditionalBindingsRenderer.java @@ -32,14 +32,15 @@ public class AdditionalBindingsRenderer { private String valueSet; private String doco; private UsageContext usage; - private boolean any; - private boolean unchanged; - private boolean matched; + private boolean any = false; + private boolean isUnchanged = false; + private boolean matched = false; + private boolean removed = false; private AdditionalBindingDetail compare; private int count = 1; private String getKey() { // Todo: Consider extending this with content from usageContext if purpose isn't sufficiently differentiating - return purpose; + return purpose + Integer.toString(count); } private void incrementCount() { count++; @@ -51,9 +52,21 @@ public class AdditionalBindingsRenderer { private boolean alreadyMatched() { return matched; } + public boolean unchanged() { + if (!isUnchanged) + return false; + if (compare==null) + return true; + isUnchanged = true; + isUnchanged = isUnchanged && ((purpose==null && compare.purpose==null) || purpose.equals(compare.purpose)); + isUnchanged = isUnchanged && ((valueSet==null && compare.valueSet==null) || valueSet.equals(compare.valueSet)); + isUnchanged = isUnchanged && ((doco==null && compare.doco==null) || doco.equals(compare.doco)); + isUnchanged = isUnchanged && ((usage==null && compare.usage==null) || usage.equals(compare.usage)); + return isUnchanged; + } } - private static String STYLE_UNCHANGED = "font-color: darkgray;"; + private static String STYLE_UNCHANGED = "opacity: 0.5;"; private static String STYLE_REMOVED = STYLE_UNCHANGED + "text-decoration: line-through;"; private List bindings = new ArrayList<>(); @@ -86,12 +99,12 @@ public class AdditionalBindingsRenderer { abr.purpose = label; abr.valueSet = ext.getValue().primitiveValue(); if (compare) { - abr.unchanged = compExt!=null && ext.getValue().primitiveValue().equals(compExt.getValue().primitiveValue()); + abr.isUnchanged = compExt!=null && ext.getValue().primitiveValue().equals(compExt.getValue().primitiveValue()); abr.compare = new AdditionalBindingDetail(); abr.compare.valueSet = compExt==null ? null : compExt.getValue().primitiveValue(); } else { - abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); + abr.isUnchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); } bindings.add(abr); } @@ -113,8 +126,9 @@ public class AdditionalBindingsRenderer { if (compare && compList!=null) { for (Extension ext : compList) { AdditionalBindingDetail abr = additionalBinding(ext); - while (compBindings.containsKey(abr.getKey())) + if (compBindings.containsKey(abr.getKey())) { abr.incrementCount(); + } compBindings.put(abr.getKey(), abr); } } @@ -127,12 +141,19 @@ public class AdditionalBindingsRenderer { match = compBindings.get(abr.getKey()); if (abr.alreadyMatched()) abr.incrementCount(); - } while (match!=null && !match.alreadyMatched()); + } while (match!=null && abr.alreadyMatched()); if (match!=null) abr.setCompare(match); + bindings.add(abr); + if (abr.compare!=null) + compBindings.remove(abr.compare.getKey()); } else bindings.add(abr); } + for (AdditionalBindingDetail b: compBindings.values()) { + b.removed = true; + bindings.add(b); + } } protected AdditionalBindingDetail additionalBinding(Extension ext) { @@ -142,7 +163,7 @@ public class AdditionalBindingsRenderer { abr.doco = ext.getExtensionString("documentation"); abr.usage = (ext.hasExtension("usage")) && ext.getExtensionByUrl("usage").hasValueUsageContext() ? ext.getExtensionByUrl("usage").getValueUsageContext() : null; abr.any = "any".equals(ext.getExtensionString("scope")); - abr.unchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); + abr.isUnchanged = ext.hasUserData(ProfileUtilities.DERIVATION_EQUALS); return abr; } @@ -192,8 +213,10 @@ public class AdditionalBindingsRenderer { } for (AdditionalBindingDetail binding : bindings) { tr = new XhtmlNode(NodeType.Element, "tr"); - if (binding.unchanged) { - tr.style("opacity: 0.5"); + if (binding.unchanged()) { + tr.style(STYLE_REMOVED); + } else if (binding.removed) { + tr.style(STYLE_REMOVED); } children.add(tr); BindingResolution br = pkp == null ? makeNullBr(binding) : pkp.resolveBinding(profile, binding.valueSet, path); diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java index a4c79f0ae..dd41ac3f0 100644 --- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java +++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/conformance/ProfileUtilities.java @@ -355,7 +355,6 @@ public class ProfileUtilities extends TranslatingUtilities { private boolean wantFixDifferentialFirstElementType; private Set masterSourceFileNames; private Map> childMapCache = new HashMap<>(); - private Map> sdMapCache = new HashMap<>(); public ProfileUtilities(IWorkerContext context, List messages, ProfileKnowledgeProvider pkp, FHIRPathEngine fpe) { super(); @@ -432,22 +431,6 @@ public class ProfileUtilities extends TranslatingUtilities { String getLinkForUrl(String corePath, String s); } - public ElementDefinition getElementById(String structureCanonical, String id) { - Map sdCache = sdMapCache.get(structureCanonical); - - if (sdCache == null) { - StructureDefinition sd = (StructureDefinition) context.fetchResource(StructureDefinition.class, structureCanonical); - if (sd==null) - throw new FHIRException("Unable to retrieve StructureDefinition with URL " + structureCanonical); - sdCache = new HashMap(); - sdMapCache.put(structureCanonical, sdCache); - for (ElementDefinition e : sd.getSnapshot().getElement()) { - sdCache.put(e.getId(), e); - } - } - return sdCache.get(id); - } - public List getChildMap(StructureDefinition profile, ElementDefinition element) throws DefinitionException { if (childMapCache .containsKey(element)) { return childMapCache.get(element);