From a888b3ef2372ccc01dec713c712138072b299b65 Mon Sep 17 00:00:00 2001 From: Daniel Danis Date: Fri, 1 Mar 2024 15:15:05 -0500 Subject: [PATCH 1/3] In genotype-aware mode, do not include the differential diagnoses in HTML and TSV reports, unless there is a predicted deleterious variant in the associated gene. --- docs/running.rst | 3 +- .../lirical/cli/cmd/OutputCommand.java | 3 +- .../likelihoodratio/GenotypeLrMatchType.java | 19 ++++++++++- .../lirical/core/output/OutputOptions.java | 14 +++++++- .../GenotypeLrMatchTypeTest.java | 33 +++++++++++++++++++ .../lirical/io/output/HtmlTemplate.java | 9 +++-- .../lirical/io/output/LiricalTemplate.java | 19 +++++++++++ .../lirical/io/output/SparklinePacket.java | 2 ++ .../lirical/io/output/TsvTemplate.java | 1 + 9 files changed, 97 insertions(+), 6 deletions(-) create mode 100644 lirical-core/src/test/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchTypeTest.java diff --git a/docs/running.rst b/docs/running.rst index 3345cbd18..7ce2fe5ab 100644 --- a/docs/running.rst +++ b/docs/running.rst @@ -76,7 +76,8 @@ The configuration options tweak the analysis. * ``--sdwndv``: show diseases even if no deleterious variants are found in the gene associated with the disease. The option is a flag (takes no value) and its presence will lead to showing *all* diseases, even those with no deleterious variants. - Only applicable to the HTML report when running with a VCF file (genotype-aware mode). + Only applicable to the HTML and TSV reports when running with a VCF file (genotype-aware mode). + The JSON report will include *all* diseases all the time. * ``--transcript-db``: transcript database (default: ``RefSeq``), see :ref:`rsttx-dbs` for more info. * ``--use-orphanet``: use `Orphanet `_ annotations (default: ``false``). * ``--strict``: use strict penalties if the genotype does not match the disease model diff --git a/lirical-cli/src/main/java/org/monarchinitiative/lirical/cli/cmd/OutputCommand.java b/lirical-cli/src/main/java/org/monarchinitiative/lirical/cli/cmd/OutputCommand.java index 3e3801310..e78906ff4 100644 --- a/lirical-cli/src/main/java/org/monarchinitiative/lirical/cli/cmd/OutputCommand.java +++ b/lirical-cli/src/main/java/org/monarchinitiative/lirical/cli/cmd/OutputCommand.java @@ -72,6 +72,7 @@ protected OutputOptions createOutputOptions(String prefix) { LrThreshold lrThreshold = output.lrThreshold == null ? LrThreshold.notInitialized() : LrThreshold.setToUserDefinedThreshold(output.lrThreshold); MinDiagnosisCount minDiagnosisCount = output.minDifferentialsToShow == null ? MinDiagnosisCount.notInitialized() : MinDiagnosisCount.setToUserDefinedMinCount(output.minDifferentialsToShow); return new OutputOptions(lrThreshold, minDiagnosisCount, runConfiguration.pathogenicityThreshold, - output.displayAllVariants, output.outdir, prefix); + output.displayAllVariants, runConfiguration.showDiseasesWithNoDeleteriousVariants, + output.outdir, prefix); } } diff --git a/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java b/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java index 2cb60c4b7..da7b4fb73 100644 --- a/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java +++ b/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java @@ -78,5 +78,22 @@ public enum GenotypeLrMatchType { */ @Deprecated(forRemoval = true) // REMOVE(v3.0.0) - UNKNOWN + UNKNOWN; + + /** + * Returns {@code true} if the genotype LR match indicates that the enclosing {@link GenotypeLrWithExplanation} + * was created for a gene with some deleterious variants in the gene were observed or if the LR was produced using + * the LIRICAL genotype model, etc... + *

+ * Returns {@code false} if the genotype LR represents state with no deleterious variants in a gene. + */ + public boolean hasDeleteriousVariants() { + return switch (this) { + case NO_VARIANTS_DETECTED_AD, NO_VARIANTS_DETECTED_AR, UNKNOWN -> false; + case ONE_P_OR_LP_CLINVAR_ALLELE_IN_AD, LIRICAL_GT_MODEL, + ONE_DELETERIOUS_CLINVAR_VARIANT_IN_AD, TWO_DELETERIOUS_CLINVAR_VARIANTS_IN_AR, + TWO_P_OR_LP_CLINVAR_ALLELES_IN_AR, ONE_DELETERIOUS_VARIANT_IN_AR, + HIGH_NUMBER_OF_OBSERVED_PREDICTED_PATHOGENIC_VARIANTS -> true; + }; + } } diff --git a/lirical-core/src/main/java/org/monarchinitiative/lirical/core/output/OutputOptions.java b/lirical-core/src/main/java/org/monarchinitiative/lirical/core/output/OutputOptions.java index cccf2465f..38b6765e3 100644 --- a/lirical-core/src/main/java/org/monarchinitiative/lirical/core/output/OutputOptions.java +++ b/lirical-core/src/main/java/org/monarchinitiative/lirical/core/output/OutputOptions.java @@ -7,7 +7,11 @@ public final class OutputOptions { private final LrThreshold lrThreshold; private final MinDiagnosisCount minDiagnosisCount; private final float pathogenicityThreshold; + // If set to true, all variants, pathogenic, VUSs, and benign will be shown in the report. private final boolean displayAllVariants; + // If set to true, all differential diagnoses, even those with no deleterious variants in the associated gene + // will be shown in the report. + private final boolean showDiseasesWithNoDeleteriousVariants; private final Path outputDirectory; private final String prefix; @@ -15,12 +19,14 @@ public OutputOptions(LrThreshold lrThreshold, MinDiagnosisCount minDiagnosisCount, float pathogenicityThreshold, boolean displayAllVariants, + boolean showDiseasesWithNoDeleteriousVariants, Path outputDirectory, String prefix) { this.lrThreshold = lrThreshold; this.minDiagnosisCount = minDiagnosisCount; this.pathogenicityThreshold = pathogenicityThreshold; this.displayAllVariants = displayAllVariants; + this.showDiseasesWithNoDeleteriousVariants = showDiseasesWithNoDeleteriousVariants; this.outputDirectory = outputDirectory; this.prefix = prefix; } @@ -41,6 +47,10 @@ public boolean displayAllVariants() { return displayAllVariants; } + public boolean showDiseasesWithNoDeleteriousVariants() { + return showDiseasesWithNoDeleteriousVariants; + } + public Path outputDirectory() { return outputDirectory; } @@ -58,13 +68,14 @@ public boolean equals(Object obj) { Objects.equals(this.minDiagnosisCount, that.minDiagnosisCount) && Float.floatToIntBits(this.pathogenicityThreshold) == Float.floatToIntBits(that.pathogenicityThreshold) && this.displayAllVariants == that.displayAllVariants && + this.showDiseasesWithNoDeleteriousVariants == that.showDiseasesWithNoDeleteriousVariants && Objects.equals(this.outputDirectory, that.outputDirectory) && Objects.equals(this.prefix, that.prefix); } @Override public int hashCode() { - return Objects.hash(lrThreshold, minDiagnosisCount, pathogenicityThreshold, displayAllVariants, outputDirectory, prefix); + return Objects.hash(lrThreshold, minDiagnosisCount, pathogenicityThreshold, displayAllVariants, showDiseasesWithNoDeleteriousVariants, outputDirectory, prefix); } @Override @@ -74,6 +85,7 @@ public String toString() { "minDiagnosisCount=" + minDiagnosisCount + ", " + "pathogenicityThreshold=" + pathogenicityThreshold + ", " + "displayAllVariants=" + displayAllVariants + ", " + + "showDiseasesWithNoDeleteriousVariants=" + showDiseasesWithNoDeleteriousVariants + ", " + "outputDirectory=" + outputDirectory + ", " + "prefix=" + prefix + ']'; } diff --git a/lirical-core/src/test/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchTypeTest.java b/lirical-core/src/test/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchTypeTest.java new file mode 100644 index 000000000..4528894b5 --- /dev/null +++ b/lirical-core/src/test/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchTypeTest.java @@ -0,0 +1,33 @@ +package org.monarchinitiative.lirical.core.likelihoodratio; + +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.CsvSource; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; + +public class GenotypeLrMatchTypeTest { + + @ParameterizedTest + @CsvSource( + { + "NO_VARIANTS_DETECTED_AD, false", + "NO_VARIANTS_DETECTED_AR, false", + "UNKNOWN, false", + "ONE_P_OR_LP_CLINVAR_ALLELE_IN_AD, true", + "LIRICAL_GT_MODEL, true", + "ONE_DELETERIOUS_CLINVAR_VARIANT_IN_AD, true", + "TWO_DELETERIOUS_CLINVAR_VARIANTS_IN_AR, true", + "TWO_P_OR_LP_CLINVAR_ALLELES_IN_AR, true", + "ONE_DELETERIOUS_VARIANT_IN_AR, true", + "HIGH_NUMBER_OF_OBSERVED_PREDICTED_PATHOGENIC_VARIANTS, true", + } + ) + public void hasDeleteriousVariants(String value, boolean expected) { + // We depend on correctness of `hasDeleteriousVariants` in HTML and TSV report generators, + // where we may choose to only show the diff dgs with a deleterious variant. + GenotypeLrMatchType glmt = GenotypeLrMatchType.valueOf(value); + + assertThat(glmt.hasDeleteriousVariants(), equalTo(expected)); + } +} \ No newline at end of file diff --git a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/HtmlTemplate.java b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/HtmlTemplate.java index faaf94390..fdfd5a554 100644 --- a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/HtmlTemplate.java +++ b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/HtmlTemplate.java @@ -6,7 +6,10 @@ import org.monarchinitiative.lirical.core.analysis.AnalysisResults; import org.monarchinitiative.lirical.core.likelihoodratio.GenotypeLrWithExplanation; import org.monarchinitiative.lirical.core.model.Gene2Genotype; -import org.monarchinitiative.lirical.core.output.*; +import org.monarchinitiative.lirical.core.output.AnalysisResultsMetadata; +import org.monarchinitiative.lirical.core.output.LrThreshold; +import org.monarchinitiative.lirical.core.output.MinDiagnosisCount; +import org.monarchinitiative.lirical.core.output.OutputOptions; import org.monarchinitiative.lirical.io.output.svg.Lr2Svg; import org.monarchinitiative.phenol.annotations.formats.GeneIdentifier; import org.monarchinitiative.phenol.annotations.formats.hpo.HpoDisease; @@ -70,7 +73,8 @@ public class HtmlTemplate extends LiricalTemplate { cfg.setClassForTemplateLoading(HtmlTemplate.class, ""); templateData.put("postprobthreshold", String.format("%.1f%%", 100 * this.lrThreshold.getThreshold())); int N = totalDetailedDiagnosesToShow(analysisResults); - List sparklinePackets = SparklinePacket.sparklineFactory(analysisResults, diseases, hpo, N); + List sparklinePackets = SparklinePacket.sparklineFactory(analysisResults, diseases, hpo, + outputOptions.showDiseasesWithNoDeleteriousVariants(), N); this.templateData.put("sparkline", sparklinePackets); if (symbolsWithoutGeneIds != null && !symbolsWithoutGeneIds.isEmpty()) { this.templateData.put("geneSymbolsWithoutIds", symbolsWithoutGeneIds); @@ -81,6 +85,7 @@ public class HtmlTemplate extends LiricalTemplate { List improbableDifferentials = new ArrayList<>(); AtomicInteger rank = new AtomicInteger(); analysisResults.resultsWithDescendingPostTestProbability().sequential() + .filter(handleCasesWithNoDeleteriousVariants(outputOptions.showDiseasesWithNoDeleteriousVariants())) .forEachOrdered(result -> { int current = rank.incrementAndGet(); diff --git a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/LiricalTemplate.java b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/LiricalTemplate.java index 094364ec8..d096ff950 100644 --- a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/LiricalTemplate.java +++ b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/LiricalTemplate.java @@ -3,6 +3,7 @@ import freemarker.template.Configuration; import freemarker.template.Version; import org.monarchinitiative.lirical.core.analysis.AnalysisData; +import org.monarchinitiative.lirical.core.analysis.TestResult; import org.monarchinitiative.lirical.core.exception.LiricalRuntimeException; import org.monarchinitiative.lirical.core.model.Gene2Genotype; import org.monarchinitiative.lirical.core.model.LiricalVariant; @@ -17,6 +18,7 @@ import java.nio.file.Path; import java.util.*; import java.util.function.Function; +import java.util.function.Predicate; import java.util.stream.Collectors; /** @@ -126,4 +128,21 @@ private boolean isPassingPathogenicThreshold(LiricalVariant lv) { return lv.pathogenicityScore().orElse(0f) >= pathogenicityThreshold; } + /** + * Include the result from the differential diagnoses + * IF we are interested the diseases with no deleterious variants + * OR if the genotype LR represents a situation where some pathogenic variants were found in the associated gene + * OR if genotype LR is missing (phenotype only mode). + * + * @param showDiseasesWithNoDeleteriousVariants set to {@code true} if you wish to see the differential diagnoses + * with no deleterious variants regardless of anything. + */ + static Predicate handleCasesWithNoDeleteriousVariants(boolean showDiseasesWithNoDeleteriousVariants) { + return r -> + showDiseasesWithNoDeleteriousVariants + || r.genotypeLr() + .map(g -> g.matchType().hasDeleteriousVariants()) + .orElse(true); + } + } diff --git a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/SparklinePacket.java b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/SparklinePacket.java index 92b549edd..a8b464c19 100644 --- a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/SparklinePacket.java +++ b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/SparklinePacket.java @@ -41,6 +41,7 @@ public class SparklinePacket { public static List sparklineFactory(AnalysisResults results, HpoDiseases diseases, MinimalOntology ontology, + boolean showDiseasesWithNoDeleteriousVariants, int N) { if (results.isEmpty()) return List.of(); @@ -53,6 +54,7 @@ public static List sparklineFactory(AnalysisResults results, Map diseaseById = diseases.diseaseById(); Sparkline2Svg sparkline2Svg = new Sparkline2Svg(topResult, true, ontology); results.resultsWithDescendingPostTestProbability() + .filter(LiricalTemplate.handleCasesWithNoDeleteriousVariants(showDiseasesWithNoDeleteriousVariants)) .limit(N) .forEachOrdered(result -> { double posttestProb = result.posttestProbability(); diff --git a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/TsvTemplate.java b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/TsvTemplate.java index bd18cef51..fb0d084fd 100644 --- a/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/TsvTemplate.java +++ b/lirical-io/src/main/java/org/monarchinitiative/lirical/io/output/TsvTemplate.java @@ -43,6 +43,7 @@ public class TsvTemplate extends LiricalTemplate { Map diseaseById = diseases.diseaseById(); List diff = new ArrayList<>(); analysisResults.resultsWithDescendingPostTestProbability().sequential() + .filter(handleCasesWithNoDeleteriousVariants(outputOptions.showDiseasesWithNoDeleteriousVariants())) .forEachOrdered(result -> { int current = rank.incrementAndGet(); List variants = result.genotypeLr() From 47a192a714f2e3be05427f76af275309a7b0809a Mon Sep 17 00:00:00 2001 From: Daniel Danis Date: Fri, 1 Mar 2024 15:22:48 -0500 Subject: [PATCH 2/3] Update Javadoc. --- .../lirical/core/likelihoodratio/GenotypeLrMatchType.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java b/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java index da7b4fb73..e79547adb 100644 --- a/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java +++ b/lirical-core/src/main/java/org/monarchinitiative/lirical/core/likelihoodratio/GenotypeLrMatchType.java @@ -82,8 +82,8 @@ public enum GenotypeLrMatchType { /** * Returns {@code true} if the genotype LR match indicates that the enclosing {@link GenotypeLrWithExplanation} - * was created for a gene with some deleterious variants in the gene were observed or if the LR was produced using - * the LIRICAL genotype model, etc... + * was created for a gene that contains deleterious variants. This includes the genotype LRs generated + * for ClinVar variants, using the LIRICAL genotype model, etc... *

* Returns {@code false} if the genotype LR represents state with no deleterious variants in a gene. */ From 03b0bee4b1db81f7b283469a65f3f0560c7b4028 Mon Sep 17 00:00:00 2001 From: Daniel Danis Date: Fri, 1 Mar 2024 15:26:32 -0500 Subject: [PATCH 3/3] Fix the test. --- .../lirical/io/output/JsonAnalysisResultWriterTest.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lirical-io/src/test/java/org/monarchinitiative/lirical/io/output/JsonAnalysisResultWriterTest.java b/lirical-io/src/test/java/org/monarchinitiative/lirical/io/output/JsonAnalysisResultWriterTest.java index 1db72d236..da336e35b 100644 --- a/lirical-io/src/test/java/org/monarchinitiative/lirical/io/output/JsonAnalysisResultWriterTest.java +++ b/lirical-io/src/test/java/org/monarchinitiative/lirical/io/output/JsonAnalysisResultWriterTest.java @@ -63,7 +63,7 @@ public void writeExampleAnalysisOutput() throws Exception { AnalysisResults results = createTestAnalysisResults(); AnalysisResultsMetadata metadata = createTestMetadata(); Path current = Path.of("."); - OutputOptions oo = new OutputOptions(LrThreshold.notInitialized(), MinDiagnosisCount.setToUserDefinedMinCount(2), 1.f, true, current, "test"); + OutputOptions oo = new OutputOptions(LrThreshold.notInitialized(), MinDiagnosisCount.setToUserDefinedMinCount(2), 1.f, true, false, current, "test"); writer.process(analysisData, results, metadata, oo); String output = readFileIntoString(TEST_OUTPUT);