1 Contrasts

zymodeme_keeper <- list(
    "zymodeme" = c("z23", "z22"))
susceptibility_keepers <- list(
    "resistant_sensitive" = c("resistant", "sensitive"),
    "resistant_ambiguous" = c("resistant", "ambiguous"),
    "sensitive_ambiguous" = c("sensitive", "ambiguous"))

1.1 Zymodeme enzyme gene IDs

Najib read me an email listing off the gene names associated with the zymodeme classification. I took those names and cross referenced them against the Leishmania panamensis gene annotations and found the following:

They are:

  1. ALAT: LPAL13_120010900 – alanine aminotransferase
  2. ASAT: LPAL13_340013000 – aspartate aminotransferase
  3. G6PD: LPAL13_000054100 – glucase-6-phosphate 1-dehydrogenase
  4. NH: LPAL13_14006100, LPAL13_180018500 – inosine-guanine nucleoside hydrolase
  5. MPI: LPAL13_320022300 (maybe) – mannose phosphate isomerase (I chose phosphomannose isomerase)

Given these 6 gene IDs (NH has two gene IDs associated with it), I can do some looking for specific differences among the various samples.

1.1.1 Expression levels of zymodeme genes

The following creates a colorspace (red to green) heatmap showing the observed expression of these genes in every sample.

my_genes <- c("LPAL13_120010900", "LPAL13_340013000", "LPAL13_000054100",
              "LPAL13_140006100", "LPAL13_180018500", "LPAL13_320022300",
              "other")
my_names <- c("ALAT", "ASAT", "G6PD", "NHv1", "NHv2", "MPI", "other")

zymo_six_genes <- exclude_genes_expt(lp_two_strains, ids = my_genes, method = "keep")
## Note, I renamed this to subset_genes().
## remove_genes_expt(), before removal, there were 8710 genes, now there are 6.
## There are 93 samples which kept less than 90 percent counts.
## TMRC20001 TMRC20065 TMRC20005 TMRC20066 TMRC20039 TMRC20037 TMRC20038 TMRC20067 TMRC20068 TMRC20041 TMRC20015 TMRC20009 TMRC20010 TMRC20016 
##   0.12101   0.12835   0.13438   0.10804   0.13756   0.11451   0.11446   0.10858   0.11269   0.12840   0.11072   0.11376   0.10292   0.10451 
## TMRC20011 TMRC20012 TMRC20013 TMRC20017 TMRC20014 TMRC20018 TMRC20019 TMRC20070 TMRC20020 TMRC20021 TMRC20022 TMRC20024 TMRC20036 TMRC20069 
##   0.10967   0.11676   0.11865   0.10594   0.10591   0.11517   0.12027   0.11179   0.11100   0.10764   0.12952   0.11481   0.12100   0.11631 
## TMRC20033 TMRC20026 TMRC20031 TMRC20076 TMRC20073 TMRC20055 TMRC20079 TMRC20071 TMRC20078 TMRC20094 TMRC20042 TMRC20058 TMRC20072 TMRC20059 
##   0.11188   0.13633   0.10013   0.12403   0.12398   0.13783   0.12639   0.12003   0.13420   0.12021   0.13766   0.11900   0.14675   0.10984 
## TMRC20048 TMRC20057 TMRC20088 TMRC20056 TMRC20060 TMRC20077 TMRC20074 TMRC20063 TMRC20053 TMRC20052 TMRC20064 TMRC20075 TMRC20051 TMRC20050 
##   0.10679   0.13443   0.12936   0.13510   0.10882   0.12810   0.12494   0.12239   0.12145   0.11690   0.12051   0.11261   0.13149   0.11526 
## TMRC20049 TMRC20062 TMRC20110 TMRC20080 TMRC20043 TMRC20083 TMRC20054 TMRC20085 TMRC20046 TMRC20093 TMRC20089 TMRC20047 TMRC20090 TMRC20044 
##   0.14480   0.13429   0.13812   0.12120   0.11306   0.12235   0.12723   0.12316   0.13182   0.13588   0.11884   0.12391   0.11566   0.13691 
## TMRC20045 TMRC20105 TMRC20108 TMRC20109 TMRC20098 TMRC20096 TMRC20097 TMRC20101 TMRC20092 TMRC20082 TMRC20102 TMRC20099 TMRC20100 TMRC20091 
##   0.12812   0.12219   0.11601   0.11684   0.11771   0.11364   0.11706   0.11784   0.11465   0.10358   0.11399   0.11888   0.10820   0.12935 
## TMRC20084 TMRC20087 TMRC20103 TMRC20104 TMRC20086 TMRC20107 TMRC20081 TMRC20106 TMRC20095 
##   0.11273   0.12564   0.13704   0.11723   0.10752   0.09364   0.10335   0.09894   0.06566
strain_norm <- normalize_expt(zymo_six_genes, convert="rpkm", filter=TRUE, transform="log2")
## Removing 0 low-count genes (6 remaining).
zymo_heatmap <- plot_sample_heatmap(strain_norm, row_label = my_names)
zymo_heatmap

lp_norm <- normalize_expt(lp_two_strains, filter=TRUE, convert="rpkm",
                          norm="quant", transform="log2")
## Removing 151 low-count genes (8559 remaining).
## There appear to be 24 genes without a length.
## transform_counts: Found 103 values equal to 0, adding 1 to the matrix.
zymo_heatmap_all <- plot_sample_heatmap(lp_norm)
zymo_heatmap_all

1.2 Compare to highly expressed, variant genes

I want to compare the above heatmap with one which is comprised of all genes with some ‘significantly high’ expression value and also a not-negligible coefficient of variance.

zymo_high_genes <- normalize_expt(lp_two_strains, filter = "cv", cv_min = 0.9)
## Removing 5564 low-count genes (3146 remaining).
high_strain_norm <- normalize_expt(zymo_high_genes, convert = "rpkm",
                                   norm = "quant", transform = "log2")
## There appear to be 105 genes without a length.
## transform_counts: Found 238 values equal to 0, adding 1 to the matrix.
zymo_heatmap <- plot_sample_heatmap(high_strain_norm, row_label = my_names)
zymo_heatmap

I think this plot suggests that the difference between the two primary strains is not really one of a few specific genes, but instead a global pattern.

2 Zymodeme differential expression

2.1 No attempt at batch estimation

two_zymo <- set_expt_conditions(lp_two_strains, fact = "zymodemecategorical") %>%
  subset_expt(subset = "condition!='unknown'")
## The numbers of samples by condition are:
## 
## z21 z22 z23 z24 
##   7  43  41   2
## subset_expt(): There were 93, now there are 93 samples.
zymo_de_nobatch <- all_pairwise(two_zymo, filter = TRUE, model_batch = FALSE)
## 
## z21 z22 z23 z24 
##   7  43  41   2

zymo_table_nobatch <- combine_de_tables(
    zymo_de_nobatch, keepers = zymodeme_keeper,
    rda = glue("rda/zymo_tables_nobatch-v{ver}.rda"),
    excel = glue("excel/zymo_tables_nobatch-v{ver}.xlsx"))
## Deleting the file excel/zymo_tables_nobatch-v202305.xlsx before writing the tables.
## Adding venn plots for zymodeme.
## Saving de result as zymo_tables_nobatch-v202305 to rda/zymo_tables_nobatch-v202305.rda.
zymo_sig_nobatch <- extract_significant_genes(
    zymo_table_nobatch,
    according_to = "deseq", current_id = "GID", required_id = "GID",
    gmt = glue("gmt/zymodeme_nobatch-v{ver}.gmt"),
    excel = glue("excel/zymo_sig_nobatch_deseq-v{ver}.xlsx"))
## Deleting the file excel/zymo_sig_nobatch_deseq-v202305.xlsx before writing the tables.
## Number of up IDs in contrast zymodeme: 45.
## Number of down IDs in contrast zymodeme: 87.

2.1.1 Plot DE genes without batch estimation/adjustment

zymo_table_nobatch[["plots"]][["zymodeme"]][["deseq_ma_plots"]][["plot"]]
## NULL
zymo_table_nobatch[["plots"]][["zymodeme"]][["deseq_vol_plots"]][["plot"]]
## NULL

Log ratio, mean average plot and volcano plot of the comparison of the two primary zymodeme transcriptomes. When the transcriptomes of the two main strains (43 and 41 samples of z2.3 and z2.1) were compared without any attempt at batch/surrogate estimation with DESeq2, 45 and 85 genes were observed as significantly higher in strain z2.3 and z2.2 respectively using a cutoff of 1.0 logFC and 0.05 FDR adjusted p-value. There remain a large number of genes which are likely significantly different between the two strains, but fall below the 2-fold difference required for ‘significance.’ This follows prior observations that the parasite transcriptomes are constituitively expressed.

When the same data was plotted via a volcano plot, the relatively small range of fold changes compared to the large range of adjusted p-values is visible.

2.2 Attempt SVA estimate

zymo_de_sva <- all_pairwise(two_zymo, filter = TRUE, model_batch = "svaseq")
## 
## z21 z22 z23 z24 
##   7  43  41   2
## Removing 0 low-count genes (8559 remaining).
## Setting 495 low elements to zero.
## transform_counts: Found 495 values equal to 0, adding 1 to the matrix.

zymo_table_sva <- combine_de_tables(
    zymo_de_sva, keepers = zymodeme_keeper,
    rda = glue("rda/zymo_tables_sva-v{ver}.rda"),
    excel = glue("excel/zymo_tables_sva-v{ver}.xlsx"))
## Deleting the file excel/zymo_tables_sva-v202305.xlsx before writing the tables.
## Adding venn plots for zymodeme.
## Saving de result as zymo_tables_sva-v202305 to rda/zymo_tables_sva-v202305.rda.
zymo_sig_sva <- extract_significant_genes(
    zymo_table_sva,
    according_to = "deseq",
    current_id = "GID", required_id = "GID",
    gmt = glue("gmt/zymodeme_sva-v{ver}.gmt"),
    excel = glue("excel/zymo_sig_sva-v{ver}.xlsx"))
## Deleting the file excel/zymo_sig_sva-v202305.xlsx before writing the tables.
## Number of up IDs in contrast zymodeme: 46.
## Number of down IDs in contrast zymodeme: 86.

2.2.1 Plot zymodeme DE genes with sva batch estimation/adjustment

When estimates from SVA were included in the statistical model used by EdgeR, DESeq2, and limma; a nearly identical view of the data emerged. I think this shows with a high degree of confidence, that sva is not having a significant effect on this dataset.

zymo_table_sva[["plots"]][["zymodeme"]][["deseq_ma_plots"]][["plot"]]
## NULL
zymo_table_sva[["plots"]][["zymodeme"]][["deseq_vol_plots"]][["plot"]]
## NULL

3 Parasite Susceptibility to Drug (Current)

This susceptibility comparison is using the ‘current’ dataset.

sus_de_nobatch <- all_pairwise(lp_susceptibility, filter = TRUE, model_batch = FALSE)
## 
## resistant sensitive   unknown 
##        47        49         5
sus_de_nobatch
## A pairwise differential expression with results from: basic, deseq, edger, limma.
## This used a surrogate/batch estimate from: none.
## The primary analysis performed 6 comparisons.
sus_table_nobatch <- combine_de_tables(
    sus_de_nobatch, keepers = susceptibility_keepers,
    rda = glue("rda/sus_tables_nobatch-v{ver}.rda"),
    excel = glue("excel/sus_tables_nobatch-v{ver}.xlsx"))
## Deleting the file excel/sus_tables_nobatch-v202305.xlsx before writing the tables.
## Warning in extract_keepers(extracted, keepers, table_names, all_coefficients, : The table for resistant_ambiguous does not appear in the pairwise
## data.
## Warning in extract_keepers(extracted, keepers, table_names, all_coefficients, : The table for sensitive_ambiguous does not appear in the pairwise
## data.
## Adding venn plots for resistant_sensitive.
## Saving de result as sus_tables_nobatch-v202305 to rda/sus_tables_nobatch-v202305.rda.
sus_table_nobatch
## A set of combined differential expression results.
##                             table deseq_sigup deseq_sigdown edger_sigup edger_sigdown limma_sigup limma_sigdown
## 1 sensitive_vs_resistant-inverted          31            75          34            76          48            68
## `geom_line()`: Each group consists of only one observation.
## ℹ Do you need to adjust the group aesthetic?

sus_sig_nobatch <- extract_significant_genes(

    sus_table_nobatch,
    excel = glue("excel/sus_sig_nobatch-v{ver}.xlsx"))
## Deleting the file excel/sus_sig_nobatch-v202305.xlsx before writing the tables.
sus_de_sva <- all_pairwise(lp_susceptibility, filter = TRUE, model_batch = "svaseq")
## 
## resistant sensitive   unknown 
##        47        49         5
## Removing 0 low-count genes (8576 remaining).
## Setting 636 low elements to zero.
## transform_counts: Found 636 values equal to 0, adding 1 to the matrix.

sus_table_sva <- combine_de_tables(
    sus_de_sva, keepers = susceptibility_keepers,
    rda = glue("rda/sus_tables_sva-v{ver}.rda"),
    excel = glue("excel/sus_tables_sva-v{ver}.xlsx"))
## Deleting the file excel/sus_tables_sva-v202305.xlsx before writing the tables.
## Warning in extract_keepers(extracted, keepers, table_names, all_coefficients, : The table for resistant_ambiguous does not appear in the pairwise
## data.

## Warning in extract_keepers(extracted, keepers, table_names, all_coefficients, : The table for sensitive_ambiguous does not appear in the pairwise
## data.
## Adding venn plots for resistant_sensitive.
## Saving de result as sus_tables_sva-v202305 to rda/sus_tables_sva-v202305.rda.
sus_sig_sva <- extract_significant_genes(
    sus_table_sva, according_to = "deseq",
    excel = glue("excel/sus_sig_sva-v{ver}.xlsx"))
## Deleting the file excel/sus_sig_sva-v202305.xlsx before writing the tables.
## To get a more true sense of sensitive vs resistant with sva, we kind of need to get rid of the
## unknown samples and perhaps the ambiguous.
no_ambiguous <- subset_expt(lp_susceptibility, subset="condition!='ambiguous'") %>%
  subset_expt(subset="condition!='unknown'")
## subset_expt(): There were 101, now there are 101 samples.
## subset_expt(): There were 101, now there are 96 samples.
no_ambiguous_de_sva <- all_pairwise(no_ambiguous, filter = TRUE, model_batch = "svaseq")
## 
## resistant sensitive 
##        47        49
## Removing 0 low-count genes (8565 remaining).
## Setting 423 low elements to zero.
## transform_counts: Found 423 values equal to 0, adding 1 to the matrix.
## Let us see if my keeper code will fail hard or soft with extra contrasts...
no_ambiguous_table_sva <- combine_de_tables(
    no_ambiguous_de_sva, keepers = susceptibility_keepers,
    excel = glue("excel/no_ambiguous_tables_sva-v{ver}.xlsx"))
## Deleting the file excel/no_ambiguous_tables_sva-v202305.xlsx before writing the tables.
## Warning in extract_keepers(extracted, keepers, table_names, all_coefficients, : The table for resistant_ambiguous does not appear in the pairwise
## data.

## Warning in extract_keepers(extracted, keepers, table_names, all_coefficients, : The table for sensitive_ambiguous does not appear in the pairwise
## data.
## Adding venn plots for resistant_sensitive.
no_ambiguous_sig_sva <- extract_significant_genes(
    no_ambiguous_table_sva, according_to = "deseq",
    excel = glue("excel/no_ambiguous_sig_sva-v{ver}.xlsx"))
## Deleting the file excel/no_ambiguous_sig_sva-v202305.xlsx before writing the tables.

3.0.1 Plot Susceptibility DE genes with sva batch estimation/adjustment

sus_table_nobatch[["plots"]][["resistant_sensitive"]][["deseq_ma_plots"]][["plot"]]
## NULL
sus_table_nobatch[["plots"]][["resistant_sensitive"]][["deseq_vol_plots"]][["plot"]]
## NULL
sus_table_sva[["plots"]][["resistant_sensitive"]][["deseq_ma_plots"]][["plot"]]
## NULL
sus_table_sva[["plots"]][["resistant_sensitive"]][["deseq_vol_plots"]][["plot"]]
## NULL
no_ambiguous_table_sva[["plots"]][["resistant_sensitive"]][["deseq_ma_plots"]][["plot"]]
## NULL
no_ambiguous_table_sva[["plots"]][["resistant_sensitive"]][["deseq_vol_plots"]][["plot"]]
## NULL

Given that resistance/sensitivity tends to be correlated with strain, one might expect similar results. One caveat in this context though: there are fewer strains with resistance/sensitivity definitions. This when the analysis was repeated without the ambiguous/unknown samples, a few more genes were observed as significant.

4 Comparing DE results from strain/sensitivity

## zymo_table_sva[["plots"]][["zymodeme"]][["deseq_ma_plots"]][["plot"]]

zy_df <- zymo_table_sva[["data"]][["z23_vs_z22"]]
sus_df <- sus_table_sva[["data"]][["sensitive_vs_resistant"]]

both_df <- merge(zy_df, sus_df, by = "row.names")
plot_df <- both_df[, c("deseq_logfc.x", "deseq_logfc.y")]
## Error in `[.data.frame`(both_df, , c("deseq_logfc.x", "deseq_logfc.y")): undefined columns selected
rownames(plot_df) <- both_df[["Row.names"]]
## Error in rownames(plot_df) <- both_df[["Row.names"]]: object 'plot_df' not found
colnames(plot_df) <- c("z23_vs_z22", "sensitive_vs_resistant")
## Error in colnames(plot_df) <- c("z23_vs_z22", "sensitive_vs_resistant"): object 'plot_df' not found
compare <- plot_linear_scatter(plot_df)
## Error in is.data.frame(x): object 'plot_df' not found
pp(file = "images/compare_sus_zy.png", image = compare$scatter)
## Error in compare$scatter: object of type 'closure' is not subsettable
compare$cor
## Error in compare$cor: object of type 'closure' is not subsettable

5 Parasite Susceptibility to Drug (Historical)

This susceptibility comparison is using the historical dataset.

sushist_de_nobatch <- all_pairwise(lp_susceptibility_historical, filter = TRUE, model_batch = FALSE)
## 
## ambiguous resistant sensitive   unknown 
##         5        12        29        55

sushist_table_nobatch <- combine_de_tables(
    sushist_de_nobatch, keepers = susceptibility_keepers,
    excel = glue("excel/sushist_tables_nobatch-v{ver}.xlsx"))
## Deleting the file excel/sushist_tables_nobatch-v202305.xlsx before writing the tables.
## Adding venn plots for resistant_sensitive.
## Adding venn plots for resistant_ambiguous.
## Adding venn plots for sensitive_ambiguous.
sushist_sig_nobatch <- extract_significant_genes(
    sushist_table_nobatch,
    excel = glue("excel/sushist_sig_nobatch-v{ver}.xlsx"))
## Deleting the file excel/sushist_sig_nobatch-v202305.xlsx before writing the tables.
sushist_de_sva <- all_pairwise(lp_susceptibility_historical, filter = TRUE, model_batch = "svaseq")
## 
## ambiguous resistant sensitive   unknown 
##         5        12        29        55
## Removing 0 low-count genes (8576 remaining).
## Setting 374 low elements to zero.
## transform_counts: Found 374 values equal to 0, adding 1 to the matrix.

sushist_table_sva <- combine_de_tables(
    sushist_de_sva, keepers = susceptibility_keepers,
    excel = glue("excel/sushist_tables_sva-v{ver}.xlsx"))
## Deleting the file excel/sushist_tables_sva-v202305.xlsx before writing the tables.
## Adding venn plots for resistant_sensitive.
## Adding venn plots for resistant_ambiguous.
## Adding venn plots for sensitive_ambiguous.
sushist_sig_sva <- extract_significant_genes(
    sushist_table_sva, according_to = "deseq",
    excel = glue("excel/sushist_sig_sva-v{ver}.xlsx"))
## Deleting the file excel/sushist_sig_sva-v202305.xlsx before writing the tables.

6 Cure/Fail association

##cf_nb_input <- subset_expt(cf_expt, subset="condition!='unknown'")
cf_de_nobatch <- all_pairwise(lp_cf_known, filter = TRUE, model_batch = FALSE)
## 
## cure fail 
##   40   37
cf_table_nobatch <- combine_de_tables(cf_de_nobatch, excel = glue("excel/cf_tables_nobatch-v{ver}.xlsx"))
## Deleting the file excel/cf_tables_nobatch-v202305.xlsx before writing the tables.
## Adding venn plots for fail_vs_cure.
cf_sig_nobatch <- extract_significant_genes(cf_table_nobatch, excel = glue("excel/cf_sig_nobatch-v{ver}.xlsx"))
## Deleting the file excel/cf_sig_nobatch-v202305.xlsx before writing the tables.
cf_de <- all_pairwise(lp_cf_known, filter = TRUE, model_batch = "svaseq")
## 
## cure fail 
##   40   37
## Removing 0 low-count genes (8548 remaining).
## Setting 118 low elements to zero.
## transform_counts: Found 118 values equal to 0, adding 1 to the matrix.
cf_table <- combine_de_tables(cf_de, excel = glue("excel/cf_tables-v{ver}.xlsx"))
## Deleting the file excel/cf_tables-v202305.xlsx before writing the tables.
## Adding venn plots for fail_vs_cure.
cf_sig <- extract_significant_genes(cf_table, excel = glue("excel/cf_sig-v{ver}.xlsx"))
## Deleting the file excel/cf_sig-v202305.xlsx before writing the tables.

6.1 Cure/Fail DE plots

It is not surprising that few or no genes are deemed significantly differentially expressed across samples which were taken from cure or fail patients.

cf_table_nobatch[["plots"]][["fail_vs_cure"]][["deseq_ma_plots"]][["plot"]]
## NULL
dev <- pp(file = "images/cf_ma.png")
cf_table[["plots"]][["fail_vs_cure"]][["deseq_ma_plots"]][["plot"]]
## NULL
closed <- dev.off()
cf_table[["plots"]][["fail_vs_cure"]][["deseq_ma_plots"]][["plot"]]
## NULL

7 Combining the macrophage infected amastigotes with in-vitro promastigotes

One query we have not yet addressed: what are the similarities and differences among the strains used to infect the macrophage samples and the promastigote samples used in the TMRC2 parasite data?

## I just fixed this in the datasets Rmd, but until that propagates just set it manually
annotation(lp_expt) <- annotation(lp_macrophage)
tmrc2_macrophage_norm <- normalize_expt(lp_macrophage, transform="log2", convert="cpm",
                                        norm="quant", filter=TRUE)
## Removing 0 low-count genes (8710 remaining).
## transform_counts: Found 3735 values equal to 0, adding 1 to the matrix.
all_tmrc2 <- combine_expts(lp_expt, lp_macrophage)
## The numbers of samples by condition are:
## 
##   b2904 unknown    z1.0    z1.5    z2.0    z2.1    z2.2    z2.3    z2.4    z3.0    z3.2 
##       1       2       1       1       1       7      64      70       2       1       1
all_nosb <- all_tmrc2
pData(all_nosb)[["stage"]] <- "promastigote"
na_idx <- is.na(pData(all_nosb)[["macrophagetreatment"]])
pData(all_nosb)[na_idx, "macrophagetreatment"] <- "undefined"
all_nosb <- subset_expt(all_nosb, subset="macrophagetreatment!='inf_sb'")
## subset_expt(): There were 151, now there are 130 samples.
ama_idx <- pData(all_nosb)[["macrophagetreatment"]] == "inf"
pData(all_nosb)[ama_idx, "stage" ] <- "amastigote"
pData(all_nosb)[["batch"]] <- pData(all_nosb)[["stage"]]

I think the above picture is sort of the opposite of what we want to compare in a DE analysis for this set of data, e.g. we want to compare promastigotes from amastigotes?

all_nosb <- set_expt_batches(all_nosb, fact="condition") %>%
  set_expt_conditions(fact="stage")
## The number of samples by batch are:
## 
##   b2904 unknown    z1.0    z1.5    z2.0    z2.1    z2.2    z2.3    z2.4    z3.0    z3.2 
##       1       2       1       1       1       7      57      56       2       1       1
## The numbers of samples by condition are:
## 
##   amastigote promastigote 
##           29          101
two_zymo <- subset_expt(
  all_nosb,
  subset="zymodemecategorical=='z22'|zymodemecategorical=='z23'|zymodemecategorical=='unknown'")
## subset_expt(): There were 130, now there are 86 samples.
pro_ama <- all_pairwise(all_nosb, filter=TRUE, model_batch="svaseq")
## 
##   amastigote promastigote 
##           29          101
## Removing 0 low-count genes (8625 remaining).
## Setting 1811 low elements to zero.
## transform_counts: Found 1811 values equal to 0, adding 1 to the matrix.
pro_ama_table <- combine_de_tables(
    pro_ama,
    excel = glue("excel/tmrc2_pro_vs_ama_table-v{ver}.xlsx"))
## Adding venn plots for promastigote_vs_amastigote.
pro_ama_sig <- extract_significant_genes(
    pro_ama_table,
    excel = glue("excel/tmrc2_pro_vs_ama_sig-v{ver}.xlsx"))

7.0.1 Plot promastigote/amastigote DE genes

pro_ama_table[["plots"]][["promastigote_vs_amastigote"]][["deseq_ma_plots"]][["plot"]]
## NULL

I am a little surprised by this plot, I somewhat expected there to be few genes which passed the 2-fold difference demarcation line.

if (!isTRUE(get0("skip_load"))) {
  pander::pander(sessionInfo())
  message(paste0("This is hpgltools commit: ", get_git_commit()))
  message(paste0("Saving to ", savefile))
##  tmp <- sm(saveme(filename = savefile))
}
## If you wish to reproduce this exact build of hpgltools, invoke the following:
## > git clone http://github.com/abelew/hpgltools.git
## > git reset d4aecb6c9f50378b78f8d3c152430d455ccf17a9
## This is hpgltools commit: Tue Jul 11 17:12:15 2023 -0400: d4aecb6c9f50378b78f8d3c152430d455ccf17a9
## Saving to tmrc2_differential_expression_202305.rda.xz
tmp <- loadme(filename = savefile)
LS0tCnRpdGxlOiAiVE1SQzIgMjAyMzA1OiBQcm9tYXN0aWdvdGUgKG1vc3RseSkgRGlmZmVyZW50aWFsIEV4cHJlc3Npb24gQW5hbHlzZXMuIgphdXRob3I6ICJhdGIgYWJlbGV3QGdtYWlsLmNvbSIKZGF0ZTogImByIFN5cy5EYXRlKClgIgpvdXRwdXQ6CiBodG1sX2RvY3VtZW50OgogIGNvZGVfZG93bmxvYWQ6IHRydWUKICBjb2RlX2ZvbGRpbmc6IHNob3cKICBmaWdfY2FwdGlvbjogdHJ1ZQogIGZpZ19oZWlnaHQ6IDcKICBmaWdfd2lkdGg6IDcKICBoaWdobGlnaHQ6IGRlZmF1bHQKICBrZWVwX21kOiBmYWxzZQogIG1vZGU6IHNlbGZjb250YWluZWQKICBudW1iZXJfc2VjdGlvbnM6IHRydWUKICBzZWxmX2NvbnRhaW5lZDogdHJ1ZQogIHRoZW1lOiByZWFkYWJsZQogIHRvYzogdHJ1ZQogIHRvY19mbG9hdDoKICAgY29sbGFwc2VkOiBmYWxzZQogICBzbW9vdGhfc2Nyb2xsOiBmYWxzZQotLS0KCjxzdHlsZT4KICBib2R5IC5tYWluLWNvbnRhaW5lciB7CiAgICBtYXgtd2lkdGg6IDE2MDBweDsKICB9Cjwvc3R5bGU+CgpgYGB7ciBvcHRpb25zLCBpbmNsdWRlID0gRkFMU0V9CmxpYnJhcnkoSGVhdHBsdXMpCmxpYnJhcnkoaHBnbHRvb2xzKQpsaWJyYXJ5KGdsdWUpCnR0IDwtIGRldnRvb2xzOjpsb2FkX2FsbCgifi9ocGdsdG9vbHMiKQprbml0cjo6b3B0c19rbml0JHNldChwcm9ncmVzcyA9IFRSVUUsCiAgICAgICAgICAgICAgICAgICAgIHZlcmJvc2UgPSBUUlVFLAogICAgICAgICAgICAgICAgICAgICB3aWR0aCA9IDkwLAogICAgICAgICAgICAgICAgICAgICBlY2hvID0gVFJVRSkKa25pdHI6Om9wdHNfY2h1bmskc2V0KGVycm9yID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgIGZpZy53aWR0aCA9IDgsCiAgICAgICAgICAgICAgICAgICAgICBmaWcuaGVpZ2h0ID0gOCwKICAgICAgICAgICAgICAgICAgICAgIGRwaSA9IDk2KQpvbGRfb3B0aW9ucyA8LSBvcHRpb25zKGRpZ2l0cyA9IDQsCiAgICAgICAgICAgICAgICAgICAgICAgc3RyaW5nc0FzRmFjdG9ycyA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgIGtuaXRyLmR1cGxpY2F0ZS5sYWJlbCA9ICJhbGxvdyIpCmdncGxvdDI6OnRoZW1lX3NldChnZ3Bsb3QyOjp0aGVtZV9idyhiYXNlX3NpemUgPSAxMikpCnZlciA8LSAiMjAyMzA1IgpwcmV2aW91c19maWxlIDwtICIiCnJ1bmRhdGUgPC0gZm9ybWF0KFN5cy5EYXRlKCksIGZvcm1hdCA9ICIlWSVtJWQiKQoKIyMgdG1wIDwtIHRyeShzbShsb2FkbWUoZmlsZW5hbWUgPSBnc3ViKHBhdHRlcm4gPSAiXFwuUm1kIiwgcmVwbGFjZSA9ICJcXC5yZGFcXC54eiIsIHggPSBwcmV2aW91c19maWxlKSkpKQpybWRfZmlsZSA8LSBnbHVlKCJ0bXJjMl9kaWZmZXJlbnRpYWxfZXhwcmVzc2lvbl97dmVyfS5SbWQiKQpzYXZlZmlsZSA8LSBnc3ViKHBhdHRlcm4gPSAiXFwuUm1kIiwgcmVwbGFjZSA9ICJcXC5yZGFcXC54eiIsIHggPSBybWRfZmlsZSkKbG9hZGVkIDwtIGxvYWQoZmlsZT1nbHVlKCJyZGEvdG1yYzJfZGF0YV9zdHJ1Y3R1cmVzLXZ7dmVyfS5yZGEiKSkKYGBgCgojIENvbnRyYXN0cwoKYGBge3Iga2VlcGVyc30Kenltb2RlbWVfa2VlcGVyIDwtIGxpc3QoCiAgICAienltb2RlbWUiID0gYygiejIzIiwgInoyMiIpKQpzdXNjZXB0aWJpbGl0eV9rZWVwZXJzIDwtIGxpc3QoCiAgICAicmVzaXN0YW50X3NlbnNpdGl2ZSIgPSBjKCJyZXNpc3RhbnQiLCAic2Vuc2l0aXZlIiksCiAgICAicmVzaXN0YW50X2FtYmlndW91cyIgPSBjKCJyZXNpc3RhbnQiLCAiYW1iaWd1b3VzIiksCiAgICAic2Vuc2l0aXZlX2FtYmlndW91cyIgPSBjKCJzZW5zaXRpdmUiLCAiYW1iaWd1b3VzIikpCmBgYAoKIyMgWnltb2RlbWUgZW56eW1lIGdlbmUgSURzCgpOYWppYiByZWFkIG1lIGFuIGVtYWlsIGxpc3Rpbmcgb2ZmIHRoZSBnZW5lIG5hbWVzIGFzc29jaWF0ZWQgd2l0aCB0aGUgenltb2RlbWUKY2xhc3NpZmljYXRpb24uICBJIHRvb2sgdGhvc2UgbmFtZXMgYW5kIGNyb3NzIHJlZmVyZW5jZWQgdGhlbSBhZ2FpbnN0IHRoZQpMZWlzaG1hbmlhIHBhbmFtZW5zaXMgZ2VuZSBhbm5vdGF0aW9ucyBhbmQgZm91bmQgdGhlIGZvbGxvd2luZzoKClRoZXkgYXJlOgoKMS4gQUxBVDogTFBBTDEzXzEyMDAxMDkwMCAtLSBhbGFuaW5lIGFtaW5vdHJhbnNmZXJhc2UKMi4gQVNBVDogTFBBTDEzXzM0MDAxMzAwMCAtLSBhc3BhcnRhdGUgYW1pbm90cmFuc2ZlcmFzZQozLiBHNlBEOiBMUEFMMTNfMDAwMDU0MTAwIC0tIGdsdWNhc2UtNi1waG9zcGhhdGUgMS1kZWh5ZHJvZ2VuYXNlCjQuIE5IOiBMUEFMMTNfMTQwMDYxMDAsIExQQUwxM18xODAwMTg1MDAgLS0gaW5vc2luZS1ndWFuaW5lIG51Y2xlb3NpZGUgaHlkcm9sYXNlCjUuIE1QSTogTFBBTDEzXzMyMDAyMjMwMCAobWF5YmUpIC0tIG1hbm5vc2UgcGhvc3BoYXRlIGlzb21lcmFzZSAoSSBjaG9zZSBwaG9zcGhvbWFubm9zZSBpc29tZXJhc2UpCgpHaXZlbiB0aGVzZSA2IGdlbmUgSURzIChOSCBoYXMgdHdvIGdlbmUgSURzIGFzc29jaWF0ZWQgd2l0aCBpdCksIEkgY2FuIGRvIHNvbWUKbG9va2luZyBmb3Igc3BlY2lmaWMgZGlmZmVyZW5jZXMgYW1vbmcgdGhlIHZhcmlvdXMgc2FtcGxlcy4KCiMjIyBFeHByZXNzaW9uIGxldmVscyBvZiB6eW1vZGVtZSBnZW5lcwoKVGhlIGZvbGxvd2luZyBjcmVhdGVzIGEgY29sb3JzcGFjZSAocmVkIHRvIGdyZWVuKSBoZWF0bWFwIHNob3dpbmcgdGhlIG9ic2VydmVkCmV4cHJlc3Npb24gb2YgdGhlc2UgZ2VuZXMgaW4gZXZlcnkgc2FtcGxlLgoKYGBge3Igenltb2RlbWVzfQpteV9nZW5lcyA8LSBjKCJMUEFMMTNfMTIwMDEwOTAwIiwgIkxQQUwxM18zNDAwMTMwMDAiLCAiTFBBTDEzXzAwMDA1NDEwMCIsCiAgICAgICAgICAgICAgIkxQQUwxM18xNDAwMDYxMDAiLCAiTFBBTDEzXzE4MDAxODUwMCIsICJMUEFMMTNfMzIwMDIyMzAwIiwKICAgICAgICAgICAgICAib3RoZXIiKQpteV9uYW1lcyA8LSBjKCJBTEFUIiwgIkFTQVQiLCAiRzZQRCIsICJOSHYxIiwgIk5IdjIiLCAiTVBJIiwgIm90aGVyIikKCnp5bW9fc2l4X2dlbmVzIDwtIGV4Y2x1ZGVfZ2VuZXNfZXhwdChscF90d29fc3RyYWlucywgaWRzID0gbXlfZ2VuZXMsIG1ldGhvZCA9ICJrZWVwIikKc3RyYWluX25vcm0gPC0gbm9ybWFsaXplX2V4cHQoenltb19zaXhfZ2VuZXMsIGNvbnZlcnQ9InJwa20iLCBmaWx0ZXI9VFJVRSwgdHJhbnNmb3JtPSJsb2cyIikKCnp5bW9faGVhdG1hcCA8LSBwbG90X3NhbXBsZV9oZWF0bWFwKHN0cmFpbl9ub3JtLCByb3dfbGFiZWwgPSBteV9uYW1lcykKenltb19oZWF0bWFwCgpscF9ub3JtIDwtIG5vcm1hbGl6ZV9leHB0KGxwX3R3b19zdHJhaW5zLCBmaWx0ZXI9VFJVRSwgY29udmVydD0icnBrbSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgbm9ybT0icXVhbnQiLCB0cmFuc2Zvcm09ImxvZzIiKQp6eW1vX2hlYXRtYXBfYWxsIDwtIHBsb3Rfc2FtcGxlX2hlYXRtYXAobHBfbm9ybSkKenltb19oZWF0bWFwX2FsbApgYGAKCiMjIENvbXBhcmUgdG8gaGlnaGx5IGV4cHJlc3NlZCwgdmFyaWFudCBnZW5lcwoKSSB3YW50IHRvIGNvbXBhcmUgdGhlIGFib3ZlIGhlYXRtYXAgd2l0aCBvbmUgd2hpY2ggaXMgY29tcHJpc2VkIG9mIGFsbApnZW5lcyB3aXRoIHNvbWUgJ3NpZ25pZmljYW50bHkgaGlnaCcgZXhwcmVzc2lvbiB2YWx1ZSBhbmQgYWxzbyBhCm5vdC1uZWdsaWdpYmxlIGNvZWZmaWNpZW50IG9mIHZhcmlhbmNlLgoKYGBge3IgY3ZfaGVhdG1hcH0Kenltb19oaWdoX2dlbmVzIDwtIG5vcm1hbGl6ZV9leHB0KGxwX3R3b19zdHJhaW5zLCBmaWx0ZXIgPSAiY3YiLCBjdl9taW4gPSAwLjkpCgpoaWdoX3N0cmFpbl9ub3JtIDwtIG5vcm1hbGl6ZV9leHB0KHp5bW9faGlnaF9nZW5lcywgY29udmVydCA9ICJycGttIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBub3JtID0gInF1YW50IiwgdHJhbnNmb3JtID0gImxvZzIiKQp6eW1vX2hlYXRtYXAgPC0gcGxvdF9zYW1wbGVfaGVhdG1hcChoaWdoX3N0cmFpbl9ub3JtLCByb3dfbGFiZWwgPSBteV9uYW1lcykKenltb19oZWF0bWFwCmBgYAoKSSB0aGluayB0aGlzIHBsb3Qgc3VnZ2VzdHMgdGhhdCB0aGUgZGlmZmVyZW5jZSBiZXR3ZWVuIHRoZSB0d28gcHJpbWFyeQpzdHJhaW5zIGlzIG5vdCByZWFsbHkgb25lIG9mIGEgZmV3IHNwZWNpZmljIGdlbmVzLCBidXQgaW5zdGVhZCBhCmdsb2JhbCBwYXR0ZXJuLgoKIyBaeW1vZGVtZSBkaWZmZXJlbnRpYWwgZXhwcmVzc2lvbgoKIyMgTm8gYXR0ZW1wdCBhdCBiYXRjaCBlc3RpbWF0aW9uCgpgYGB7ciB6eW1vZGVtZV9kZX0KdHdvX3p5bW8gPC0gc2V0X2V4cHRfY29uZGl0aW9ucyhscF90d29fc3RyYWlucywgZmFjdCA9ICJ6eW1vZGVtZWNhdGVnb3JpY2FsIikgJT4lCiAgc3Vic2V0X2V4cHQoc3Vic2V0ID0gImNvbmRpdGlvbiE9J3Vua25vd24nIikKCnp5bW9fZGVfbm9iYXRjaCA8LSBhbGxfcGFpcndpc2UodHdvX3p5bW8sIGZpbHRlciA9IFRSVUUsIG1vZGVsX2JhdGNoID0gRkFMU0UpCnp5bW9fdGFibGVfbm9iYXRjaCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIHp5bW9fZGVfbm9iYXRjaCwga2VlcGVycyA9IHp5bW9kZW1lX2tlZXBlciwKICAgIHJkYSA9IGdsdWUoInJkYS96eW1vX3RhYmxlc19ub2JhdGNoLXZ7dmVyfS5yZGEiKSwKICAgIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvenltb190YWJsZXNfbm9iYXRjaC12e3Zlcn0ueGxzeCIpKQp6eW1vX3NpZ19ub2JhdGNoIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgICB6eW1vX3RhYmxlX25vYmF0Y2gsCiAgICBhY2NvcmRpbmdfdG8gPSAiZGVzZXEiLCBjdXJyZW50X2lkID0gIkdJRCIsIHJlcXVpcmVkX2lkID0gIkdJRCIsCiAgICBnbXQgPSBnbHVlKCJnbXQvenltb2RlbWVfbm9iYXRjaC12e3Zlcn0uZ210IiksCiAgICBleGNlbCA9IGdsdWUoImV4Y2VsL3p5bW9fc2lnX25vYmF0Y2hfZGVzZXEtdnt2ZXJ9Lnhsc3giKSkKYGBgCgojIyMgUGxvdCBERSBnZW5lcyB3aXRob3V0IGJhdGNoIGVzdGltYXRpb24vYWRqdXN0bWVudAoKYGBge3Igenltb2RlbWVfZGVfcGxvdHN9Cnp5bW9fdGFibGVfbm9iYXRjaFtbInBsb3RzIl1dW1sienltb2RlbWUiXV1bWyJkZXNlcV9tYV9wbG90cyJdXVtbInBsb3QiXV0Kenltb190YWJsZV9ub2JhdGNoW1sicGxvdHMiXV1bWyJ6eW1vZGVtZSJdXVtbImRlc2VxX3ZvbF9wbG90cyJdXVtbInBsb3QiXV0KYGBgCgpMb2cgcmF0aW8sIG1lYW4gYXZlcmFnZSBwbG90IGFuZCB2b2xjYW5vIHBsb3Qgb2YgdGhlIGNvbXBhcmlzb24gb2YgdGhlCnR3byBwcmltYXJ5IHp5bW9kZW1lIHRyYW5zY3JpcHRvbWVzLiAgV2hlbiB0aGUgdHJhbnNjcmlwdG9tZXMgb2YgdGhlCnR3byBtYWluIHN0cmFpbnMgKDQzIGFuZCA0MSBzYW1wbGVzIG9mIHoyLjMgYW5kIHoyLjEpIHdlcmUgY29tcGFyZWQKd2l0aG91dCBhbnkgYXR0ZW1wdCBhdCBiYXRjaC9zdXJyb2dhdGUgZXN0aW1hdGlvbiB3aXRoIERFU2VxMiwgNDUgYW5kCjg1IGdlbmVzIHdlcmUgb2JzZXJ2ZWQgYXMgc2lnbmlmaWNhbnRseSBoaWdoZXIgaW4gc3RyYWluIHoyLjMgYW5kIHoyLjIKcmVzcGVjdGl2ZWx5IHVzaW5nIGEgY3V0b2ZmIG9mIDEuMCBsb2dGQyBhbmQgMC4wNSBGRFIgYWRqdXN0ZWQKcC12YWx1ZS4gIFRoZXJlIHJlbWFpbiBhIGxhcmdlIG51bWJlciBvZiBnZW5lcyB3aGljaCBhcmUgbGlrZWx5CnNpZ25pZmljYW50bHkgZGlmZmVyZW50IGJldHdlZW4gdGhlIHR3byBzdHJhaW5zLCBidXQgZmFsbCBiZWxvdyB0aGUKMi1mb2xkIGRpZmZlcmVuY2UgcmVxdWlyZWQgZm9yICdzaWduaWZpY2FuY2UuJyAgVGhpcyBmb2xsb3dzIHByaW9yCm9ic2VydmF0aW9ucyB0aGF0IHRoZSBwYXJhc2l0ZSB0cmFuc2NyaXB0b21lcyBhcmUgY29uc3RpdHVpdGl2ZWx5CmV4cHJlc3NlZC4KCldoZW4gdGhlIHNhbWUgZGF0YSB3YXMgcGxvdHRlZCB2aWEgYSB2b2xjYW5vIHBsb3QsIHRoZSByZWxhdGl2ZWx5CnNtYWxsIHJhbmdlIG9mIGZvbGQgY2hhbmdlcyBjb21wYXJlZCB0byB0aGUgbGFyZ2UgcmFuZ2Ugb2YgYWRqdXN0ZWQKcC12YWx1ZXMgaXMgdmlzaWJsZS4KCiMjIEF0dGVtcHQgU1ZBIGVzdGltYXRlCgpgYGB7ciB6eW1vX2RlX3N2YX0Kenltb19kZV9zdmEgPC0gYWxsX3BhaXJ3aXNlKHR3b196eW1vLCBmaWx0ZXIgPSBUUlVFLCBtb2RlbF9iYXRjaCA9ICJzdmFzZXEiKQp6eW1vX3RhYmxlX3N2YSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIHp5bW9fZGVfc3ZhLCBrZWVwZXJzID0genltb2RlbWVfa2VlcGVyLAogICAgcmRhID0gZ2x1ZSgicmRhL3p5bW9fdGFibGVzX3N2YS12e3Zlcn0ucmRhIiksCiAgICBleGNlbCA9IGdsdWUoImV4Y2VsL3p5bW9fdGFibGVzX3N2YS12e3Zlcn0ueGxzeCIpKQp6eW1vX3NpZ19zdmEgPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICAgIHp5bW9fdGFibGVfc3ZhLAogICAgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICAgIGN1cnJlbnRfaWQgPSAiR0lEIiwgcmVxdWlyZWRfaWQgPSAiR0lEIiwKICAgIGdtdCA9IGdsdWUoImdtdC96eW1vZGVtZV9zdmEtdnt2ZXJ9LmdtdCIpLAogICAgZXhjZWwgPSBnbHVlKCJleGNlbC96eW1vX3NpZ19zdmEtdnt2ZXJ9Lnhsc3giKSkKYGBgCgojIyMgUGxvdCB6eW1vZGVtZSBERSBnZW5lcyB3aXRoIHN2YSBiYXRjaCBlc3RpbWF0aW9uL2FkanVzdG1lbnQKCldoZW4gZXN0aW1hdGVzIGZyb20gU1ZBIHdlcmUgaW5jbHVkZWQgaW4gdGhlIHN0YXRpc3RpY2FsIG1vZGVsIHVzZWQgYnkKRWRnZVIsIERFU2VxMiwgYW5kIGxpbW1hOyBhIG5lYXJseSBpZGVudGljYWwgdmlldyBvZiB0aGUgZGF0YSBlbWVyZ2VkLgpJIHRoaW5rIHRoaXMgc2hvd3Mgd2l0aCBhIGhpZ2ggZGVncmVlIG9mIGNvbmZpZGVuY2UsIHRoYXQgc3ZhIGlzIG5vdApoYXZpbmcgYSBzaWduaWZpY2FudCBlZmZlY3Qgb24gdGhpcyBkYXRhc2V0LgoKYGBge3Igenltb2RlbWVfc3ZhX2RlX21hX3ZvbGNhbm99Cnp5bW9fdGFibGVfc3ZhW1sicGxvdHMiXV1bWyJ6eW1vZGVtZSJdXVtbImRlc2VxX21hX3Bsb3RzIl1dW1sicGxvdCJdXQp6eW1vX3RhYmxlX3N2YVtbInBsb3RzIl1dW1sienltb2RlbWUiXV1bWyJkZXNlcV92b2xfcGxvdHMiXV1bWyJwbG90Il1dCmBgYAoKIyBQYXJhc2l0ZSBTdXNjZXB0aWJpbGl0eSB0byBEcnVnIChDdXJyZW50KQoKVGhpcyBzdXNjZXB0aWJpbGl0eSBjb21wYXJpc29uIGlzIHVzaW5nIHRoZSAnY3VycmVudCcgZGF0YXNldC4KCmBgYHtyIHBhcmFzaXRlX3N1c2NlcHRpYmlsaXR5X2RlfQpzdXNfZGVfbm9iYXRjaCA8LSBhbGxfcGFpcndpc2UobHBfc3VzY2VwdGliaWxpdHksIGZpbHRlciA9IFRSVUUsIG1vZGVsX2JhdGNoID0gRkFMU0UpCnN1c19kZV9ub2JhdGNoCgpzdXNfdGFibGVfbm9iYXRjaCA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIHN1c19kZV9ub2JhdGNoLCBrZWVwZXJzID0gc3VzY2VwdGliaWxpdHlfa2VlcGVycywKICAgIHJkYSA9IGdsdWUoInJkYS9zdXNfdGFibGVzX25vYmF0Y2gtdnt2ZXJ9LnJkYSIpLAogICAgZXhjZWwgPSBnbHVlKCJleGNlbC9zdXNfdGFibGVzX25vYmF0Y2gtdnt2ZXJ9Lnhsc3giKSkKc3VzX3RhYmxlX25vYmF0Y2gKc3VzX3NpZ19ub2JhdGNoIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCgogICAgc3VzX3RhYmxlX25vYmF0Y2gsCiAgICBleGNlbCA9IGdsdWUoImV4Y2VsL3N1c19zaWdfbm9iYXRjaC12e3Zlcn0ueGxzeCIpKQoKc3VzX2RlX3N2YSA8LSBhbGxfcGFpcndpc2UobHBfc3VzY2VwdGliaWxpdHksIGZpbHRlciA9IFRSVUUsIG1vZGVsX2JhdGNoID0gInN2YXNlcSIpCgpzdXNfdGFibGVfc3ZhIDwtIGNvbWJpbmVfZGVfdGFibGVzKAogICAgc3VzX2RlX3N2YSwga2VlcGVycyA9IHN1c2NlcHRpYmlsaXR5X2tlZXBlcnMsCiAgICByZGEgPSBnbHVlKCJyZGEvc3VzX3RhYmxlc19zdmEtdnt2ZXJ9LnJkYSIpLAogICAgZXhjZWwgPSBnbHVlKCJleGNlbC9zdXNfdGFibGVzX3N2YS12e3Zlcn0ueGxzeCIpKQpzdXNfc2lnX3N2YSA8LSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzKAogICAgc3VzX3RhYmxlX3N2YSwgYWNjb3JkaW5nX3RvID0gImRlc2VxIiwKICAgIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvc3VzX3NpZ19zdmEtdnt2ZXJ9Lnhsc3giKSkKCiMjIFRvIGdldCBhIG1vcmUgdHJ1ZSBzZW5zZSBvZiBzZW5zaXRpdmUgdnMgcmVzaXN0YW50IHdpdGggc3ZhLCB3ZSBraW5kIG9mIG5lZWQgdG8gZ2V0IHJpZCBvZiB0aGUKIyMgdW5rbm93biBzYW1wbGVzIGFuZCBwZXJoYXBzIHRoZSBhbWJpZ3VvdXMuCm5vX2FtYmlndW91cyA8LSBzdWJzZXRfZXhwdChscF9zdXNjZXB0aWJpbGl0eSwgc3Vic2V0PSJjb25kaXRpb24hPSdhbWJpZ3VvdXMnIikgJT4lCiAgc3Vic2V0X2V4cHQoc3Vic2V0PSJjb25kaXRpb24hPSd1bmtub3duJyIpCm5vX2FtYmlndW91c19kZV9zdmEgPC0gYWxsX3BhaXJ3aXNlKG5vX2FtYmlndW91cywgZmlsdGVyID0gVFJVRSwgbW9kZWxfYmF0Y2ggPSAic3Zhc2VxIikKIyMgTGV0IHVzIHNlZSBpZiBteSBrZWVwZXIgY29kZSB3aWxsIGZhaWwgaGFyZCBvciBzb2Z0IHdpdGggZXh0cmEgY29udHJhc3RzLi4uCm5vX2FtYmlndW91c190YWJsZV9zdmEgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgICBub19hbWJpZ3VvdXNfZGVfc3ZhLCBrZWVwZXJzID0gc3VzY2VwdGliaWxpdHlfa2VlcGVycywKICAgIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvbm9fYW1iaWd1b3VzX3RhYmxlc19zdmEtdnt2ZXJ9Lnhsc3giKSkKbm9fYW1iaWd1b3VzX3NpZ19zdmEgPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcygKICAgIG5vX2FtYmlndW91c190YWJsZV9zdmEsIGFjY29yZGluZ190byA9ICJkZXNlcSIsCiAgICBleGNlbCA9IGdsdWUoImV4Y2VsL25vX2FtYmlndW91c19zaWdfc3ZhLXZ7dmVyfS54bHN4IikpCmBgYAoKIyMjIFBsb3QgU3VzY2VwdGliaWxpdHkgREUgZ2VuZXMgd2l0aCBzdmEgYmF0Y2ggZXN0aW1hdGlvbi9hZGp1c3RtZW50CgpgYGB7ciBwYXJhc2l0ZV9zdXNjZXB0aWJpbGl0eV9waWN0dXJlc192MX0Kc3VzX3RhYmxlX25vYmF0Y2hbWyJwbG90cyJdXVtbInJlc2lzdGFudF9zZW5zaXRpdmUiXV1bWyJkZXNlcV9tYV9wbG90cyJdXVtbInBsb3QiXV0Kc3VzX3RhYmxlX25vYmF0Y2hbWyJwbG90cyJdXVtbInJlc2lzdGFudF9zZW5zaXRpdmUiXV1bWyJkZXNlcV92b2xfcGxvdHMiXV1bWyJwbG90Il1dCgpzdXNfdGFibGVfc3ZhW1sicGxvdHMiXV1bWyJyZXNpc3RhbnRfc2Vuc2l0aXZlIl1dW1siZGVzZXFfbWFfcGxvdHMiXV1bWyJwbG90Il1dCnN1c190YWJsZV9zdmFbWyJwbG90cyJdXVtbInJlc2lzdGFudF9zZW5zaXRpdmUiXV1bWyJkZXNlcV92b2xfcGxvdHMiXV1bWyJwbG90Il1dCgpub19hbWJpZ3VvdXNfdGFibGVfc3ZhW1sicGxvdHMiXV1bWyJyZXNpc3RhbnRfc2Vuc2l0aXZlIl1dW1siZGVzZXFfbWFfcGxvdHMiXV1bWyJwbG90Il1dCm5vX2FtYmlndW91c190YWJsZV9zdmFbWyJwbG90cyJdXVtbInJlc2lzdGFudF9zZW5zaXRpdmUiXV1bWyJkZXNlcV92b2xfcGxvdHMiXV1bWyJwbG90Il1dCmBgYAoKR2l2ZW4gdGhhdCByZXNpc3RhbmNlL3NlbnNpdGl2aXR5IHRlbmRzIHRvIGJlIGNvcnJlbGF0ZWQgd2l0aCBzdHJhaW4sCm9uZSBtaWdodCBleHBlY3Qgc2ltaWxhciByZXN1bHRzLiAgT25lIGNhdmVhdCBpbiB0aGlzIGNvbnRleHQgdGhvdWdoOgp0aGVyZSBhcmUgZmV3ZXIgc3RyYWlucyB3aXRoIHJlc2lzdGFuY2Uvc2Vuc2l0aXZpdHkgZGVmaW5pdGlvbnMuICBUaGlzCndoZW4gdGhlIGFuYWx5c2lzIHdhcyByZXBlYXRlZCB3aXRob3V0IHRoZSBhbWJpZ3VvdXMvdW5rbm93biBzYW1wbGVzLAphIGZldyBtb3JlIGdlbmVzIHdlcmUgb2JzZXJ2ZWQgYXMgc2lnbmlmaWNhbnQuCgojIENvbXBhcmluZyBERSByZXN1bHRzIGZyb20gc3RyYWluL3NlbnNpdGl2aXR5CgpgYGB7ciBzZW5zaXRpdmVfdnNfenltb30KIyMgenltb190YWJsZV9zdmFbWyJwbG90cyJdXVtbInp5bW9kZW1lIl1dW1siZGVzZXFfbWFfcGxvdHMiXV1bWyJwbG90Il1dCgp6eV9kZiA8LSB6eW1vX3RhYmxlX3N2YVtbImRhdGEiXV1bWyJ6MjNfdnNfejIyIl1dCnN1c19kZiA8LSBzdXNfdGFibGVfc3ZhW1siZGF0YSJdXVtbInNlbnNpdGl2ZV92c19yZXNpc3RhbnQiXV0KCmJvdGhfZGYgPC0gbWVyZ2UoenlfZGYsIHN1c19kZiwgYnkgPSAicm93Lm5hbWVzIikKcGxvdF9kZiA8LSBib3RoX2RmWywgYygiZGVzZXFfbG9nZmMueCIsICJkZXNlcV9sb2dmYy55IildCnJvd25hbWVzKHBsb3RfZGYpIDwtIGJvdGhfZGZbWyJSb3cubmFtZXMiXV0KY29sbmFtZXMocGxvdF9kZikgPC0gYygiejIzX3ZzX3oyMiIsICJzZW5zaXRpdmVfdnNfcmVzaXN0YW50IikKCmNvbXBhcmUgPC0gcGxvdF9saW5lYXJfc2NhdHRlcihwbG90X2RmKQpwcChmaWxlID0gImltYWdlcy9jb21wYXJlX3N1c196eS5wbmciLCBpbWFnZSA9IGNvbXBhcmUkc2NhdHRlcikKY29tcGFyZSRjb3IKYGBgCgojIFBhcmFzaXRlIFN1c2NlcHRpYmlsaXR5IHRvIERydWcgKEhpc3RvcmljYWwpCgpUaGlzIHN1c2NlcHRpYmlsaXR5IGNvbXBhcmlzb24gaXMgdXNpbmcgdGhlIGhpc3RvcmljYWwgZGF0YXNldC4KCmBgYHtyIHBhcmFzaXRlX3N1c2NlcHRpYmlsaXR5X2RlX2hpc3RvcmljYWx9CnN1c2hpc3RfZGVfbm9iYXRjaCA8LSBhbGxfcGFpcndpc2UobHBfc3VzY2VwdGliaWxpdHlfaGlzdG9yaWNhbCwgZmlsdGVyID0gVFJVRSwgbW9kZWxfYmF0Y2ggPSBGQUxTRSkKc3VzaGlzdF90YWJsZV9ub2JhdGNoIDwtIGNvbWJpbmVfZGVfdGFibGVzKAogICAgc3VzaGlzdF9kZV9ub2JhdGNoLCBrZWVwZXJzID0gc3VzY2VwdGliaWxpdHlfa2VlcGVycywKICAgIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvc3VzaGlzdF90YWJsZXNfbm9iYXRjaC12e3Zlcn0ueGxzeCIpKQpzdXNoaXN0X3NpZ19ub2JhdGNoIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgICBzdXNoaXN0X3RhYmxlX25vYmF0Y2gsCiAgICBleGNlbCA9IGdsdWUoImV4Y2VsL3N1c2hpc3Rfc2lnX25vYmF0Y2gtdnt2ZXJ9Lnhsc3giKSkKCnN1c2hpc3RfZGVfc3ZhIDwtIGFsbF9wYWlyd2lzZShscF9zdXNjZXB0aWJpbGl0eV9oaXN0b3JpY2FsLCBmaWx0ZXIgPSBUUlVFLCBtb2RlbF9iYXRjaCA9ICJzdmFzZXEiKQpzdXNoaXN0X3RhYmxlX3N2YSA8LSBjb21iaW5lX2RlX3RhYmxlcygKICAgIHN1c2hpc3RfZGVfc3ZhLCBrZWVwZXJzID0gc3VzY2VwdGliaWxpdHlfa2VlcGVycywKICAgIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvc3VzaGlzdF90YWJsZXNfc3ZhLXZ7dmVyfS54bHN4IikpCnN1c2hpc3Rfc2lnX3N2YSA8LSBleHRyYWN0X3NpZ25pZmljYW50X2dlbmVzKAogICAgc3VzaGlzdF90YWJsZV9zdmEsIGFjY29yZGluZ190byA9ICJkZXNlcSIsCiAgICBleGNlbCA9IGdsdWUoImV4Y2VsL3N1c2hpc3Rfc2lnX3N2YS12e3Zlcn0ueGxzeCIpKQpgYGAKCiMgQ3VyZS9GYWlsIGFzc29jaWF0aW9uCgpgYGB7ciBjdXJlX2ZhaWxfZGV9CiMjY2ZfbmJfaW5wdXQgPC0gc3Vic2V0X2V4cHQoY2ZfZXhwdCwgc3Vic2V0PSJjb25kaXRpb24hPSd1bmtub3duJyIpCmNmX2RlX25vYmF0Y2ggPC0gYWxsX3BhaXJ3aXNlKGxwX2NmX2tub3duLCBmaWx0ZXIgPSBUUlVFLCBtb2RlbF9iYXRjaCA9IEZBTFNFKQpjZl90YWJsZV9ub2JhdGNoIDwtIGNvbWJpbmVfZGVfdGFibGVzKGNmX2RlX25vYmF0Y2gsIGV4Y2VsID0gZ2x1ZSgiZXhjZWwvY2ZfdGFibGVzX25vYmF0Y2gtdnt2ZXJ9Lnhsc3giKSkKY2Zfc2lnX25vYmF0Y2ggPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcyhjZl90YWJsZV9ub2JhdGNoLCBleGNlbCA9IGdsdWUoImV4Y2VsL2NmX3NpZ19ub2JhdGNoLXZ7dmVyfS54bHN4IikpCgpjZl9kZSA8LSBhbGxfcGFpcndpc2UobHBfY2Zfa25vd24sIGZpbHRlciA9IFRSVUUsIG1vZGVsX2JhdGNoID0gInN2YXNlcSIpCmNmX3RhYmxlIDwtIGNvbWJpbmVfZGVfdGFibGVzKGNmX2RlLCBleGNlbCA9IGdsdWUoImV4Y2VsL2NmX3RhYmxlcy12e3Zlcn0ueGxzeCIpKQpjZl9zaWcgPC0gZXh0cmFjdF9zaWduaWZpY2FudF9nZW5lcyhjZl90YWJsZSwgZXhjZWwgPSBnbHVlKCJleGNlbC9jZl9zaWctdnt2ZXJ9Lnhsc3giKSkKYGBgCgojIyBDdXJlL0ZhaWwgREUgcGxvdHMKCkl0IGlzIG5vdCBzdXJwcmlzaW5nIHRoYXQgZmV3IG9yIG5vIGdlbmVzIGFyZSBkZWVtZWQgc2lnbmlmaWNhbnRseQpkaWZmZXJlbnRpYWxseSBleHByZXNzZWQgYWNyb3NzIHNhbXBsZXMgd2hpY2ggd2VyZSB0YWtlbiBmcm9tIGN1cmUgb3IKZmFpbCBwYXRpZW50cy4KCmBgYHtyIGNmX2RlX3Bsb3RzfQpjZl90YWJsZV9ub2JhdGNoW1sicGxvdHMiXV1bWyJmYWlsX3ZzX2N1cmUiXV1bWyJkZXNlcV9tYV9wbG90cyJdXVtbInBsb3QiXV0KCmRldiA8LSBwcChmaWxlID0gImltYWdlcy9jZl9tYS5wbmciKQpjZl90YWJsZVtbInBsb3RzIl1dW1siZmFpbF92c19jdXJlIl1dW1siZGVzZXFfbWFfcGxvdHMiXV1bWyJwbG90Il1dCmNsb3NlZCA8LSBkZXYub2ZmKCkKY2ZfdGFibGVbWyJwbG90cyJdXVtbImZhaWxfdnNfY3VyZSJdXVtbImRlc2VxX21hX3Bsb3RzIl1dW1sicGxvdCJdXQpgYGAKCiMgQ29tYmluaW5nIHRoZSBtYWNyb3BoYWdlIGluZmVjdGVkIGFtYXN0aWdvdGVzIHdpdGggaW4tdml0cm8gcHJvbWFzdGlnb3RlcwoKT25lIHF1ZXJ5IHdlIGhhdmUgbm90IHlldCBhZGRyZXNzZWQ6IHdoYXQgYXJlIHRoZSBzaW1pbGFyaXRpZXMgYW5kCmRpZmZlcmVuY2VzIGFtb25nIHRoZSBzdHJhaW5zIHVzZWQgdG8gaW5mZWN0IHRoZSBtYWNyb3BoYWdlIHNhbXBsZXMKYW5kIHRoZSBwcm9tYXN0aWdvdGUgc2FtcGxlcyB1c2VkIGluIHRoZSBUTVJDMiBwYXJhc2l0ZSBkYXRhPwoKYGBge3IgY29tYmluZV9tYWNyb3BoYWdlfQojIyBJIGp1c3QgZml4ZWQgdGhpcyBpbiB0aGUgZGF0YXNldHMgUm1kLCBidXQgdW50aWwgdGhhdCBwcm9wYWdhdGVzIGp1c3Qgc2V0IGl0IG1hbnVhbGx5CmFubm90YXRpb24obHBfZXhwdCkgPC0gYW5ub3RhdGlvbihscF9tYWNyb3BoYWdlKQp0bXJjMl9tYWNyb3BoYWdlX25vcm0gPC0gbm9ybWFsaXplX2V4cHQobHBfbWFjcm9waGFnZSwgdHJhbnNmb3JtPSJsb2cyIiwgY29udmVydD0iY3BtIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5vcm09InF1YW50IiwgZmlsdGVyPVRSVUUpCmFsbF90bXJjMiA8LSBjb21iaW5lX2V4cHRzKGxwX2V4cHQsIGxwX21hY3JvcGhhZ2UpCgphbGxfbm9zYiA8LSBhbGxfdG1yYzIKcERhdGEoYWxsX25vc2IpW1sic3RhZ2UiXV0gPC0gInByb21hc3RpZ290ZSIKbmFfaWR4IDwtIGlzLm5hKHBEYXRhKGFsbF9ub3NiKVtbIm1hY3JvcGhhZ2V0cmVhdG1lbnQiXV0pCnBEYXRhKGFsbF9ub3NiKVtuYV9pZHgsICJtYWNyb3BoYWdldHJlYXRtZW50Il0gPC0gInVuZGVmaW5lZCIKYWxsX25vc2IgPC0gc3Vic2V0X2V4cHQoYWxsX25vc2IsIHN1YnNldD0ibWFjcm9waGFnZXRyZWF0bWVudCE9J2luZl9zYiciKQphbWFfaWR4IDwtIHBEYXRhKGFsbF9ub3NiKVtbIm1hY3JvcGhhZ2V0cmVhdG1lbnQiXV0gPT0gImluZiIKcERhdGEoYWxsX25vc2IpW2FtYV9pZHgsICJzdGFnZSIgXSA8LSAiYW1hc3RpZ290ZSIKcERhdGEoYWxsX25vc2IpW1siYmF0Y2giXV0gPC0gcERhdGEoYWxsX25vc2IpW1sic3RhZ2UiXV0KYGBgCgpJIHRoaW5rIHRoZSBhYm92ZSBwaWN0dXJlIGlzIHNvcnQgb2YgdGhlIG9wcG9zaXRlIG9mIHdoYXQgd2Ugd2FudCB0bwpjb21wYXJlIGluIGEgREUgYW5hbHlzaXMgZm9yIHRoaXMgc2V0IG9mIGRhdGEsIGUuZy4gd2Ugd2FudCB0byBjb21wYXJlCnByb21hc3RpZ290ZXMgZnJvbSBhbWFzdGlnb3Rlcz8KCmBgYHtyIGNvbXBhcmVfcHJvX2FtYX0KYWxsX25vc2IgPC0gc2V0X2V4cHRfYmF0Y2hlcyhhbGxfbm9zYiwgZmFjdD0iY29uZGl0aW9uIikgJT4lCiAgc2V0X2V4cHRfY29uZGl0aW9ucyhmYWN0PSJzdGFnZSIpCnR3b196eW1vIDwtIHN1YnNldF9leHB0KAogIGFsbF9ub3NiLAogIHN1YnNldD0ienltb2RlbWVjYXRlZ29yaWNhbD09J3oyMid8enltb2RlbWVjYXRlZ29yaWNhbD09J3oyMyd8enltb2RlbWVjYXRlZ29yaWNhbD09J3Vua25vd24nIikKCnByb19hbWEgPC0gYWxsX3BhaXJ3aXNlKGFsbF9ub3NiLCBmaWx0ZXI9VFJVRSwgbW9kZWxfYmF0Y2g9InN2YXNlcSIpCnByb19hbWFfdGFibGUgPC0gY29tYmluZV9kZV90YWJsZXMoCiAgICBwcm9fYW1hLAogICAgZXhjZWwgPSBnbHVlKCJleGNlbC90bXJjMl9wcm9fdnNfYW1hX3RhYmxlLXZ7dmVyfS54bHN4IikpCnByb19hbWFfc2lnIDwtIGV4dHJhY3Rfc2lnbmlmaWNhbnRfZ2VuZXMoCiAgICBwcm9fYW1hX3RhYmxlLAogICAgZXhjZWwgPSBnbHVlKCJleGNlbC90bXJjMl9wcm9fdnNfYW1hX3NpZy12e3Zlcn0ueGxzeCIpKQpgYGAKCiMjIyBQbG90IHByb21hc3RpZ290ZS9hbWFzdGlnb3RlIERFIGdlbmVzCgpgYGB7ciBwYXJhc2l0ZV9zdXNjZXB0aWJpbGl0eV9waWN0dXJlc30KcHJvX2FtYV90YWJsZVtbInBsb3RzIl1dW1sicHJvbWFzdGlnb3RlX3ZzX2FtYXN0aWdvdGUiXV1bWyJkZXNlcV9tYV9wbG90cyJdXVtbInBsb3QiXV0KYGBgCgpJIGFtIGEgbGl0dGxlIHN1cnByaXNlZCBieSB0aGlzIHBsb3QsIEkgc29tZXdoYXQgZXhwZWN0ZWQgdGhlcmUgdG8gYmUKZmV3IGdlbmVzIHdoaWNoIHBhc3NlZCB0aGUgMi1mb2xkIGRpZmZlcmVuY2UgZGVtYXJjYXRpb24gbGluZS4KCmBgYHtyIHNhdmVtZX0KaWYgKCFpc1RSVUUoZ2V0MCgic2tpcF9sb2FkIikpKSB7CiAgcGFuZGVyOjpwYW5kZXIoc2Vzc2lvbkluZm8oKSkKICBtZXNzYWdlKHBhc3RlMCgiVGhpcyBpcyBocGdsdG9vbHMgY29tbWl0OiAiLCBnZXRfZ2l0X2NvbW1pdCgpKSkKICBtZXNzYWdlKHBhc3RlMCgiU2F2aW5nIHRvICIsIHNhdmVmaWxlKSkKIyMgIHRtcCA8LSBzbShzYXZlbWUoZmlsZW5hbWUgPSBzYXZlZmlsZSkpCn0KYGBgCgpgYGB7ciBsb2FkbWVfYWZ0ZXIsIGV2YWwgPSBGQUxTRX0KdG1wIDwtIGxvYWRtZShmaWxlbmFtZSA9IHNhdmVmaWxlKQpgYGAK