Gene expression summary for Acropora pulchra sRNA-seq data.

  • trimmed reads generated in deep-dive project

  • Reads aligned to Acropora pulchra genome

0.0.1 Install and load packages

library(tidyverse)
library(ggplot2)
library(reshape2)
library(pheatmap)
library(RColorBrewer)
library(DESeq2)

1 sRNA

1.1 Load count data

Load in the sRNA count matrix generated using ShortStack 4.1.0. Keep in mind this data includes counts of all sRNAs, not just miRNAs Counts generated in 11-Apul-sRNA-ShortStack_4.1.0-pulchra_genome

# Read in sRNA counts data
Apul_counts_sRNA_data_OG <- read_delim("../output/11-Apul-sRNA-ShortStack_4.1.0-pulchra_genome/ShortStack_out/Counts.txt", delim="\t") 
head(Apul_counts_sRNA_data_OG)
# A tibble: 6 × 8
  Coords               Name  MIRNA sRNA-ACR-140-S1-TP2-…¹ sRNA-ACR-145-S1-TP2-…²
  <chr>                <chr> <chr>                  <dbl>                  <dbl>
1 ntLink_7:2229-2654   Clus… N                          2                     34
2 ntLink_7:3045-3472   Clus… N                          3                     29
3 ntLink_7:9749-10310  Clus… N                        111                     72
4 ntLink_7:21735-22163 Clus… N                          5                     24
5 ntLink_7:22554-22980 Clus… N                          3                     30
6 ntLink_7:29261-30185 Clus… N                         85                     83
# ℹ abbreviated names:
#   ¹​`sRNA-ACR-140-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed`,
#   ²​`sRNA-ACR-145-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed`
# ℹ 3 more variables:
#   `sRNA-ACR-150-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed` <dbl>,
#   `sRNA-ACR-173-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed` <dbl>,
#   `sRNA-ACR-178-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed` <dbl>

1.2 Count data munging

Apul_counts_sRNA <- Apul_counts_sRNA_data_OG

# Remove excess portions of sample column names to just "sample###"
colnames(Apul_counts_sRNA) <- sub("-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed", "", colnames(Apul_counts_sRNA))
colnames(Apul_counts_sRNA) <- sub("sRNA-ACR-", "sample", colnames(Apul_counts_sRNA))

# Keep just the counts and cluster names
Apul_counts_sRNA <- Apul_counts_sRNA %>% select("sample140", "sample145", "sample150", "sample173", "sample178", "Name")

# I'm not going to be doing any removal of low-count sRNAs for now

# Make the cluster names our new row names
Apul_counts_sRNA <- Apul_counts_sRNA %>% column_to_rownames(var = "Name")

write.table(Apul_counts_sRNA, file = "../output/03.1-Apul-sRNA-summary/Apul_sRNA_ShortStack_counts_formatted.txt", sep = "\t", row.names = TRUE, col.names = TRUE, quote = FALSE)

head(Apul_counts_sRNA)
          sample140 sample145 sample150 sample173 sample178
Cluster_1         2        34        48        25        68
Cluster_2         3        29        61        30        31
Cluster_3       111        72       219       116       162
Cluster_4         5        24        23        50        94
Cluster_5         3        30        61        32        20
Cluster_6        85        83       276       107       257

1.3 Expression levels

Plot histograms of the expression levels in each sample

# Melt the count matrix into long format
Apul_counts_sRNA_melted <- melt(Apul_counts_sRNA, variable.name = "sample", value.name = "counts")

# Plot the expression level histograms for each sample
ggplot(Apul_counts_sRNA_melted, aes(x = counts)) +
  geom_histogram(binwidth = 1, fill = "#408EC6", color = "black") +
  scale_x_log10() +  # Optional: Log-transform the x-axis for better visualization
  facet_wrap(~sample, scales = "free_y") +
  labs(title = "Gene Expression Level Histogram for Each Sample",
       x = "Expression Level (Counts)",
       y = "Frequency") +
  theme_minimal()

1.4 Transcript counts

First let’s check the total number of transcripts in each sample – keep in mind this expression data has not been normalized yet, so there may be different totals for each sample

# Calculate the total number of transcripts for each sample
total_transcripts <- colSums(Apul_counts_sRNA)

# Create a data frame for plotting
total_transcripts_df <- data.frame(sample = names(total_transcripts),
                                   totals = total_transcripts)

# Plot the total number of transcripts for each sample
ggplot(total_transcripts_df, aes(x = sample, y = totals)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = totals), vjust = -0.3, size = 3.5) + 
  labs(title = "Total Number of Transcripts per Sample",
       x = "Sample",
       y = "Total Transcripts") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

Now let’s check the number of unique transcripts in each sample – that is, how many unique sRNAs are expressed in each sample? This should be pretty much the same across samples, even without normalization.

# Calculate the number of unique transcripts (non-zero counts) for each sample
unique_transcripts <- colSums(Apul_counts_sRNA > 0)

# Create a data frame for plotting
unique_transcripts_df <- data.frame(sample = names(unique_transcripts),
                                    uniques = unique_transcripts)

# Plot the total number of unique transcripts for each sample
ggplot(unique_transcripts_df, aes(x = sample, y = uniques)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = uniques), vjust = -0.3, size = 3.5) + 
  labs(title = "Total Number of Unique Expressed Transcripts per Sample",
       x = "Sample",
       y = "Unique Transcripts") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

2 miRNA

2.1 Load miRNA metadata

The ShortStack output Results.txt includes all clusters of sRNA reads, including those not annotated as valid miRNAs. Now that we’ve looked at all the sRNAs a bit, let’s focus in on those classified as miRNAs.

## This code needs miRNA names first ##
# # Join with full metadata sheet, which only contains valid miRNAs
# Apul_metadata_miRNA <- read_csv("../../../deep-dive/DEF-cross-species/output/10-shortRNA-ShortStack-comparison/Apul_results_mature_named.csv") 
# 
# Apul_counts_sRNA <- rownames_to_column(Apul_counts_sRNA, var = "Name")
# 
# Apul_counts_miRNA <- left_join(Apul_metadata_miRNA, Apul_counts_sRNA, by = c("Name" = "Name"))
# 
# # Keep just the counts and given miRNA names (e.g., based on match to previously described miRNA)
# Apul_counts_miRNA <- Apul_counts_miRNA %>% select("sample140", "sample145", "sample150", "sample173", "sample178", "given_miRNA_name")
# 
# # Make the miRNA names our new row names
# Apul_counts_miRNA <- Apul_counts_miRNA %>% column_to_rownames(var = "given_miRNA_name")
# 
# head(Apul_counts_miRNA)

## This code can be used until we have miRNA names ##
Apul_counts_miRNA <- Apul_counts_sRNA_data_OG

# Remove excess portions of sample column names to just "sample###"
colnames(Apul_counts_miRNA) <- sub("-S1-TP2-fastp-adapters-polyG-31bp-merged_condensed", "", colnames(Apul_counts_miRNA))
colnames(Apul_counts_miRNA) <- sub("sRNA-ACR-", "sample", colnames(Apul_counts_miRNA))

# Keep only the sRNAs ID'd as valid miRNAs
Apul_counts_miRNA <- Apul_counts_miRNA %>% filter(MIRNA == "Y")

# Keep just the counts and cluster names
Apul_counts_miRNA <- Apul_counts_miRNA %>% select("sample140", "sample145", "sample150", "sample173", "sample178", "Name")

# Make the cluster names our new row names
Apul_counts_miRNA <- Apul_counts_miRNA %>% column_to_rownames(var = "Name")

write.table(Apul_counts_miRNA, file = "../output/03.1-Apul-sRNA-summary/Apul_miRNA_ShortStack_counts_formatted.txt", sep = "\t", row.names = TRUE, col.names = TRUE, quote = FALSE)

head(Apul_counts_miRNA)
             sample140 sample145 sample150 sample173 sample178
Cluster_1826      1099      2236      2579      3599      3135
Cluster_1832      1060      1886      3843      2550      1864
Cluster_1862      3575      1317       962       817       799
Cluster_1951      2503      2847      2788      3169      2800
Cluster_2463      5271     14671     28809      9634     21691
Cluster_2859       563       895      1066       901      1128

2.2 Expression levels

Plot histograms of the expression levels in each sample

# Melt the count matrix into long format
Apul_counts_miRNA_melted <- melt(Apul_counts_miRNA, variable.name = "sample", value.name = "counts")

# Plot the expression level histograms for each sample
ggplot(Apul_counts_miRNA_melted, aes(x = counts)) +
  geom_histogram(binwidth = 1, fill = "#408EC6", color = "black") +
  scale_x_log10() +  # Optional: Log-transform the x-axis for better visualization
  facet_wrap(~sample, scales = "free_y") +
  labs(title = "miRNA Expression Level Histogram for Each Sample",
       x = "Expression Level (Counts)",
       y = "Frequency") +
  theme_minimal()

2.3 miRNA counts

First let’s check the total number of miRNAs in each sample – keep in mind this expression data has not been normalized yet, so there may be different totals for each sample

# Calculate the total number of transcripts for each sample
total_miRNA <- colSums(Apul_counts_miRNA)

# Create a data frame for plotting
total_miRNA_df <- data.frame(sample = names(total_miRNA),
                                   totals = total_miRNA)

# Plot the total number of transcripts for each sample
ggplot(total_miRNA_df, aes(x = sample, y = totals)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = totals), vjust = -0.3, size = 3.5) + 
  labs(title = "Total Number of miRNAs per Sample",
       x = "Sample",
       y = "Total miRNAs") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

Now let’s check the number of unique miRNAs in each sample – This should be pretty much the same across samples, even without normalization.

# Calculate the number of unique transcripts (non-zero counts) for each sample
unique_miRNA <- colSums(Apul_counts_miRNA > 0)

# Create a data frame for plotting
unique_miRNA_df <- data.frame(sample = names(unique_miRNA),
                                    uniques = unique_miRNA)

# Plot the total number of unique transcripts for each sample
ggplot(unique_miRNA_df, aes(x = sample, y = uniques)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = uniques), vjust = -0.3, size = 3.5) + 
  labs(title = "Total Number of Unique Expressed miRNAs per Sample",
       x = "Sample",
       y = "Unique miRNA") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

2.4 Heatmap

pheatmap(Apul_counts_miRNA,
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         color = colorRampPalette(c("blue", "white", "red"))(50),
         fontsize_row = 8,
         fontsize_col = 8)

Well… there’s like 2 miRNAs with much higher expression than the others, which is making visualizing relative differences difficult. Let’s redo the heatmap, normalizing each row to view relative difference in expression between samples (scale='row')

pheatmap(Apul_counts_miRNA,
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         scale = 'row',
         color = colorRampPalette(c("blue", "white", "red"))(50),
         fontsize_row = 8,
         fontsize_col = 8)

3 siRNA

ShortStack’s primary purpose is to identify miRNAs from sRNA-seq data, but it also automatically annotates siRNA loci! Since siRNA potentially play an important role in transposon silencing in invertebrates, we should generate count matrices for siRNAs as well.

We can see clusters annotated as siRNAs in the Results.gff3 output file of ShortStack (sRNA ID shown in the 3rd column)

Apul_Resultsgff <- read.table("../output/11-Apul-sRNA-ShortStack_4.1.0-pulchra_genome/ShortStack_out/Results.gff3")
# Separate last column info into multiple columns for filtering
Apul_Resultsgff <- Apul_Resultsgff %>%
  separate(V9, into = c("Name", "DicerCall", "MIRNA"), sep = ";") %>%
  mutate(Name = sub("ID=", "", Name),
         DicerCall = sub("DicerCall=", "", DicerCall),
         MIRNA = sub("MIRNA=", "", MIRNA))
head(Apul_Resultsgff)
        V1         V2                 V3    V4    V5  V6 V7 V8      Name
1 ntLink_7 ShortStack Unknown_sRNA_locus  2229  2654 177  -  . Cluster_1
2 ntLink_7 ShortStack Unknown_sRNA_locus  3045  3472 154  -  . Cluster_2
3 ntLink_7 ShortStack Unknown_sRNA_locus  9749 10310 680  .  . Cluster_3
4 ntLink_7 ShortStack Unknown_sRNA_locus 21735 22163 196  -  . Cluster_4
5 ntLink_7 ShortStack Unknown_sRNA_locus 22554 22980 146  -  . Cluster_5
6 ntLink_7 ShortStack Unknown_sRNA_locus 29261 30185 808  .  . Cluster_6
  DicerCall MIRNA
1         N     N
2         N     N
3         N     N
4         N     N
5         N     N
6         N     N
# keep just the sRNA category column (V3), and the cluster names (Name)
# filter to only keep clusters ID'd as siRNAs
Apul_siRNA_clusters <- Apul_Resultsgff %>%
  select(V3, Name) %>%
  filter(str_detect(V3, regex("siRNA")))
head(Apul_siRNA_clusters)
             V3         Name
1 siRNA22_locus   Cluster_43
2 siRNA23_locus   Cluster_45
3 siRNA22_locus   Cluster_89
4 siRNA21_locus  Cluster_127
5 siRNA24_locus Cluster_1054
6 siRNA22_locus Cluster_1122
# Now use this list of clusters ID'd as siRNAs to filter our sRNA count matrix
# keep only the sample counts and cluster names
Apul_counts_sRNA <- rownames_to_column(Apul_counts_sRNA, var = "Name")
Apul_counts_siRNA <- left_join(Apul_siRNA_clusters, Apul_counts_sRNA, by = c("Name" = "Name")) %>%
  select(-V3)

# convert the column of cluster names into the df row names
Apul_counts_sRNA <- Apul_counts_sRNA %>% column_to_rownames(var="Name")
Apul_counts_siRNA <- Apul_counts_siRNA %>% column_to_rownames(var="Name")

head(Apul_counts_siRNA)
             sample140 sample145 sample150 sample173 sample178
Cluster_43           1       104         0        14       254
Cluster_45           0         0        45        13        60
Cluster_89          12        17         7        40         5
Cluster_127         16        18        19        28        18
Cluster_1054        42       105        47        13        39
Cluster_1122        78        25        10        11        33
write.table(Apul_counts_siRNA, file = "../output/03.1-Apul-sRNA-summary/Apul_siRNA_ShortStack_counts_formatted.txt", sep = "\t", row.names = TRUE, col.names = TRUE, quote = FALSE)

3.1 Expression levels

Plot histograms of the expression levels in each sample

# Melt the count matrix into long format
Apul_counts_siRNA_melted <- melt(Apul_counts_siRNA, variable.name = "sample", value.name = "counts")

# Plot the expression level histograms for each sample
ggplot(Apul_counts_siRNA_melted, aes(x = counts)) +
  geom_histogram(binwidth = 1, fill = "#408EC6", color = "black") +
  scale_x_log10() +  # Optional: Log-transform the x-axis for better visualization
  facet_wrap(~sample, scales = "free_y") +
  labs(title = "siRNA Expression Level Histogram for Each Sample",
       x = "Expression Level (Counts)",
       y = "Frequency") +
  theme_minimal()

3.2 siRNA counts

First let’s check the total number of siRNAs in each sample – keep in mind this expression data has not been normalized yet, so there may be different totals for each sample

# Calculate the total number of transcripts for each sample
total_siRNA <- colSums(Apul_counts_siRNA)

# Create a data frame for plotting
total_siRNA_df <- data.frame(sample = names(total_siRNA),
                                   totals = total_siRNA)

# Plot the total number of transcripts for each sample
ggplot(total_siRNA_df, aes(x = sample, y = totals)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = totals), vjust = -0.3, size = 3.5) + 
  labs(title = "Total Number of siRNAs per Sample",
       x = "Sample",
       y = "Total siRNAs") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

Now let’s check the number of unique siRNAs in each sample – This should be pretty much the same across samples, even without normalization.

# Calculate the number of unique transcripts (non-zero counts) for each sample
unique_siRNA <- colSums(Apul_counts_siRNA > 0)

# Create a data frame for plotting
unique_siRNA_df <- data.frame(sample = names(unique_siRNA),
                                    uniques = unique_siRNA)

# Plot the total number of unique transcripts for each sample
ggplot(unique_siRNA_df, aes(x = sample, y = uniques)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = uniques), vjust = -0.3, size = 3.5) + 
  labs(title = "Total Number of Unique Expressed siRNAs per Sample",
       x = "Sample",
       y = "Unique siRNA") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

3.3 Heatmap

pheatmap(Apul_counts_siRNA,
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         color = colorRampPalette(c("blue", "white", "red"))(50),
         fontsize_row = 8,
         fontsize_col = 8)

pheatmap(Apul_counts_siRNA,
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         scale = 'row',
         color = colorRampPalette(c("blue", "white", "red"))(50),
         fontsize_row = 8,
         fontsize_col = 8)

4 ………..

5 Normalized sRNA counts

5.1 Normalize counts with DESeq2

5.1.1 Plot unnormalized sRNA data

Apul_counts_sRNA %>% 
#  select(-Name) %>%
  pivot_longer( cols = everything(), names_to = "sample", values_to = "count") %>%
  ggplot(., aes(x = sample, y = count)) +
  geom_violin() + 
  geom_point(alpha = 0.2) +
  theme_minimal() +
  labs(title = "Unnormalized sRNA counts",
       x = "Sample",
       y = "count")

5.1.2 Metadata

DESeq2 requires a metadata data frame as input. I don’t have sample metadata though so, since we’re just doing DESeq2 for normalization purposes (not analysis purposes), I’m just going to create a dummy sheet

Apul_sample_names <- Apul_counts_sRNA %>%
#  select(-Name) %>%
  colnames()

Apul_metadata_sRNA <- data.frame(Sample = Apul_sample_names,
                            Species = rep("A.pulchra", 5))
rownames(Apul_metadata_sRNA) <- Apul_sample_names

head(Apul_metadata_sRNA)
             Sample   Species
sample140 sample140 A.pulchra
sample145 sample145 A.pulchra
sample150 sample150 A.pulchra
sample173 sample173 A.pulchra
sample178 sample178 A.pulchra

5.1.3 DESeq object

# Calculate DESeq object
Apul_counts_sRNA_rowNames <- Apul_counts_sRNA 
#%>% column_to_rownames(var = "Name")

dds_Apul_sRNA <- DESeqDataSetFromMatrix(countData = Apul_counts_sRNA_rowNames,
                              colData = Apul_metadata_sRNA,
                              design = ~ 1) 

# Run differential expression analysis 
# (Note that this DESeq() function runs all necessary steps, including data normalization, 
# estimating size factors, estimating dispersions, gene-wise dispersion estimates, mean-dispersion 
# relationship, final dispersion estimates, fitting model, and testing)
# Using design = ~1 because we don't have treatment groups

dds_Apul_sRNA <- DESeq(dds_Apul_sRNA)

It’s worth noting here that I’m actually going to be doing two different types of transformation on the counts data, which serve different purposes.

  • First is normalizing the transcript counts, which adjusts for differences in library size or sequencing depth, but retains count-like properties. Normalized counts are most useful for things like visualizing expression levels and differential expression analysis.

  • Second is variance stabilizing the counts data, which aims to make the variance of the transformed data approximately independent of the mean, reducing heteroscedasticity (the relationship between variance and mean) and “smoothing” out the variance at low counts. Notably, the transformed data is no longer on the original count scale. The transformation makes the variance roughly constant across the range of counts, which makes it easier to interpret patterns in the data visually. Variance stabilized data is most useful for exploratory data analysis, like PCA, clustering, and heatmaps, and is also the transformation we’ll want to use before WGCNA.

# extract normalized counts
# (normalization is automatically performed by deseq2)
Apul_counts_sRNA_norm <- counts(dds_Apul_sRNA, normalized=TRUE) %>% data.frame()

write.table(Apul_counts_sRNA_norm, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_sRNA_normalized.txt", sep = "\t", row.names = TRUE, col.names = TRUE, quote = FALSE)


# variance stabilized data
vsd_Apul_sRNA <- varianceStabilizingTransformation(dds_Apul_sRNA, blind=TRUE)
wpn_vsd_Apul_sRNA <- getVarianceStabilizedData(dds_Apul_sRNA)
rv_wpn_Apul_sRNA <- rowVars(wpn_vsd_Apul_sRNA, useNames=TRUE)

Apul_counts_sRNA_vsd <- data.frame(wpn_vsd_Apul_sRNA)
write.table(Apul_counts_sRNA_vsd, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_sRNA_variancestabilized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

q75_wpn_Apul_sRNA <- quantile(rowVars(wpn_vsd_Apul_sRNA, useNames=TRUE), .75)  # 75th quantile variability
Apul_counts_sRNA_vsd_q75 <- wpn_vsd_Apul_sRNA[ rv_wpn_Apul_sRNA > q75_wpn_Apul_sRNA, ] %>% data.frame # filter to retain only the most variable genes
write.table(Apul_counts_sRNA_vsd_q75, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_sRNA_variancestabilized_q75.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

q95_wpn_Apul_sRNA <- quantile(rowVars(wpn_vsd_Apul_sRNA, useNames=TRUE), .95)  # 95th quantile variability
Apul_counts_sRNA_vsd_q95 <- wpn_vsd_Apul_sRNA[ rv_wpn_Apul_sRNA > q95_wpn_Apul_sRNA, ] %>% data.frame # filter to retain only the most variable genes
write.table(Apul_counts_sRNA_vsd_q95, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_sRNA_variancestabilized_q95.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

5.2 Plot normalized data

Apul_counts_sRNA_norm_long <- Apul_counts_sRNA_norm %>%
  mutate(
    Gene_id = row.names(Apul_counts_sRNA_norm)
  ) %>%
  pivot_longer(-Gene_id)

Apul_counts_sRNA_norm_long %>%
  ggplot(., aes(x = name, y = value)) +
  geom_violin() +
  geom_point() +
  theme_bw() +
  theme(
    axis.text.x = element_text( angle = 90)
  ) +
  ylim(0, NA) +
  labs(
    title = "Normalized Expression",
    x = "Sample",
    y = "Normalized counts"
  )

5.3 Plot variance stabilized data

Apul_counts_sRNA_vsd_long <- Apul_counts_sRNA_vsd %>%
  mutate(
    Gene_id = row.names(Apul_counts_sRNA_vsd)
  ) %>%
  pivot_longer(-Gene_id)

Apul_counts_sRNA_vsd_long %>%
  ggplot(., aes(x = name, y = value)) +
  geom_violin() +
  geom_point() +
  theme_bw() +
  theme(
    axis.text.x = element_text( angle = 90)
  ) +
  ylim(0, NA) +
  labs(
    title = "Variance Stabilized Expression",
    x = "Sample",
    y = "Variance stabilized data"
  )

5.4 Normalized expression levels

Plot histograms of the normalized expression levels in each sample

# Melt the count matrix into long format
Apul_counts_norm_melted <- melt(Apul_counts_sRNA_norm, variable.name = "sample", value.name = "counts")

# Plot the expression level histograms for each sample
ggplot(Apul_counts_norm_melted, aes(x = counts)) +
  geom_histogram(binwidth = 1, fill = "#408EC6", color = "black") +
  scale_x_log10() +  # Optional: Log-transform the x-axis for better visualization
  facet_wrap(~sample, scales = "free_y") +
  labs(title = "Gene Expression Level Histogram for Each Sample",
       x = "Expression Level (Counts)",
       y = "Frequency") +
  theme_minimal()

5.5 Normalized transcript counts

Check the total number of transcripts in each sample – now that we’ve normalized the data these totals should be similar

# Calculate the total number of transcripts for each sample
total_transcripts_norm <- colSums(Apul_counts_sRNA_norm)

# Create a data frame for plotting
total_transcripts_norm_df <- data.frame(sample = names(total_transcripts_norm),
                                   totals = total_transcripts_norm)

# Plot the total number of transcripts for each sample
ggplot(total_transcripts_norm_df, aes(x = sample, y = totals)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = totals), vjust = -0.3, size = 3.5) +
  labs(title = "Total Number of Transcripts per Sample",
       x = "Sample",
       y = "Total Transcripts") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

5.6 PCA of variance stabilized data

plotPCA(vsd_Apul_sRNA, intgroup="Sample")

5.7 Sample clustering

sample_dists <- dist(t(assay(vsd_Apul_sRNA)))
pheatmap(as.matrix(sample_dists), clustering_distance_rows = "euclidean", 
         clustering_distance_cols = "euclidean", main="Sample Clustering")

5.8 Heatmaps

Of most variable variance stabilized sRNA transcripts

# 75th quantile
heat_colors <- rev(brewer.pal(12, "RdYlBu"))
pheatmap(Apul_counts_sRNA_vsd_q75, 
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         color = heat_colors,
         scale="row")

# 95th quantile
pheatmap(Apul_counts_sRNA_vsd_q95, 
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         color = heat_colors,
         scale="row")

6 Normalized miRNA counts

6.1 Isolate normalized/vsd miRNA

## Also written to use given miRNA names ##
# Apul_counts_sRNA_norm$Name <- rownames(Apul_counts_sRNA_norm)
# Apul_counts_miRNA_norm <- left_join(Apul_metadata_miRNA, Apul_counts_sRNA_norm, by = c("Name" = "Name")) %>%
#   column_to_rownames(var="given_miRNA_name") %>%
#   select(starts_with("sample"))
# write.table(Apul_counts_miRNA_norm, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_miRNA_normalized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)
# 
# Apul_counts_sRNA_vsd$Name <- rownames(Apul_counts_sRNA_vsd)
# Apul_counts_miRNA_vsd <- left_join(Apul_metadata_miRNA, Apul_counts_sRNA_vsd, by = c("Name" = "Name")) %>%
#   column_to_rownames(var="given_miRNA_name") %>%
#   select(starts_with("sample"))
# write.table(Apul_counts_miRNA_vsd, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_miRNA_variancestabilized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

## Use below until you have miRNA names ##
Apul_counts_sRNA_norm$Name <- rownames(Apul_counts_sRNA_norm)
Apul_counts_sRNA_vsd$Name <- rownames(Apul_counts_sRNA_vsd)

Apul_counts_miRNA_namesdf <- data.frame(Name = rownames(Apul_counts_miRNA)) 

Apul_counts_miRNA_norm <- left_join(Apul_counts_miRNA_namesdf, Apul_counts_sRNA_norm, by = c("Name" = "Name")) %>%
  column_to_rownames(var = "Name")
write.table(Apul_counts_miRNA_norm, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_miRNA_normalized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

Apul_counts_miRNA_vsd <- left_join(Apul_counts_miRNA_namesdf, Apul_counts_sRNA_vsd, by = c("Name" = "Name")) %>%
  column_to_rownames(var = "Name")
write.table(Apul_counts_miRNA_vsd, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_miRNA_variancestabilized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

6.2 Normalized expression levels

Plot histograms of the normalized expression levels in each sample

# Melt the count matrix into long format
Apul_counts_miRNA_norm_melted <- melt(Apul_counts_miRNA_norm, variable.name = "sample", value.name = "counts")

# Plot the expression level histograms for each sample
ggplot(Apul_counts_miRNA_norm_melted, aes(x = counts)) +
  geom_histogram(binwidth = 1, fill = "#408EC6", color = "black") +
  scale_x_log10() +  # Optional: Log-transform the x-axis for better visualization
  facet_wrap(~sample, scales = "free_y") +
  labs(title = "Gene Expression Level Histogram for Each Sample",
       x = "Expression Level (Counts)",
       y = "Frequency") +
  theme_minimal()

6.3 Normalized transcript counts

Check the total number of transcripts in each sample – now that we’ve normalized the data these totals should be similar

# Calculate the total number of transcripts for each sample
total_transcripts_miRNA_norm <- colSums(Apul_counts_miRNA_norm)

# Create a data frame for plotting
total_transcripts_miRNA_norm_df <- data.frame(sample = names(total_transcripts_miRNA_norm),
                                   totals = total_transcripts_miRNA_norm)

# Plot the total number of transcripts for each sample
ggplot(total_transcripts_miRNA_norm_df, aes(x = sample, y = totals)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = totals), vjust = -0.3, size = 3.5) +
  labs(title = "Total Number of miRNA Transcripts per Sample",
       x = "Sample",
       y = "Total Transcripts") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

6.4 Heatmap

Of all miRNAs

heat_colors <- rev(brewer.pal(12, "RdYlBu"))
pheatmap(as.matrix(Apul_counts_miRNA_vsd[apply(Apul_counts_miRNA_vsd, 1, var) > 0, ]), 
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         color = heat_colors,
         scale="row")

7 Normalized siRNA counts

7.1 Isolate normalized/vsd siRNA

Apul_counts_sRNA_norm$Name <- rownames(Apul_counts_sRNA_norm)
Apul_counts_sRNA_vsd$Name <- rownames(Apul_counts_sRNA_vsd)

Apul_counts_siRNA_namesdf <- data.frame(Name = rownames(Apul_counts_siRNA)) 

Apul_counts_siRNA_norm <- left_join(Apul_counts_siRNA_namesdf, Apul_counts_sRNA_norm, by = c("Name" = "Name")) %>%
  column_to_rownames(var = "Name")
write.table(Apul_counts_siRNA_norm, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_siRNA_normalized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

Apul_counts_siRNA_vsd <- left_join(Apul_counts_siRNA_namesdf, Apul_counts_sRNA_vsd, by = c("Name" = "Name")) %>%
  column_to_rownames(var = "Name")
write.table(Apul_counts_siRNA_vsd, file = "../output/03.1-Apul-sRNA-summary/Apul_counts_siRNA_variancestabilized.txt", sep = "\t", row.names = TRUE, col.names = TRUE,quote = FALSE)

7.2 Normalized expression levels

Plot histograms of the normalized expression levels in each sample

# Melt the count matrix into long format
Apul_counts_siRNA_norm_melted <- melt(Apul_counts_siRNA_norm, variable.name = "sample", value.name = "counts")

# Plot the expression level histograms for each sample
ggplot(Apul_counts_siRNA_norm_melted, aes(x = counts)) +
  geom_histogram(binwidth = 1, fill = "#408EC6", color = "black") +
  scale_x_log10() +  # Optional: Log-transform the x-axis for better visualization
  facet_wrap(~sample, scales = "free_y") +
  labs(title = "Gene Expression Level Histogram for Each Sample",
       x = "Expression Level (Counts)",
       y = "Frequency") +
  theme_minimal()

7.3 Normalized transcript counts

Check the total number of transcripts in each sample – now that we’ve normalized the data these totals should be similar

# Calculate the total number of transcripts for each sample
total_transcripts_siRNA_norm <- colSums(Apul_counts_siRNA_norm)

# Create a data frame for plotting
total_transcripts_siRNA_norm_df <- data.frame(sample = names(total_transcripts_siRNA_norm),
                                   totals = total_transcripts_siRNA_norm)

# Plot the total number of transcripts for each sample
ggplot(total_transcripts_siRNA_norm_df, aes(x = sample, y = totals)) +
  geom_bar(stat = "identity", fill = "#408EC6", color = "black") +
  geom_text(aes(label = totals), vjust = -0.3, size = 3.5) +
  labs(title = "Total Number of siRNA Transcripts per Sample",
       x = "Sample",
       y = "Total Transcripts") +
  theme_minimal() +
  theme(axis.text.x = element_text(angle = 45, hjust = 1))  # Rotate x-axis labels for readability

7.4 Heatmap

Of all siRNAs

heat_colors <- rev(brewer.pal(12, "RdYlBu"))
pheatmap(as.matrix(Apul_counts_siRNA_vsd[apply(Apul_counts_siRNA_vsd, 1, var) > 0, ]), 
         cluster_rows = TRUE,
         cluster_cols = TRUE,
         show_rownames = TRUE,
         show_colnames = TRUE,
         color = heat_colors,
         scale="row")

LS0tCnRpdGxlOiAiMDMuMS1BcHVsLXNSTkEtc3VtbWFyeSIKYXV0aG9yOiAiS2F0aGxlZW4gRHVya2luIgpkYXRlOiAiMjAyNC0wOS0wNSIKYWx3YXlzX2FsbG93X2h0bWw6IHRydWUKb3V0cHV0OiAKICBib29rZG93bjo6aHRtbF9kb2N1bWVudDI6CiAgICB0aGVtZTogY29zbW8KICAgIHRvYzogdHJ1ZQogICAgdG9jX2Zsb2F0OiB0cnVlCiAgICBudW1iZXJfc2VjdGlvbnM6IHRydWUKICAgIGNvZGVfZm9sZGluZzogc2hvdwogICAgY29kZV9kb3dubG9hZDogdHJ1ZQogIGdpdGh1Yl9kb2N1bWVudDoKICAgIHRvYzogdHJ1ZQogICAgdG9jX2RlcHRoOiAzCiAgICBudW1iZXJfc2VjdGlvbnM6IHRydWUKICAgIGh0bWxfcHJldmlldzogdHJ1ZSAKLS0tCgpgYGB7ciBzZXR1cCwgaW5jbHVkZT1GQUxTRX0KbGlicmFyeShrbml0cikKa25pdHI6Om9wdHNfY2h1bmskc2V0KAogIGVjaG8gPSBUUlVFLCAgICAgICAgICMgRGlzcGxheSBjb2RlIGNodW5rcwogIGV2YWwgPSBUUlVFLCAgICAgICAgICMgRXZhbHVhdGUgY29kZSBjaHVua3MKICB3YXJuaW5nID0gRkFMU0UsICAgICAjIEhpZGUgd2FybmluZ3MKICBtZXNzYWdlID0gRkFMU0UsICAgICAjIEhpZGUgbWVzc2FnZXMKICBjb21tZW50ID0gIiIgICAgICAgICAjIFByZXZlbnRzIGFwcGVuZGluZyAnIyMnIHRvIGJlZ2lubmluZyBvZiBsaW5lcyBpbiBjb2RlIG91dHB1dAopCmBgYAoKR2VuZSBleHByZXNzaW9uIHN1bW1hcnkgZm9yICpBY3JvcG9yYSBwdWxjaHJhKiBzUk5BLXNlcSBkYXRhLgoKLSAgIHRyaW1tZWQgcmVhZHMgZ2VuZXJhdGVkIGluIGBkZWVwLWRpdmVgIHByb2plY3QKCi0gICBSZWFkcyBhbGlnbmVkIHRvICpBY3JvcG9yYSBwdWxjaHJhKiBnZW5vbWUgCgojIyMgSW5zdGFsbCBhbmQgbG9hZCBwYWNrYWdlcwoKYGBge3IgbG9hZF9saWJyYXJpZXMsIGlubGN1ZGUgPSBUUlVFfQpsaWJyYXJ5KHRpZHl2ZXJzZSkKbGlicmFyeShnZ3Bsb3QyKQpsaWJyYXJ5KHJlc2hhcGUyKQpsaWJyYXJ5KHBoZWF0bWFwKQpsaWJyYXJ5KFJDb2xvckJyZXdlcikKbGlicmFyeShERVNlcTIpCmBgYAoKCiMgc1JOQQoKIyMgTG9hZCBjb3VudCBkYXRhCgpMb2FkIGluIHRoZSBzUk5BIGNvdW50IG1hdHJpeCBnZW5lcmF0ZWQgdXNpbmcgU2hvcnRTdGFjayA0LjEuMC4gS2VlcCBpbiBtaW5kIHRoaXMgZGF0YSBpbmNsdWRlcyBjb3VudHMgb2YgYWxsIHNSTkFzLCBub3QganVzdCBtaVJOQXMKQ291bnRzIGdlbmVyYXRlZCBpbiBgMTEtQXB1bC1zUk5BLVNob3J0U3RhY2tfNC4xLjAtcHVsY2hyYV9nZW5vbWVgCgpgYGB7ciBsb2FkLXNSTkEtY291bnRzfQojIFJlYWQgaW4gc1JOQSBjb3VudHMgZGF0YQpBcHVsX2NvdW50c19zUk5BX2RhdGFfT0cgPC0gcmVhZF9kZWxpbSgiLi4vb3V0cHV0LzExLUFwdWwtc1JOQS1TaG9ydFN0YWNrXzQuMS4wLXB1bGNocmFfZ2Vub21lL1Nob3J0U3RhY2tfb3V0L0NvdW50cy50eHQiLCBkZWxpbT0iXHQiKSAKaGVhZChBcHVsX2NvdW50c19zUk5BX2RhdGFfT0cpCmBgYAoKIyMgQ291bnQgZGF0YSBtdW5naW5nCgpgYGB7ciBzUk5BLWNvdW50LWRhdGEtbXVuZ2luZ30KQXB1bF9jb3VudHNfc1JOQSA8LSBBcHVsX2NvdW50c19zUk5BX2RhdGFfT0cKCiMgUmVtb3ZlIGV4Y2VzcyBwb3J0aW9ucyBvZiBzYW1wbGUgY29sdW1uIG5hbWVzIHRvIGp1c3QgInNhbXBsZSMjIyIKY29sbmFtZXMoQXB1bF9jb3VudHNfc1JOQSkgPC0gc3ViKCItUzEtVFAyLWZhc3RwLWFkYXB0ZXJzLXBvbHlHLTMxYnAtbWVyZ2VkX2NvbmRlbnNlZCIsICIiLCBjb2xuYW1lcyhBcHVsX2NvdW50c19zUk5BKSkKY29sbmFtZXMoQXB1bF9jb3VudHNfc1JOQSkgPC0gc3ViKCJzUk5BLUFDUi0iLCAic2FtcGxlIiwgY29sbmFtZXMoQXB1bF9jb3VudHNfc1JOQSkpCgojIEtlZXAganVzdCB0aGUgY291bnRzIGFuZCBjbHVzdGVyIG5hbWVzCkFwdWxfY291bnRzX3NSTkEgPC0gQXB1bF9jb3VudHNfc1JOQSAlPiUgc2VsZWN0KCJzYW1wbGUxNDAiLCAic2FtcGxlMTQ1IiwgInNhbXBsZTE1MCIsICJzYW1wbGUxNzMiLCAic2FtcGxlMTc4IiwgIk5hbWUiKQoKIyBJJ20gbm90IGdvaW5nIHRvIGJlIGRvaW5nIGFueSByZW1vdmFsIG9mIGxvdy1jb3VudCBzUk5BcyBmb3Igbm93CgojIE1ha2UgdGhlIGNsdXN0ZXIgbmFtZXMgb3VyIG5ldyByb3cgbmFtZXMKQXB1bF9jb3VudHNfc1JOQSA8LSBBcHVsX2NvdW50c19zUk5BICU+JSBjb2x1bW5fdG9fcm93bmFtZXModmFyID0gIk5hbWUiKQoKd3JpdGUudGFibGUoQXB1bF9jb3VudHNfc1JOQSwgZmlsZSA9ICIuLi9vdXRwdXQvMDMuMS1BcHVsLXNSTkEtc3VtbWFyeS9BcHVsX3NSTkFfU2hvcnRTdGFja19jb3VudHNfZm9ybWF0dGVkLnR4dCIsIHNlcCA9ICJcdCIsIHJvdy5uYW1lcyA9IFRSVUUsIGNvbC5uYW1lcyA9IFRSVUUsIHF1b3RlID0gRkFMU0UpCgpoZWFkKEFwdWxfY291bnRzX3NSTkEpCmBgYAoKCiMjIEV4cHJlc3Npb24gbGV2ZWxzCgpQbG90IGhpc3RvZ3JhbXMgb2YgdGhlIGV4cHJlc3Npb24gbGV2ZWxzIGluIGVhY2ggc2FtcGxlCgpgYGB7ciBleHByZXNzaW9uLWxldmVsLWhpc3RvZ3JhbXN9CiMgTWVsdCB0aGUgY291bnQgbWF0cml4IGludG8gbG9uZyBmb3JtYXQKQXB1bF9jb3VudHNfc1JOQV9tZWx0ZWQgPC0gbWVsdChBcHVsX2NvdW50c19zUk5BLCB2YXJpYWJsZS5uYW1lID0gInNhbXBsZSIsIHZhbHVlLm5hbWUgPSAiY291bnRzIikKCiMgUGxvdCB0aGUgZXhwcmVzc2lvbiBsZXZlbCBoaXN0b2dyYW1zIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QoQXB1bF9jb3VudHNfc1JOQV9tZWx0ZWQsIGFlcyh4ID0gY291bnRzKSkgKwogIGdlb21faGlzdG9ncmFtKGJpbndpZHRoID0gMSwgZmlsbCA9ICIjNDA4RUM2IiwgY29sb3IgPSAiYmxhY2siKSArCiAgc2NhbGVfeF9sb2cxMCgpICsgICMgT3B0aW9uYWw6IExvZy10cmFuc2Zvcm0gdGhlIHgtYXhpcyBmb3IgYmV0dGVyIHZpc3VhbGl6YXRpb24KICBmYWNldF93cmFwKH5zYW1wbGUsIHNjYWxlcyA9ICJmcmVlX3kiKSArCiAgbGFicyh0aXRsZSA9ICJHZW5lIEV4cHJlc3Npb24gTGV2ZWwgSGlzdG9ncmFtIGZvciBFYWNoIFNhbXBsZSIsCiAgICAgICB4ID0gIkV4cHJlc3Npb24gTGV2ZWwgKENvdW50cykiLAogICAgICAgeSA9ICJGcmVxdWVuY3kiKSArCiAgdGhlbWVfbWluaW1hbCgpCmBgYAoKIyMgVHJhbnNjcmlwdCBjb3VudHMKCkZpcnN0IGxldCdzIGNoZWNrIHRoZSB0b3RhbCBudW1iZXIgb2YgdHJhbnNjcmlwdHMgaW4gZWFjaCBzYW1wbGUgLS0ga2VlcCBpbiBtaW5kIHRoaXMgZXhwcmVzc2lvbiBkYXRhIGhhcyAqbm90KiBiZWVuIG5vcm1hbGl6ZWQgeWV0LCBzbyB0aGVyZSBtYXkgYmUgZGlmZmVyZW50IHRvdGFscyBmb3IgZWFjaCBzYW1wbGUKYGBge3IgdHJhbnNjcmlwdC1jb3VudHMtcGxvdH0KIyBDYWxjdWxhdGUgdGhlIHRvdGFsIG51bWJlciBvZiB0cmFuc2NyaXB0cyBmb3IgZWFjaCBzYW1wbGUKdG90YWxfdHJhbnNjcmlwdHMgPC0gY29sU3VtcyhBcHVsX2NvdW50c19zUk5BKQoKIyBDcmVhdGUgYSBkYXRhIGZyYW1lIGZvciBwbG90dGluZwp0b3RhbF90cmFuc2NyaXB0c19kZiA8LSBkYXRhLmZyYW1lKHNhbXBsZSA9IG5hbWVzKHRvdGFsX3RyYW5zY3JpcHRzKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b3RhbHMgPSB0b3RhbF90cmFuc2NyaXB0cykKCiMgUGxvdCB0aGUgdG90YWwgbnVtYmVyIG9mIHRyYW5zY3JpcHRzIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QodG90YWxfdHJhbnNjcmlwdHNfZGYsIGFlcyh4ID0gc2FtcGxlLCB5ID0gdG90YWxzKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBnZW9tX3RleHQoYWVzKGxhYmVsID0gdG90YWxzKSwgdmp1c3QgPSAtMC4zLCBzaXplID0gMy41KSArIAogIGxhYnModGl0bGUgPSAiVG90YWwgTnVtYmVyIG9mIFRyYW5zY3JpcHRzIHBlciBTYW1wbGUiLAogICAgICAgeCA9ICJTYW1wbGUiLAogICAgICAgeSA9ICJUb3RhbCBUcmFuc2NyaXB0cyIpICsKICB0aGVtZV9taW5pbWFsKCkgKwogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUsIGhqdXN0ID0gMSkpICAjIFJvdGF0ZSB4LWF4aXMgbGFiZWxzIGZvciByZWFkYWJpbGl0eQpgYGAKCk5vdyBsZXQncyBjaGVjayB0aGUgbnVtYmVyIG9mIHVuaXF1ZSB0cmFuc2NyaXB0cyBpbiBlYWNoIHNhbXBsZSAtLSB0aGF0IGlzLCBob3cgbWFueSB1bmlxdWUgc1JOQXMgYXJlIGV4cHJlc3NlZCBpbiBlYWNoIHNhbXBsZT8gVGhpcyBzaG91bGQgYmUgcHJldHR5IG11Y2ggdGhlIHNhbWUgYWNyb3NzIHNhbXBsZXMsIGV2ZW4gd2l0aG91dCBub3JtYWxpemF0aW9uLgoKYGBge3IgdG90YWwtdW5pcXVlLXRyYW5zY3JpcHRzLXBsb3R9CiMgQ2FsY3VsYXRlIHRoZSBudW1iZXIgb2YgdW5pcXVlIHRyYW5zY3JpcHRzIChub24temVybyBjb3VudHMpIGZvciBlYWNoIHNhbXBsZQp1bmlxdWVfdHJhbnNjcmlwdHMgPC0gY29sU3VtcyhBcHVsX2NvdW50c19zUk5BID4gMCkKCiMgQ3JlYXRlIGEgZGF0YSBmcmFtZSBmb3IgcGxvdHRpbmcKdW5pcXVlX3RyYW5zY3JpcHRzX2RmIDwtIGRhdGEuZnJhbWUoc2FtcGxlID0gbmFtZXModW5pcXVlX3RyYW5zY3JpcHRzKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdW5pcXVlcyA9IHVuaXF1ZV90cmFuc2NyaXB0cykKCiMgUGxvdCB0aGUgdG90YWwgbnVtYmVyIG9mIHVuaXF1ZSB0cmFuc2NyaXB0cyBmb3IgZWFjaCBzYW1wbGUKZ2dwbG90KHVuaXF1ZV90cmFuc2NyaXB0c19kZiwgYWVzKHggPSBzYW1wbGUsIHkgPSB1bmlxdWVzKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBnZW9tX3RleHQoYWVzKGxhYmVsID0gdW5pcXVlcyksIHZqdXN0ID0gLTAuMywgc2l6ZSA9IDMuNSkgKyAKICBsYWJzKHRpdGxlID0gIlRvdGFsIE51bWJlciBvZiBVbmlxdWUgRXhwcmVzc2VkIFRyYW5zY3JpcHRzIHBlciBTYW1wbGUiLAogICAgICAgeCA9ICJTYW1wbGUiLAogICAgICAgeSA9ICJVbmlxdWUgVHJhbnNjcmlwdHMiKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSAgIyBSb3RhdGUgeC1heGlzIGxhYmVscyBmb3IgcmVhZGFiaWxpdHkKYGBgCiAgIAoKIyBtaVJOQQoKIyMgTG9hZCBtaVJOQSBtZXRhZGF0YQoKVGhlIFNob3J0U3RhY2sgb3V0cHV0IFJlc3VsdHMudHh0IGluY2x1ZGVzIGFsbCBjbHVzdGVycyBvZiBzUk5BIHJlYWRzLCBpbmNsdWRpbmcgdGhvc2Ugbm90IGFubm90YXRlZCBhcyB2YWxpZCBtaVJOQXMuIE5vdyB0aGF0IHdlJ3ZlIGxvb2tlZCBhdCBhbGwgdGhlIHNSTkFzIGEgYml0LCBsZXQncyBmb2N1cyBpbiBvbiB0aG9zZSBjbGFzc2lmaWVkIGFzIG1pUk5Bcy4KCmBgYHtyIG1pUk5BLWNvdW50LWRhdGEtbXVuZ2luZ30KIyMgVGhpcyBjb2RlIG5lZWRzIG1pUk5BIG5hbWVzIGZpcnN0ICMjCiMgIyBKb2luIHdpdGggZnVsbCBtZXRhZGF0YSBzaGVldCwgd2hpY2ggb25seSBjb250YWlucyB2YWxpZCBtaVJOQXMKIyBBcHVsX21ldGFkYXRhX21pUk5BIDwtIHJlYWRfY3N2KCIuLi8uLi8uLi9kZWVwLWRpdmUvREVGLWNyb3NzLXNwZWNpZXMvb3V0cHV0LzEwLXNob3J0Uk5BLVNob3J0U3RhY2stY29tcGFyaXNvbi9BcHVsX3Jlc3VsdHNfbWF0dXJlX25hbWVkLmNzdiIpIAojIAojIEFwdWxfY291bnRzX3NSTkEgPC0gcm93bmFtZXNfdG9fY29sdW1uKEFwdWxfY291bnRzX3NSTkEsIHZhciA9ICJOYW1lIikKIyAKIyBBcHVsX2NvdW50c19taVJOQSA8LSBsZWZ0X2pvaW4oQXB1bF9tZXRhZGF0YV9taVJOQSwgQXB1bF9jb3VudHNfc1JOQSwgYnkgPSBjKCJOYW1lIiA9ICJOYW1lIikpCiMgCiMgIyBLZWVwIGp1c3QgdGhlIGNvdW50cyBhbmQgZ2l2ZW4gbWlSTkEgbmFtZXMgKGUuZy4sIGJhc2VkIG9uIG1hdGNoIHRvIHByZXZpb3VzbHkgZGVzY3JpYmVkIG1pUk5BKQojIEFwdWxfY291bnRzX21pUk5BIDwtIEFwdWxfY291bnRzX21pUk5BICU+JSBzZWxlY3QoInNhbXBsZTE0MCIsICJzYW1wbGUxNDUiLCAic2FtcGxlMTUwIiwgInNhbXBsZTE3MyIsICJzYW1wbGUxNzgiLCAiZ2l2ZW5fbWlSTkFfbmFtZSIpCiMgCiMgIyBNYWtlIHRoZSBtaVJOQSBuYW1lcyBvdXIgbmV3IHJvdyBuYW1lcwojIEFwdWxfY291bnRzX21pUk5BIDwtIEFwdWxfY291bnRzX21pUk5BICU+JSBjb2x1bW5fdG9fcm93bmFtZXModmFyID0gImdpdmVuX21pUk5BX25hbWUiKQojIAojIGhlYWQoQXB1bF9jb3VudHNfbWlSTkEpCgojIyBUaGlzIGNvZGUgY2FuIGJlIHVzZWQgdW50aWwgd2UgaGF2ZSBtaVJOQSBuYW1lcyAjIwpBcHVsX2NvdW50c19taVJOQSA8LSBBcHVsX2NvdW50c19zUk5BX2RhdGFfT0cKCiMgUmVtb3ZlIGV4Y2VzcyBwb3J0aW9ucyBvZiBzYW1wbGUgY29sdW1uIG5hbWVzIHRvIGp1c3QgInNhbXBsZSMjIyIKY29sbmFtZXMoQXB1bF9jb3VudHNfbWlSTkEpIDwtIHN1YigiLVMxLVRQMi1mYXN0cC1hZGFwdGVycy1wb2x5Ry0zMWJwLW1lcmdlZF9jb25kZW5zZWQiLCAiIiwgY29sbmFtZXMoQXB1bF9jb3VudHNfbWlSTkEpKQpjb2xuYW1lcyhBcHVsX2NvdW50c19taVJOQSkgPC0gc3ViKCJzUk5BLUFDUi0iLCAic2FtcGxlIiwgY29sbmFtZXMoQXB1bF9jb3VudHNfbWlSTkEpKQoKIyBLZWVwIG9ubHkgdGhlIHNSTkFzIElEJ2QgYXMgdmFsaWQgbWlSTkFzCkFwdWxfY291bnRzX21pUk5BIDwtIEFwdWxfY291bnRzX21pUk5BICU+JSBmaWx0ZXIoTUlSTkEgPT0gIlkiKQoKIyBLZWVwIGp1c3QgdGhlIGNvdW50cyBhbmQgY2x1c3RlciBuYW1lcwpBcHVsX2NvdW50c19taVJOQSA8LSBBcHVsX2NvdW50c19taVJOQSAlPiUgc2VsZWN0KCJzYW1wbGUxNDAiLCAic2FtcGxlMTQ1IiwgInNhbXBsZTE1MCIsICJzYW1wbGUxNzMiLCAic2FtcGxlMTc4IiwgIk5hbWUiKQoKIyBNYWtlIHRoZSBjbHVzdGVyIG5hbWVzIG91ciBuZXcgcm93IG5hbWVzCkFwdWxfY291bnRzX21pUk5BIDwtIEFwdWxfY291bnRzX21pUk5BICU+JSBjb2x1bW5fdG9fcm93bmFtZXModmFyID0gIk5hbWUiKQoKd3JpdGUudGFibGUoQXB1bF9jb3VudHNfbWlSTkEsIGZpbGUgPSAiLi4vb3V0cHV0LzAzLjEtQXB1bC1zUk5BLXN1bW1hcnkvQXB1bF9taVJOQV9TaG9ydFN0YWNrX2NvdW50c19mb3JtYXR0ZWQudHh0Iiwgc2VwID0gIlx0Iiwgcm93Lm5hbWVzID0gVFJVRSwgY29sLm5hbWVzID0gVFJVRSwgcXVvdGUgPSBGQUxTRSkKCmhlYWQoQXB1bF9jb3VudHNfbWlSTkEpCgpgYGAKCiMjIEV4cHJlc3Npb24gbGV2ZWxzCgpQbG90IGhpc3RvZ3JhbXMgb2YgdGhlIGV4cHJlc3Npb24gbGV2ZWxzIGluIGVhY2ggc2FtcGxlCgpgYGB7ciBtaVJOQS1leHByZXNzaW9uLWxldmVsLWhpc3RvZ3JhbXN9CiMgTWVsdCB0aGUgY291bnQgbWF0cml4IGludG8gbG9uZyBmb3JtYXQKQXB1bF9jb3VudHNfbWlSTkFfbWVsdGVkIDwtIG1lbHQoQXB1bF9jb3VudHNfbWlSTkEsIHZhcmlhYmxlLm5hbWUgPSAic2FtcGxlIiwgdmFsdWUubmFtZSA9ICJjb3VudHMiKQoKIyBQbG90IHRoZSBleHByZXNzaW9uIGxldmVsIGhpc3RvZ3JhbXMgZm9yIGVhY2ggc2FtcGxlCmdncGxvdChBcHVsX2NvdW50c19taVJOQV9tZWx0ZWQsIGFlcyh4ID0gY291bnRzKSkgKwogIGdlb21faGlzdG9ncmFtKGJpbndpZHRoID0gMSwgZmlsbCA9ICIjNDA4RUM2IiwgY29sb3IgPSAiYmxhY2siKSArCiAgc2NhbGVfeF9sb2cxMCgpICsgICMgT3B0aW9uYWw6IExvZy10cmFuc2Zvcm0gdGhlIHgtYXhpcyBmb3IgYmV0dGVyIHZpc3VhbGl6YXRpb24KICBmYWNldF93cmFwKH5zYW1wbGUsIHNjYWxlcyA9ICJmcmVlX3kiKSArCiAgbGFicyh0aXRsZSA9ICJtaVJOQSBFeHByZXNzaW9uIExldmVsIEhpc3RvZ3JhbSBmb3IgRWFjaCBTYW1wbGUiLAogICAgICAgeCA9ICJFeHByZXNzaW9uIExldmVsIChDb3VudHMpIiwKICAgICAgIHkgPSAiRnJlcXVlbmN5IikgKwogIHRoZW1lX21pbmltYWwoKQpgYGAKCiMjIG1pUk5BIGNvdW50cwoKRmlyc3QgbGV0J3MgY2hlY2sgdGhlIHRvdGFsIG51bWJlciBvZiBtaVJOQXMgaW4gZWFjaCBzYW1wbGUgLS0ga2VlcCBpbiBtaW5kIHRoaXMgZXhwcmVzc2lvbiBkYXRhIGhhcyAqbm90KiBiZWVuIG5vcm1hbGl6ZWQgeWV0LCBzbyB0aGVyZSBtYXkgYmUgZGlmZmVyZW50IHRvdGFscyBmb3IgZWFjaCBzYW1wbGUKYGBge3IgbWlSTkEtY291bnRzLXBsb3R9CiMgQ2FsY3VsYXRlIHRoZSB0b3RhbCBudW1iZXIgb2YgdHJhbnNjcmlwdHMgZm9yIGVhY2ggc2FtcGxlCnRvdGFsX21pUk5BIDwtIGNvbFN1bXMoQXB1bF9jb3VudHNfbWlSTkEpCgojIENyZWF0ZSBhIGRhdGEgZnJhbWUgZm9yIHBsb3R0aW5nCnRvdGFsX21pUk5BX2RmIDwtIGRhdGEuZnJhbWUoc2FtcGxlID0gbmFtZXModG90YWxfbWlSTkEpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRvdGFscyA9IHRvdGFsX21pUk5BKQoKIyBQbG90IHRoZSB0b3RhbCBudW1iZXIgb2YgdHJhbnNjcmlwdHMgZm9yIGVhY2ggc2FtcGxlCmdncGxvdCh0b3RhbF9taVJOQV9kZiwgYWVzKHggPSBzYW1wbGUsIHkgPSB0b3RhbHMpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIsIGZpbGwgPSAiIzQwOEVDNiIsIGNvbG9yID0gImJsYWNrIikgKwogIGdlb21fdGV4dChhZXMobGFiZWwgPSB0b3RhbHMpLCB2anVzdCA9IC0wLjMsIHNpemUgPSAzLjUpICsgCiAgbGFicyh0aXRsZSA9ICJUb3RhbCBOdW1iZXIgb2YgbWlSTkFzIHBlciBTYW1wbGUiLAogICAgICAgeCA9ICJTYW1wbGUiLAogICAgICAgeSA9ICJUb3RhbCBtaVJOQXMiKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSAgIyBSb3RhdGUgeC1heGlzIGxhYmVscyBmb3IgcmVhZGFiaWxpdHkKYGBgCgpOb3cgbGV0J3MgY2hlY2sgdGhlIG51bWJlciBvZiB1bmlxdWUgbWlSTkFzIGluIGVhY2ggc2FtcGxlIC0tIFRoaXMgc2hvdWxkIGJlIHByZXR0eSBtdWNoIHRoZSBzYW1lIGFjcm9zcyBzYW1wbGVzLCBldmVuIHdpdGhvdXQgbm9ybWFsaXphdGlvbi4KCmBgYHtyIHRvdGFsLXVuaXF1ZS1taVJOQS1wbG90fQojIENhbGN1bGF0ZSB0aGUgbnVtYmVyIG9mIHVuaXF1ZSB0cmFuc2NyaXB0cyAobm9uLXplcm8gY291bnRzKSBmb3IgZWFjaCBzYW1wbGUKdW5pcXVlX21pUk5BIDwtIGNvbFN1bXMoQXB1bF9jb3VudHNfbWlSTkEgPiAwKQoKIyBDcmVhdGUgYSBkYXRhIGZyYW1lIGZvciBwbG90dGluZwp1bmlxdWVfbWlSTkFfZGYgPC0gZGF0YS5mcmFtZShzYW1wbGUgPSBuYW1lcyh1bmlxdWVfbWlSTkEpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1bmlxdWVzID0gdW5pcXVlX21pUk5BKQoKIyBQbG90IHRoZSB0b3RhbCBudW1iZXIgb2YgdW5pcXVlIHRyYW5zY3JpcHRzIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QodW5pcXVlX21pUk5BX2RmLCBhZXMoeCA9IHNhbXBsZSwgeSA9IHVuaXF1ZXMpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIsIGZpbGwgPSAiIzQwOEVDNiIsIGNvbG9yID0gImJsYWNrIikgKwogIGdlb21fdGV4dChhZXMobGFiZWwgPSB1bmlxdWVzKSwgdmp1c3QgPSAtMC4zLCBzaXplID0gMy41KSArIAogIGxhYnModGl0bGUgPSAiVG90YWwgTnVtYmVyIG9mIFVuaXF1ZSBFeHByZXNzZWQgbWlSTkFzIHBlciBTYW1wbGUiLAogICAgICAgeCA9ICJTYW1wbGUiLAogICAgICAgeSA9ICJVbmlxdWUgbWlSTkEiKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSAgIyBSb3RhdGUgeC1heGlzIGxhYmVscyBmb3IgcmVhZGFiaWxpdHkKYGBgCgojIyBIZWF0bWFwCgpgYGB7ciBtaVJOQS1oZWF0bWFwfQpwaGVhdG1hcChBcHVsX2NvdW50c19taVJOQSwKICAgICAgICAgY2x1c3Rlcl9yb3dzID0gVFJVRSwKICAgICAgICAgY2x1c3Rlcl9jb2xzID0gVFJVRSwKICAgICAgICAgc2hvd19yb3duYW1lcyA9IFRSVUUsCiAgICAgICAgIHNob3dfY29sbmFtZXMgPSBUUlVFLAogICAgICAgICBjb2xvciA9IGNvbG9yUmFtcFBhbGV0dGUoYygiYmx1ZSIsICJ3aGl0ZSIsICJyZWQiKSkoNTApLAogICAgICAgICBmb250c2l6ZV9yb3cgPSA4LAogICAgICAgICBmb250c2l6ZV9jb2wgPSA4KQpgYGAKV2VsbC4uLiB0aGVyZSdzIGxpa2UgMiBtaVJOQXMgd2l0aCBtdWNoIGhpZ2hlciBleHByZXNzaW9uIHRoYW4gdGhlIG90aGVycywgd2hpY2ggaXMgbWFraW5nIHZpc3VhbGl6aW5nIHJlbGF0aXZlIGRpZmZlcmVuY2VzIGRpZmZpY3VsdC4gTGV0J3MgcmVkbyB0aGUgaGVhdG1hcCwgbm9ybWFsaXppbmcgZWFjaCByb3cgdG8gdmlldyByZWxhdGl2ZSBkaWZmZXJlbmNlIGluIGV4cHJlc3Npb24gYmV0d2VlbiBzYW1wbGVzIChgc2NhbGU9J3JvdydgKQoKYGBge3IgbWlSTkEtaGVhdG1hcC1yb3dzY2FsZX0KcGhlYXRtYXAoQXB1bF9jb3VudHNfbWlSTkEsCiAgICAgICAgIGNsdXN0ZXJfcm93cyA9IFRSVUUsCiAgICAgICAgIGNsdXN0ZXJfY29scyA9IFRSVUUsCiAgICAgICAgIHNob3dfcm93bmFtZXMgPSBUUlVFLAogICAgICAgICBzaG93X2NvbG5hbWVzID0gVFJVRSwKICAgICAgICAgc2NhbGUgPSAncm93JywKICAgICAgICAgY29sb3IgPSBjb2xvclJhbXBQYWxldHRlKGMoImJsdWUiLCAid2hpdGUiLCAicmVkIikpKDUwKSwKICAgICAgICAgZm9udHNpemVfcm93ID0gOCwKICAgICAgICAgZm9udHNpemVfY29sID0gOCkKYGBgCgojIHNpUk5BCgpTaG9ydFN0YWNrJ3MgcHJpbWFyeSBwdXJwb3NlIGlzIHRvIGlkZW50aWZ5IG1pUk5BcyBmcm9tIHNSTkEtc2VxIGRhdGEsIGJ1dCBpdCBhbHNvIGF1dG9tYXRpY2FsbHkgYW5ub3RhdGVzIHNpUk5BIGxvY2khIFNpbmNlIHNpUk5BIHBvdGVudGlhbGx5IHBsYXkgYW4gaW1wb3J0YW50IHJvbGUgaW4gdHJhbnNwb3NvbiBzaWxlbmNpbmcgaW4gaW52ZXJ0ZWJyYXRlcywgd2Ugc2hvdWxkIGdlbmVyYXRlIGNvdW50IG1hdHJpY2VzIGZvciBzaVJOQXMgYXMgd2VsbC4gCgpXZSBjYW4gc2VlIGNsdXN0ZXJzIGFubm90YXRlZCBhcyBzaVJOQXMgaW4gdGhlIGBSZXN1bHRzLmdmZjNgIG91dHB1dCBmaWxlIG9mIFNob3J0U3RhY2sgKHNSTkEgSUQgc2hvd24gaW4gdGhlIDNyZCBjb2x1bW4pCgpgYGB7ciBzaVJOQS1jb3VudC1kYXRhLW11bmdpbmd9CkFwdWxfUmVzdWx0c2dmZiA8LSByZWFkLnRhYmxlKCIuLi9vdXRwdXQvMTEtQXB1bC1zUk5BLVNob3J0U3RhY2tfNC4xLjAtcHVsY2hyYV9nZW5vbWUvU2hvcnRTdGFja19vdXQvUmVzdWx0cy5nZmYzIikKIyBTZXBhcmF0ZSBsYXN0IGNvbHVtbiBpbmZvIGludG8gbXVsdGlwbGUgY29sdW1ucyBmb3IgZmlsdGVyaW5nCkFwdWxfUmVzdWx0c2dmZiA8LSBBcHVsX1Jlc3VsdHNnZmYgJT4lCiAgc2VwYXJhdGUoVjksIGludG8gPSBjKCJOYW1lIiwgIkRpY2VyQ2FsbCIsICJNSVJOQSIpLCBzZXAgPSAiOyIpICU+JQogIG11dGF0ZShOYW1lID0gc3ViKCJJRD0iLCAiIiwgTmFtZSksCiAgICAgICAgIERpY2VyQ2FsbCA9IHN1YigiRGljZXJDYWxsPSIsICIiLCBEaWNlckNhbGwpLAogICAgICAgICBNSVJOQSA9IHN1YigiTUlSTkE9IiwgIiIsIE1JUk5BKSkKaGVhZChBcHVsX1Jlc3VsdHNnZmYpCgojIGtlZXAganVzdCB0aGUgc1JOQSBjYXRlZ29yeSBjb2x1bW4gKFYzKSwgYW5kIHRoZSBjbHVzdGVyIG5hbWVzIChOYW1lKQojIGZpbHRlciB0byBvbmx5IGtlZXAgY2x1c3RlcnMgSUQnZCBhcyBzaVJOQXMKQXB1bF9zaVJOQV9jbHVzdGVycyA8LSBBcHVsX1Jlc3VsdHNnZmYgJT4lCiAgc2VsZWN0KFYzLCBOYW1lKSAlPiUKICBmaWx0ZXIoc3RyX2RldGVjdChWMywgcmVnZXgoInNpUk5BIikpKQpoZWFkKEFwdWxfc2lSTkFfY2x1c3RlcnMpCgojIE5vdyB1c2UgdGhpcyBsaXN0IG9mIGNsdXN0ZXJzIElEJ2QgYXMgc2lSTkFzIHRvIGZpbHRlciBvdXIgc1JOQSBjb3VudCBtYXRyaXgKIyBrZWVwIG9ubHkgdGhlIHNhbXBsZSBjb3VudHMgYW5kIGNsdXN0ZXIgbmFtZXMKQXB1bF9jb3VudHNfc1JOQSA8LSByb3duYW1lc190b19jb2x1bW4oQXB1bF9jb3VudHNfc1JOQSwgdmFyID0gIk5hbWUiKQpBcHVsX2NvdW50c19zaVJOQSA8LSBsZWZ0X2pvaW4oQXB1bF9zaVJOQV9jbHVzdGVycywgQXB1bF9jb3VudHNfc1JOQSwgYnkgPSBjKCJOYW1lIiA9ICJOYW1lIikpICU+JQogIHNlbGVjdCgtVjMpCgojIGNvbnZlcnQgdGhlIGNvbHVtbiBvZiBjbHVzdGVyIG5hbWVzIGludG8gdGhlIGRmIHJvdyBuYW1lcwpBcHVsX2NvdW50c19zUk5BIDwtIEFwdWxfY291bnRzX3NSTkEgJT4lIGNvbHVtbl90b19yb3duYW1lcyh2YXI9Ik5hbWUiKQpBcHVsX2NvdW50c19zaVJOQSA8LSBBcHVsX2NvdW50c19zaVJOQSAlPiUgY29sdW1uX3RvX3Jvd25hbWVzKHZhcj0iTmFtZSIpCgpoZWFkKEFwdWxfY291bnRzX3NpUk5BKQoKd3JpdGUudGFibGUoQXB1bF9jb3VudHNfc2lSTkEsIGZpbGUgPSAiLi4vb3V0cHV0LzAzLjEtQXB1bC1zUk5BLXN1bW1hcnkvQXB1bF9zaVJOQV9TaG9ydFN0YWNrX2NvdW50c19mb3JtYXR0ZWQudHh0Iiwgc2VwID0gIlx0Iiwgcm93Lm5hbWVzID0gVFJVRSwgY29sLm5hbWVzID0gVFJVRSwgcXVvdGUgPSBGQUxTRSkKYGBgCgojIyBFeHByZXNzaW9uIGxldmVscwoKUGxvdCBoaXN0b2dyYW1zIG9mIHRoZSBleHByZXNzaW9uIGxldmVscyBpbiBlYWNoIHNhbXBsZQoKYGBge3Igc2lSTkEtZXhwcmVzc2lvbi1sZXZlbC1oaXN0b2dyYW1zfQojIE1lbHQgdGhlIGNvdW50IG1hdHJpeCBpbnRvIGxvbmcgZm9ybWF0CkFwdWxfY291bnRzX3NpUk5BX21lbHRlZCA8LSBtZWx0KEFwdWxfY291bnRzX3NpUk5BLCB2YXJpYWJsZS5uYW1lID0gInNhbXBsZSIsIHZhbHVlLm5hbWUgPSAiY291bnRzIikKCiMgUGxvdCB0aGUgZXhwcmVzc2lvbiBsZXZlbCBoaXN0b2dyYW1zIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QoQXB1bF9jb3VudHNfc2lSTkFfbWVsdGVkLCBhZXMoeCA9IGNvdW50cykpICsKICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aCA9IDEsIGZpbGwgPSAiIzQwOEVDNiIsIGNvbG9yID0gImJsYWNrIikgKwogIHNjYWxlX3hfbG9nMTAoKSArICAjIE9wdGlvbmFsOiBMb2ctdHJhbnNmb3JtIHRoZSB4LWF4aXMgZm9yIGJldHRlciB2aXN1YWxpemF0aW9uCiAgZmFjZXRfd3JhcCh+c2FtcGxlLCBzY2FsZXMgPSAiZnJlZV95IikgKwogIGxhYnModGl0bGUgPSAic2lSTkEgRXhwcmVzc2lvbiBMZXZlbCBIaXN0b2dyYW0gZm9yIEVhY2ggU2FtcGxlIiwKICAgICAgIHggPSAiRXhwcmVzc2lvbiBMZXZlbCAoQ291bnRzKSIsCiAgICAgICB5ID0gIkZyZXF1ZW5jeSIpICsKICB0aGVtZV9taW5pbWFsKCkKYGBgCgojIyBzaVJOQSBjb3VudHMKCkZpcnN0IGxldCdzIGNoZWNrIHRoZSB0b3RhbCBudW1iZXIgb2Ygc2lSTkFzIGluIGVhY2ggc2FtcGxlIC0tIGtlZXAgaW4gbWluZCB0aGlzIGV4cHJlc3Npb24gZGF0YSBoYXMgKm5vdCogYmVlbiBub3JtYWxpemVkIHlldCwgc28gdGhlcmUgbWF5IGJlIGRpZmZlcmVudCB0b3RhbHMgZm9yIGVhY2ggc2FtcGxlCmBgYHtyIHNpUk5BLWNvdW50cy1wbG90fQojIENhbGN1bGF0ZSB0aGUgdG90YWwgbnVtYmVyIG9mIHRyYW5zY3JpcHRzIGZvciBlYWNoIHNhbXBsZQp0b3RhbF9zaVJOQSA8LSBjb2xTdW1zKEFwdWxfY291bnRzX3NpUk5BKQoKIyBDcmVhdGUgYSBkYXRhIGZyYW1lIGZvciBwbG90dGluZwp0b3RhbF9zaVJOQV9kZiA8LSBkYXRhLmZyYW1lKHNhbXBsZSA9IG5hbWVzKHRvdGFsX3NpUk5BKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b3RhbHMgPSB0b3RhbF9zaVJOQSkKCiMgUGxvdCB0aGUgdG90YWwgbnVtYmVyIG9mIHRyYW5zY3JpcHRzIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QodG90YWxfc2lSTkFfZGYsIGFlcyh4ID0gc2FtcGxlLCB5ID0gdG90YWxzKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBnZW9tX3RleHQoYWVzKGxhYmVsID0gdG90YWxzKSwgdmp1c3QgPSAtMC4zLCBzaXplID0gMy41KSArIAogIGxhYnModGl0bGUgPSAiVG90YWwgTnVtYmVyIG9mIHNpUk5BcyBwZXIgU2FtcGxlIiwKICAgICAgIHggPSAiU2FtcGxlIiwKICAgICAgIHkgPSAiVG90YWwgc2lSTkFzIikgKwogIHRoZW1lX21pbmltYWwoKSArCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkgICMgUm90YXRlIHgtYXhpcyBsYWJlbHMgZm9yIHJlYWRhYmlsaXR5CmBgYAoKTm93IGxldCdzIGNoZWNrIHRoZSBudW1iZXIgb2YgdW5pcXVlIHNpUk5BcyBpbiBlYWNoIHNhbXBsZSAtLSBUaGlzIHNob3VsZCBiZSBwcmV0dHkgbXVjaCB0aGUgc2FtZSBhY3Jvc3Mgc2FtcGxlcywgZXZlbiB3aXRob3V0IG5vcm1hbGl6YXRpb24uCgpgYGB7ciB0b3RhbC11bmlxdWUtc2lSTkEtcGxvdH0KIyBDYWxjdWxhdGUgdGhlIG51bWJlciBvZiB1bmlxdWUgdHJhbnNjcmlwdHMgKG5vbi16ZXJvIGNvdW50cykgZm9yIGVhY2ggc2FtcGxlCnVuaXF1ZV9zaVJOQSA8LSBjb2xTdW1zKEFwdWxfY291bnRzX3NpUk5BID4gMCkKCiMgQ3JlYXRlIGEgZGF0YSBmcmFtZSBmb3IgcGxvdHRpbmcKdW5pcXVlX3NpUk5BX2RmIDwtIGRhdGEuZnJhbWUoc2FtcGxlID0gbmFtZXModW5pcXVlX3NpUk5BKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdW5pcXVlcyA9IHVuaXF1ZV9zaVJOQSkKCiMgUGxvdCB0aGUgdG90YWwgbnVtYmVyIG9mIHVuaXF1ZSB0cmFuc2NyaXB0cyBmb3IgZWFjaCBzYW1wbGUKZ2dwbG90KHVuaXF1ZV9zaVJOQV9kZiwgYWVzKHggPSBzYW1wbGUsIHkgPSB1bmlxdWVzKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBnZW9tX3RleHQoYWVzKGxhYmVsID0gdW5pcXVlcyksIHZqdXN0ID0gLTAuMywgc2l6ZSA9IDMuNSkgKyAKICBsYWJzKHRpdGxlID0gIlRvdGFsIE51bWJlciBvZiBVbmlxdWUgRXhwcmVzc2VkIHNpUk5BcyBwZXIgU2FtcGxlIiwKICAgICAgIHggPSAiU2FtcGxlIiwKICAgICAgIHkgPSAiVW5pcXVlIHNpUk5BIikgKwogIHRoZW1lX21pbmltYWwoKSArCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkgICMgUm90YXRlIHgtYXhpcyBsYWJlbHMgZm9yIHJlYWRhYmlsaXR5CmBgYAoKIyMgSGVhdG1hcAoKYGBge3Igc2lSTkEtaGVhdG1hcH0KcGhlYXRtYXAoQXB1bF9jb3VudHNfc2lSTkEsCiAgICAgICAgIGNsdXN0ZXJfcm93cyA9IFRSVUUsCiAgICAgICAgIGNsdXN0ZXJfY29scyA9IFRSVUUsCiAgICAgICAgIHNob3dfcm93bmFtZXMgPSBUUlVFLAogICAgICAgICBzaG93X2NvbG5hbWVzID0gVFJVRSwKICAgICAgICAgY29sb3IgPSBjb2xvclJhbXBQYWxldHRlKGMoImJsdWUiLCAid2hpdGUiLCAicmVkIikpKDUwKSwKICAgICAgICAgZm9udHNpemVfcm93ID0gOCwKICAgICAgICAgZm9udHNpemVfY29sID0gOCkKYGBgCgpgYGB7ciBzaVJOQS1oZWF0bWFwLXJvd3NjYWxlfQpwaGVhdG1hcChBcHVsX2NvdW50c19zaVJOQSwKICAgICAgICAgY2x1c3Rlcl9yb3dzID0gVFJVRSwKICAgICAgICAgY2x1c3Rlcl9jb2xzID0gVFJVRSwKICAgICAgICAgc2hvd19yb3duYW1lcyA9IFRSVUUsCiAgICAgICAgIHNob3dfY29sbmFtZXMgPSBUUlVFLAogICAgICAgICBzY2FsZSA9ICdyb3cnLAogICAgICAgICBjb2xvciA9IGNvbG9yUmFtcFBhbGV0dGUoYygiYmx1ZSIsICJ3aGl0ZSIsICJyZWQiKSkoNTApLAogICAgICAgICBmb250c2l6ZV9yb3cgPSA4LAogICAgICAgICBmb250c2l6ZV9jb2wgPSA4KQpgYGAKCgojIC4uLi4uLi4uLi4uCgojIE5vcm1hbGl6ZWQgc1JOQSBjb3VudHMKCiMjIE5vcm1hbGl6ZSBjb3VudHMgd2l0aCBERVNlcTIKCiMjIyBQbG90IHVubm9ybWFsaXplZCBzUk5BIGRhdGEKCmBgYHtyIHBsb3QtdW5ub3JtYWxpemVkLXNSTkF9CgpBcHVsX2NvdW50c19zUk5BICU+JSAKIyAgc2VsZWN0KC1OYW1lKSAlPiUKICBwaXZvdF9sb25nZXIoIGNvbHMgPSBldmVyeXRoaW5nKCksIG5hbWVzX3RvID0gInNhbXBsZSIsIHZhbHVlc190byA9ICJjb3VudCIpICU+JQogIGdncGxvdCguLCBhZXMoeCA9IHNhbXBsZSwgeSA9IGNvdW50KSkgKwogIGdlb21fdmlvbGluKCkgKyAKICBnZW9tX3BvaW50KGFscGhhID0gMC4yKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICBsYWJzKHRpdGxlID0gIlVubm9ybWFsaXplZCBzUk5BIGNvdW50cyIsCiAgICAgICB4ID0gIlNhbXBsZSIsCiAgICAgICB5ID0gImNvdW50IikKYGBgCgojIyMgTWV0YWRhdGEKCkRFU2VxMiByZXF1aXJlcyBhIG1ldGFkYXRhIGRhdGEgZnJhbWUgYXMgaW5wdXQuIEkgZG9uJ3QgaGF2ZSBzYW1wbGUgbWV0YWRhdGEgdGhvdWdoIHNvLCBzaW5jZSB3ZSdyZSBqdXN0IGRvaW5nIERFU2VxMiBmb3Igbm9ybWFsaXphdGlvbiBwdXJwb3NlcyAobm90IGFuYWx5c2lzIHB1cnBvc2VzKSwgSSdtIGp1c3QgZ29pbmcgdG8gY3JlYXRlIGEgZHVtbXkgc2hlZXQKCmBgYHtyIG1ha2Utc1JOQS1tZXRhZGF0YS1kYXRhZnJhbWV9CkFwdWxfc2FtcGxlX25hbWVzIDwtIEFwdWxfY291bnRzX3NSTkEgJT4lCiMgIHNlbGVjdCgtTmFtZSkgJT4lCiAgY29sbmFtZXMoKQoKQXB1bF9tZXRhZGF0YV9zUk5BIDwtIGRhdGEuZnJhbWUoU2FtcGxlID0gQXB1bF9zYW1wbGVfbmFtZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBTcGVjaWVzID0gcmVwKCJBLnB1bGNocmEiLCA1KSkKcm93bmFtZXMoQXB1bF9tZXRhZGF0YV9zUk5BKSA8LSBBcHVsX3NhbXBsZV9uYW1lcwoKaGVhZChBcHVsX21ldGFkYXRhX3NSTkEpCmBgYAoKIyMjIERFU2VxIG9iamVjdAoKYGBge3IgbWFrZS1zUk5BLWRlc2VxLW9iamVjdCwgY2FjaGU9VFJVRX0KIyBDYWxjdWxhdGUgREVTZXEgb2JqZWN0CkFwdWxfY291bnRzX3NSTkFfcm93TmFtZXMgPC0gQXB1bF9jb3VudHNfc1JOQSAKIyU+JSBjb2x1bW5fdG9fcm93bmFtZXModmFyID0gIk5hbWUiKQoKZGRzX0FwdWxfc1JOQSA8LSBERVNlcURhdGFTZXRGcm9tTWF0cml4KGNvdW50RGF0YSA9IEFwdWxfY291bnRzX3NSTkFfcm93TmFtZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbERhdGEgPSBBcHVsX21ldGFkYXRhX3NSTkEsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRlc2lnbiA9IH4gMSkgCgojIFJ1biBkaWZmZXJlbnRpYWwgZXhwcmVzc2lvbiBhbmFseXNpcyAKIyAoTm90ZSB0aGF0IHRoaXMgREVTZXEoKSBmdW5jdGlvbiBydW5zIGFsbCBuZWNlc3Nhcnkgc3RlcHMsIGluY2x1ZGluZyBkYXRhIG5vcm1hbGl6YXRpb24sIAojIGVzdGltYXRpbmcgc2l6ZSBmYWN0b3JzLCBlc3RpbWF0aW5nIGRpc3BlcnNpb25zLCBnZW5lLXdpc2UgZGlzcGVyc2lvbiBlc3RpbWF0ZXMsIG1lYW4tZGlzcGVyc2lvbiAKIyByZWxhdGlvbnNoaXAsIGZpbmFsIGRpc3BlcnNpb24gZXN0aW1hdGVzLCBmaXR0aW5nIG1vZGVsLCBhbmQgdGVzdGluZykKIyBVc2luZyBkZXNpZ24gPSB+MSBiZWNhdXNlIHdlIGRvbid0IGhhdmUgdHJlYXRtZW50IGdyb3VwcwoKZGRzX0FwdWxfc1JOQSA8LSBERVNlcShkZHNfQXB1bF9zUk5BKQpgYGAKCkl0J3Mgd29ydGggbm90aW5nIGhlcmUgdGhhdCBJJ20gYWN0dWFsbHkgZ29pbmcgdG8gYmUgZG9pbmcgdHdvIGRpZmZlcmVudCB0eXBlcyBvZiB0cmFuc2Zvcm1hdGlvbiBvbiB0aGUgY291bnRzIGRhdGEsIHdoaWNoIHNlcnZlIGRpZmZlcmVudCBwdXJwb3Nlcy4gCgotIEZpcnN0IGlzICoqbm9ybWFsaXppbmcqKiB0aGUgdHJhbnNjcmlwdCBjb3VudHMsIHdoaWNoIGFkanVzdHMgZm9yIGRpZmZlcmVuY2VzIGluIGxpYnJhcnkgc2l6ZSBvciBzZXF1ZW5jaW5nIGRlcHRoLCBidXQgcmV0YWlucyBjb3VudC1saWtlIHByb3BlcnRpZXMuIE5vcm1hbGl6ZWQgY291bnRzIGFyZSBtb3N0IHVzZWZ1bCBmb3IgdGhpbmdzIGxpa2UgdmlzdWFsaXppbmcgZXhwcmVzc2lvbiBsZXZlbHMgYW5kIGRpZmZlcmVudGlhbCBleHByZXNzaW9uIGFuYWx5c2lzLgoKLSBTZWNvbmQgaXMgKip2YXJpYW5jZSBzdGFiaWxpemluZyoqIHRoZSBjb3VudHMgZGF0YSwgd2hpY2ggYWltcyB0byBtYWtlIHRoZSB2YXJpYW5jZSBvZiB0aGUgdHJhbnNmb3JtZWQgZGF0YSBhcHByb3hpbWF0ZWx5IGluZGVwZW5kZW50IG9mIHRoZSBtZWFuLCByZWR1Y2luZyBoZXRlcm9zY2VkYXN0aWNpdHkgKHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiB2YXJpYW5jZSBhbmQgbWVhbikgYW5kICJzbW9vdGhpbmciIG91dCB0aGUgdmFyaWFuY2UgYXQgbG93IGNvdW50cy4gTm90YWJseSwgdGhlIHRyYW5zZm9ybWVkIGRhdGEgaXMgKm5vIGxvbmdlciBvbiB0aGUgb3JpZ2luYWwgY291bnQgc2NhbGUqLiBUaGUgdHJhbnNmb3JtYXRpb24gbWFrZXMgdGhlIHZhcmlhbmNlIHJvdWdobHkgY29uc3RhbnQgYWNyb3NzIHRoZSByYW5nZSBvZiBjb3VudHMsIHdoaWNoIG1ha2VzIGl0IGVhc2llciB0byBpbnRlcnByZXQgcGF0dGVybnMgaW4gdGhlIGRhdGEgdmlzdWFsbHkuIFZhcmlhbmNlIHN0YWJpbGl6ZWQgZGF0YSBpcyBtb3N0IHVzZWZ1bCBmb3IgZXhwbG9yYXRvcnkgZGF0YSBhbmFseXNpcywgbGlrZSBQQ0EsIGNsdXN0ZXJpbmcsIGFuZCBoZWF0bWFwcywgYW5kIGlzIGFsc28gdGhlIHRyYW5zZm9ybWF0aW9uIHdlJ2xsIHdhbnQgdG8gdXNlIGJlZm9yZSBXR0NOQS4KCmBgYHtyIGdldC1ub3JtYWxpemVkLXNSTkEtY291bnRzLCBjYWNoZT1UUlVFfQojIGV4dHJhY3Qgbm9ybWFsaXplZCBjb3VudHMKIyAobm9ybWFsaXphdGlvbiBpcyBhdXRvbWF0aWNhbGx5IHBlcmZvcm1lZCBieSBkZXNlcTIpCkFwdWxfY291bnRzX3NSTkFfbm9ybSA8LSBjb3VudHMoZGRzX0FwdWxfc1JOQSwgbm9ybWFsaXplZD1UUlVFKSAlPiUgZGF0YS5mcmFtZSgpCgp3cml0ZS50YWJsZShBcHVsX2NvdW50c19zUk5BX25vcm0sIGZpbGUgPSAiLi4vb3V0cHV0LzAzLjEtQXB1bC1zUk5BLXN1bW1hcnkvQXB1bF9jb3VudHNfc1JOQV9ub3JtYWxpemVkLnR4dCIsIHNlcCA9ICJcdCIsIHJvdy5uYW1lcyA9IFRSVUUsIGNvbC5uYW1lcyA9IFRSVUUsIHF1b3RlID0gRkFMU0UpCgoKIyB2YXJpYW5jZSBzdGFiaWxpemVkIGRhdGEKdnNkX0FwdWxfc1JOQSA8LSB2YXJpYW5jZVN0YWJpbGl6aW5nVHJhbnNmb3JtYXRpb24oZGRzX0FwdWxfc1JOQSwgYmxpbmQ9VFJVRSkKd3BuX3ZzZF9BcHVsX3NSTkEgPC0gZ2V0VmFyaWFuY2VTdGFiaWxpemVkRGF0YShkZHNfQXB1bF9zUk5BKQpydl93cG5fQXB1bF9zUk5BIDwtIHJvd1ZhcnMod3BuX3ZzZF9BcHVsX3NSTkEsIHVzZU5hbWVzPVRSVUUpCgpBcHVsX2NvdW50c19zUk5BX3ZzZCA8LSBkYXRhLmZyYW1lKHdwbl92c2RfQXB1bF9zUk5BKQp3cml0ZS50YWJsZShBcHVsX2NvdW50c19zUk5BX3ZzZCwgZmlsZSA9ICIuLi9vdXRwdXQvMDMuMS1BcHVsLXNSTkEtc3VtbWFyeS9BcHVsX2NvdW50c19zUk5BX3ZhcmlhbmNlc3RhYmlsaXplZC50eHQiLCBzZXAgPSAiXHQiLCByb3cubmFtZXMgPSBUUlVFLCBjb2wubmFtZXMgPSBUUlVFLHF1b3RlID0gRkFMU0UpCgpxNzVfd3BuX0FwdWxfc1JOQSA8LSBxdWFudGlsZShyb3dWYXJzKHdwbl92c2RfQXB1bF9zUk5BLCB1c2VOYW1lcz1UUlVFKSwgLjc1KSAgIyA3NXRoIHF1YW50aWxlIHZhcmlhYmlsaXR5CkFwdWxfY291bnRzX3NSTkFfdnNkX3E3NSA8LSB3cG5fdnNkX0FwdWxfc1JOQVsgcnZfd3BuX0FwdWxfc1JOQSA+IHE3NV93cG5fQXB1bF9zUk5BLCBdICU+JSBkYXRhLmZyYW1lICMgZmlsdGVyIHRvIHJldGFpbiBvbmx5IHRoZSBtb3N0IHZhcmlhYmxlIGdlbmVzCndyaXRlLnRhYmxlKEFwdWxfY291bnRzX3NSTkFfdnNkX3E3NSwgZmlsZSA9ICIuLi9vdXRwdXQvMDMuMS1BcHVsLXNSTkEtc3VtbWFyeS9BcHVsX2NvdW50c19zUk5BX3ZhcmlhbmNlc3RhYmlsaXplZF9xNzUudHh0Iiwgc2VwID0gIlx0Iiwgcm93Lm5hbWVzID0gVFJVRSwgY29sLm5hbWVzID0gVFJVRSxxdW90ZSA9IEZBTFNFKQoKcTk1X3dwbl9BcHVsX3NSTkEgPC0gcXVhbnRpbGUocm93VmFycyh3cG5fdnNkX0FwdWxfc1JOQSwgdXNlTmFtZXM9VFJVRSksIC45NSkgICMgOTV0aCBxdWFudGlsZSB2YXJpYWJpbGl0eQpBcHVsX2NvdW50c19zUk5BX3ZzZF9xOTUgPC0gd3BuX3ZzZF9BcHVsX3NSTkFbIHJ2X3dwbl9BcHVsX3NSTkEgPiBxOTVfd3BuX0FwdWxfc1JOQSwgXSAlPiUgZGF0YS5mcmFtZSAjIGZpbHRlciB0byByZXRhaW4gb25seSB0aGUgbW9zdCB2YXJpYWJsZSBnZW5lcwp3cml0ZS50YWJsZShBcHVsX2NvdW50c19zUk5BX3ZzZF9xOTUsIGZpbGUgPSAiLi4vb3V0cHV0LzAzLjEtQXB1bC1zUk5BLXN1bW1hcnkvQXB1bF9jb3VudHNfc1JOQV92YXJpYW5jZXN0YWJpbGl6ZWRfcTk1LnR4dCIsIHNlcCA9ICJcdCIsIHJvdy5uYW1lcyA9IFRSVUUsIGNvbC5uYW1lcyA9IFRSVUUscXVvdGUgPSBGQUxTRSkKYGBgCgojIyBQbG90IG5vcm1hbGl6ZWQgZGF0YQoKYGBge3IgcGxvdC1ub3JtYWxpemVkLXNSTkF9CkFwdWxfY291bnRzX3NSTkFfbm9ybV9sb25nIDwtIEFwdWxfY291bnRzX3NSTkFfbm9ybSAlPiUKICBtdXRhdGUoCiAgICBHZW5lX2lkID0gcm93Lm5hbWVzKEFwdWxfY291bnRzX3NSTkFfbm9ybSkKICApICU+JQogIHBpdm90X2xvbmdlcigtR2VuZV9pZCkKCkFwdWxfY291bnRzX3NSTkFfbm9ybV9sb25nICU+JQogIGdncGxvdCguLCBhZXMoeCA9IG5hbWUsIHkgPSB2YWx1ZSkpICsKICBnZW9tX3Zpb2xpbigpICsKICBnZW9tX3BvaW50KCkgKwogIHRoZW1lX2J3KCkgKwogIHRoZW1lKAogICAgYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoIGFuZ2xlID0gOTApCiAgKSArCiAgeWxpbSgwLCBOQSkgKwogIGxhYnMoCiAgICB0aXRsZSA9ICJOb3JtYWxpemVkIEV4cHJlc3Npb24iLAogICAgeCA9ICJTYW1wbGUiLAogICAgeSA9ICJOb3JtYWxpemVkIGNvdW50cyIKICApCmBgYAoKCiMjIFBsb3QgdmFyaWFuY2Ugc3RhYmlsaXplZCBkYXRhCgpgYGB7ciBwbG90LXZzZC1zUk5BfQpBcHVsX2NvdW50c19zUk5BX3ZzZF9sb25nIDwtIEFwdWxfY291bnRzX3NSTkFfdnNkICU+JQogIG11dGF0ZSgKICAgIEdlbmVfaWQgPSByb3cubmFtZXMoQXB1bF9jb3VudHNfc1JOQV92c2QpCiAgKSAlPiUKICBwaXZvdF9sb25nZXIoLUdlbmVfaWQpCgpBcHVsX2NvdW50c19zUk5BX3ZzZF9sb25nICU+JQogIGdncGxvdCguLCBhZXMoeCA9IG5hbWUsIHkgPSB2YWx1ZSkpICsKICBnZW9tX3Zpb2xpbigpICsKICBnZW9tX3BvaW50KCkgKwogIHRoZW1lX2J3KCkgKwogIHRoZW1lKAogICAgYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoIGFuZ2xlID0gOTApCiAgKSArCiAgeWxpbSgwLCBOQSkgKwogIGxhYnMoCiAgICB0aXRsZSA9ICJWYXJpYW5jZSBTdGFiaWxpemVkIEV4cHJlc3Npb24iLAogICAgeCA9ICJTYW1wbGUiLAogICAgeSA9ICJWYXJpYW5jZSBzdGFiaWxpemVkIGRhdGEiCiAgKQpgYGAKCiMjIE5vcm1hbGl6ZWQgZXhwcmVzc2lvbiBsZXZlbHMKClBsb3QgaGlzdG9ncmFtcyBvZiB0aGUgbm9ybWFsaXplZCBleHByZXNzaW9uIGxldmVscyBpbiBlYWNoIHNhbXBsZQoKYGBge3Igbm9ybS1leHByZXNzaW9uLWxldmVsLWhpc3RvZ3JhbXN9CiMgTWVsdCB0aGUgY291bnQgbWF0cml4IGludG8gbG9uZyBmb3JtYXQKQXB1bF9jb3VudHNfbm9ybV9tZWx0ZWQgPC0gbWVsdChBcHVsX2NvdW50c19zUk5BX25vcm0sIHZhcmlhYmxlLm5hbWUgPSAic2FtcGxlIiwgdmFsdWUubmFtZSA9ICJjb3VudHMiKQoKIyBQbG90IHRoZSBleHByZXNzaW9uIGxldmVsIGhpc3RvZ3JhbXMgZm9yIGVhY2ggc2FtcGxlCmdncGxvdChBcHVsX2NvdW50c19ub3JtX21lbHRlZCwgYWVzKHggPSBjb3VudHMpKSArCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGggPSAxLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBzY2FsZV94X2xvZzEwKCkgKyAgIyBPcHRpb25hbDogTG9nLXRyYW5zZm9ybSB0aGUgeC1heGlzIGZvciBiZXR0ZXIgdmlzdWFsaXphdGlvbgogIGZhY2V0X3dyYXAofnNhbXBsZSwgc2NhbGVzID0gImZyZWVfeSIpICsKICBsYWJzKHRpdGxlID0gIkdlbmUgRXhwcmVzc2lvbiBMZXZlbCBIaXN0b2dyYW0gZm9yIEVhY2ggU2FtcGxlIiwKICAgICAgIHggPSAiRXhwcmVzc2lvbiBMZXZlbCAoQ291bnRzKSIsCiAgICAgICB5ID0gIkZyZXF1ZW5jeSIpICsKICB0aGVtZV9taW5pbWFsKCkKYGBgCgojIyBOb3JtYWxpemVkIHRyYW5zY3JpcHQgY291bnRzCgpDaGVjayB0aGUgdG90YWwgbnVtYmVyIG9mIHRyYW5zY3JpcHRzIGluIGVhY2ggc2FtcGxlIC0tIG5vdyB0aGF0IHdlJ3ZlIG5vcm1hbGl6ZWQgdGhlIGRhdGEgdGhlc2UgdG90YWxzIHNob3VsZCBiZSBzaW1pbGFyCmBgYHtyIG5vcm0tdHJhbnNjcmlwdC1jb3VudHMtcGxvdH0KIyBDYWxjdWxhdGUgdGhlIHRvdGFsIG51bWJlciBvZiB0cmFuc2NyaXB0cyBmb3IgZWFjaCBzYW1wbGUKdG90YWxfdHJhbnNjcmlwdHNfbm9ybSA8LSBjb2xTdW1zKEFwdWxfY291bnRzX3NSTkFfbm9ybSkKCiMgQ3JlYXRlIGEgZGF0YSBmcmFtZSBmb3IgcGxvdHRpbmcKdG90YWxfdHJhbnNjcmlwdHNfbm9ybV9kZiA8LSBkYXRhLmZyYW1lKHNhbXBsZSA9IG5hbWVzKHRvdGFsX3RyYW5zY3JpcHRzX25vcm0pLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRvdGFscyA9IHRvdGFsX3RyYW5zY3JpcHRzX25vcm0pCgojIFBsb3QgdGhlIHRvdGFsIG51bWJlciBvZiB0cmFuc2NyaXB0cyBmb3IgZWFjaCBzYW1wbGUKZ2dwbG90KHRvdGFsX3RyYW5zY3JpcHRzX25vcm1fZGYsIGFlcyh4ID0gc2FtcGxlLCB5ID0gdG90YWxzKSkgKwogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBnZW9tX3RleHQoYWVzKGxhYmVsID0gdG90YWxzKSwgdmp1c3QgPSAtMC4zLCBzaXplID0gMy41KSArCiAgbGFicyh0aXRsZSA9ICJUb3RhbCBOdW1iZXIgb2YgVHJhbnNjcmlwdHMgcGVyIFNhbXBsZSIsCiAgICAgICB4ID0gIlNhbXBsZSIsCiAgICAgICB5ID0gIlRvdGFsIFRyYW5zY3JpcHRzIikgKwogIHRoZW1lX21pbmltYWwoKSArCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkgICMgUm90YXRlIHgtYXhpcyBsYWJlbHMgZm9yIHJlYWRhYmlsaXR5CmBgYAoKIyMgUENBIG9mIHZhcmlhbmNlIHN0YWJpbGl6ZWQgZGF0YQoKYGBge3IgUENBfQpwbG90UENBKHZzZF9BcHVsX3NSTkEsIGludGdyb3VwPSJTYW1wbGUiKQpgYGAKCiMjIFNhbXBsZSBjbHVzdGVyaW5nCgpgYGB7ciBzYW1wbGUtY2x1c3RlcmluZ30Kc2FtcGxlX2Rpc3RzIDwtIGRpc3QodChhc3NheSh2c2RfQXB1bF9zUk5BKSkpCnBoZWF0bWFwKGFzLm1hdHJpeChzYW1wbGVfZGlzdHMpLCBjbHVzdGVyaW5nX2Rpc3RhbmNlX3Jvd3MgPSAiZXVjbGlkZWFuIiwgCiAgICAgICAgIGNsdXN0ZXJpbmdfZGlzdGFuY2VfY29scyA9ICJldWNsaWRlYW4iLCBtYWluPSJTYW1wbGUgQ2x1c3RlcmluZyIpCmBgYAoKIyMgSGVhdG1hcHMKCk9mIG1vc3QgdmFyaWFibGUgdmFyaWFuY2Ugc3RhYmlsaXplZCBzUk5BIHRyYW5zY3JpcHRzCgpgYGB7ciBoZWF0bXBhc30KIyA3NXRoIHF1YW50aWxlCmhlYXRfY29sb3JzIDwtIHJldihicmV3ZXIucGFsKDEyLCAiUmRZbEJ1IikpCnBoZWF0bWFwKEFwdWxfY291bnRzX3NSTkFfdnNkX3E3NSwgCiAgICAgICAgIGNsdXN0ZXJfcm93cyA9IFRSVUUsCiAgICAgICAgIGNsdXN0ZXJfY29scyA9IFRSVUUsCiAgICAgICAgIHNob3dfcm93bmFtZXMgPSBUUlVFLAogICAgICAgICBzaG93X2NvbG5hbWVzID0gVFJVRSwKICAgICAgICAgY29sb3IgPSBoZWF0X2NvbG9ycywKICAgICAgICAgc2NhbGU9InJvdyIpCgojIDk1dGggcXVhbnRpbGUKcGhlYXRtYXAoQXB1bF9jb3VudHNfc1JOQV92c2RfcTk1LCAKICAgICAgICAgY2x1c3Rlcl9yb3dzID0gVFJVRSwKICAgICAgICAgY2x1c3Rlcl9jb2xzID0gVFJVRSwKICAgICAgICAgc2hvd19yb3duYW1lcyA9IFRSVUUsCiAgICAgICAgIHNob3dfY29sbmFtZXMgPSBUUlVFLAogICAgICAgICBjb2xvciA9IGhlYXRfY29sb3JzLAogICAgICAgICBzY2FsZT0icm93IikKYGBgCgojIE5vcm1hbGl6ZWQgbWlSTkEgY291bnRzCgojIyBJc29sYXRlIG5vcm1hbGl6ZWQvdnNkIG1pUk5BCmBgYHtyIG1pUk5BLW5vcm1hbGl6ZWQtbWlSTkF9CiMjIEFsc28gd3JpdHRlbiB0byB1c2UgZ2l2ZW4gbWlSTkEgbmFtZXMgIyMKIyBBcHVsX2NvdW50c19zUk5BX25vcm0kTmFtZSA8LSByb3duYW1lcyhBcHVsX2NvdW50c19zUk5BX25vcm0pCiMgQXB1bF9jb3VudHNfbWlSTkFfbm9ybSA8LSBsZWZ0X2pvaW4oQXB1bF9tZXRhZGF0YV9taVJOQSwgQXB1bF9jb3VudHNfc1JOQV9ub3JtLCBieSA9IGMoIk5hbWUiID0gIk5hbWUiKSkgJT4lCiMgICBjb2x1bW5fdG9fcm93bmFtZXModmFyPSJnaXZlbl9taVJOQV9uYW1lIikgJT4lCiMgICBzZWxlY3Qoc3RhcnRzX3dpdGgoInNhbXBsZSIpKQojIHdyaXRlLnRhYmxlKEFwdWxfY291bnRzX21pUk5BX25vcm0sIGZpbGUgPSAiLi4vb3V0cHV0LzAzLjEtQXB1bC1zUk5BLXN1bW1hcnkvQXB1bF9jb3VudHNfbWlSTkFfbm9ybWFsaXplZC50eHQiLCBzZXAgPSAiXHQiLCByb3cubmFtZXMgPSBUUlVFLCBjb2wubmFtZXMgPSBUUlVFLHF1b3RlID0gRkFMU0UpCiMgCiMgQXB1bF9jb3VudHNfc1JOQV92c2QkTmFtZSA8LSByb3duYW1lcyhBcHVsX2NvdW50c19zUk5BX3ZzZCkKIyBBcHVsX2NvdW50c19taVJOQV92c2QgPC0gbGVmdF9qb2luKEFwdWxfbWV0YWRhdGFfbWlSTkEsIEFwdWxfY291bnRzX3NSTkFfdnNkLCBieSA9IGMoIk5hbWUiID0gIk5hbWUiKSkgJT4lCiMgICBjb2x1bW5fdG9fcm93bmFtZXModmFyPSJnaXZlbl9taVJOQV9uYW1lIikgJT4lCiMgICBzZWxlY3Qoc3RhcnRzX3dpdGgoInNhbXBsZSIpKQojIHdyaXRlLnRhYmxlKEFwdWxfY291bnRzX21pUk5BX3ZzZCwgZmlsZSA9ICIuLi9vdXRwdXQvMDMuMS1BcHVsLXNSTkEtc3VtbWFyeS9BcHVsX2NvdW50c19taVJOQV92YXJpYW5jZXN0YWJpbGl6ZWQudHh0Iiwgc2VwID0gIlx0Iiwgcm93Lm5hbWVzID0gVFJVRSwgY29sLm5hbWVzID0gVFJVRSxxdW90ZSA9IEZBTFNFKQoKIyMgVXNlIGJlbG93IHVudGlsIHlvdSBoYXZlIG1pUk5BIG5hbWVzICMjCkFwdWxfY291bnRzX3NSTkFfbm9ybSROYW1lIDwtIHJvd25hbWVzKEFwdWxfY291bnRzX3NSTkFfbm9ybSkKQXB1bF9jb3VudHNfc1JOQV92c2QkTmFtZSA8LSByb3duYW1lcyhBcHVsX2NvdW50c19zUk5BX3ZzZCkKCkFwdWxfY291bnRzX21pUk5BX25hbWVzZGYgPC0gZGF0YS5mcmFtZShOYW1lID0gcm93bmFtZXMoQXB1bF9jb3VudHNfbWlSTkEpKSAKCkFwdWxfY291bnRzX21pUk5BX25vcm0gPC0gbGVmdF9qb2luKEFwdWxfY291bnRzX21pUk5BX25hbWVzZGYsIEFwdWxfY291bnRzX3NSTkFfbm9ybSwgYnkgPSBjKCJOYW1lIiA9ICJOYW1lIikpICU+JQogIGNvbHVtbl90b19yb3duYW1lcyh2YXIgPSAiTmFtZSIpCndyaXRlLnRhYmxlKEFwdWxfY291bnRzX21pUk5BX25vcm0sIGZpbGUgPSAiLi4vb3V0cHV0LzAzLjEtQXB1bC1zUk5BLXN1bW1hcnkvQXB1bF9jb3VudHNfbWlSTkFfbm9ybWFsaXplZC50eHQiLCBzZXAgPSAiXHQiLCByb3cubmFtZXMgPSBUUlVFLCBjb2wubmFtZXMgPSBUUlVFLHF1b3RlID0gRkFMU0UpCgpBcHVsX2NvdW50c19taVJOQV92c2QgPC0gbGVmdF9qb2luKEFwdWxfY291bnRzX21pUk5BX25hbWVzZGYsIEFwdWxfY291bnRzX3NSTkFfdnNkLCBieSA9IGMoIk5hbWUiID0gIk5hbWUiKSkgJT4lCiAgY29sdW1uX3RvX3Jvd25hbWVzKHZhciA9ICJOYW1lIikKd3JpdGUudGFibGUoQXB1bF9jb3VudHNfbWlSTkFfdnNkLCBmaWxlID0gIi4uL291dHB1dC8wMy4xLUFwdWwtc1JOQS1zdW1tYXJ5L0FwdWxfY291bnRzX21pUk5BX3ZhcmlhbmNlc3RhYmlsaXplZC50eHQiLCBzZXAgPSAiXHQiLCByb3cubmFtZXMgPSBUUlVFLCBjb2wubmFtZXMgPSBUUlVFLHF1b3RlID0gRkFMU0UpCmBgYAoKIyMgTm9ybWFsaXplZCBleHByZXNzaW9uIGxldmVscwoKUGxvdCBoaXN0b2dyYW1zIG9mIHRoZSBub3JtYWxpemVkIGV4cHJlc3Npb24gbGV2ZWxzIGluIGVhY2ggc2FtcGxlCgpgYGB7ciBtaVJOQS1ub3JtLWV4cHJlc3Npb24tbGV2ZWwtaGlzdG9ncmFtc30KIyBNZWx0IHRoZSBjb3VudCBtYXRyaXggaW50byBsb25nIGZvcm1hdApBcHVsX2NvdW50c19taVJOQV9ub3JtX21lbHRlZCA8LSBtZWx0KEFwdWxfY291bnRzX21pUk5BX25vcm0sIHZhcmlhYmxlLm5hbWUgPSAic2FtcGxlIiwgdmFsdWUubmFtZSA9ICJjb3VudHMiKQoKIyBQbG90IHRoZSBleHByZXNzaW9uIGxldmVsIGhpc3RvZ3JhbXMgZm9yIGVhY2ggc2FtcGxlCmdncGxvdChBcHVsX2NvdW50c19taVJOQV9ub3JtX21lbHRlZCwgYWVzKHggPSBjb3VudHMpKSArCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGggPSAxLCBmaWxsID0gIiM0MDhFQzYiLCBjb2xvciA9ICJibGFjayIpICsKICBzY2FsZV94X2xvZzEwKCkgKyAgIyBPcHRpb25hbDogTG9nLXRyYW5zZm9ybSB0aGUgeC1heGlzIGZvciBiZXR0ZXIgdmlzdWFsaXphdGlvbgogIGZhY2V0X3dyYXAofnNhbXBsZSwgc2NhbGVzID0gImZyZWVfeSIpICsKICBsYWJzKHRpdGxlID0gIkdlbmUgRXhwcmVzc2lvbiBMZXZlbCBIaXN0b2dyYW0gZm9yIEVhY2ggU2FtcGxlIiwKICAgICAgIHggPSAiRXhwcmVzc2lvbiBMZXZlbCAoQ291bnRzKSIsCiAgICAgICB5ID0gIkZyZXF1ZW5jeSIpICsKICB0aGVtZV9taW5pbWFsKCkKYGBgCgojIyBOb3JtYWxpemVkIHRyYW5zY3JpcHQgY291bnRzCgpDaGVjayB0aGUgdG90YWwgbnVtYmVyIG9mIHRyYW5zY3JpcHRzIGluIGVhY2ggc2FtcGxlIC0tIG5vdyB0aGF0IHdlJ3ZlIG5vcm1hbGl6ZWQgdGhlIGRhdGEgdGhlc2UgdG90YWxzIHNob3VsZCBiZSBzaW1pbGFyCmBgYHtyIG1pUk5BLW5vcm0tdHJhbnNjcmlwdC1jb3VudHMtcGxvdH0KIyBDYWxjdWxhdGUgdGhlIHRvdGFsIG51bWJlciBvZiB0cmFuc2NyaXB0cyBmb3IgZWFjaCBzYW1wbGUKdG90YWxfdHJhbnNjcmlwdHNfbWlSTkFfbm9ybSA8LSBjb2xTdW1zKEFwdWxfY291bnRzX21pUk5BX25vcm0pCgojIENyZWF0ZSBhIGRhdGEgZnJhbWUgZm9yIHBsb3R0aW5nCnRvdGFsX3RyYW5zY3JpcHRzX21pUk5BX25vcm1fZGYgPC0gZGF0YS5mcmFtZShzYW1wbGUgPSBuYW1lcyh0b3RhbF90cmFuc2NyaXB0c19taVJOQV9ub3JtKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b3RhbHMgPSB0b3RhbF90cmFuc2NyaXB0c19taVJOQV9ub3JtKQoKIyBQbG90IHRoZSB0b3RhbCBudW1iZXIgb2YgdHJhbnNjcmlwdHMgZm9yIGVhY2ggc2FtcGxlCmdncGxvdCh0b3RhbF90cmFuc2NyaXB0c19taVJOQV9ub3JtX2RmLCBhZXMoeCA9IHNhbXBsZSwgeSA9IHRvdGFscykpICsKICBnZW9tX2JhcihzdGF0ID0gImlkZW50aXR5IiwgZmlsbCA9ICIjNDA4RUM2IiwgY29sb3IgPSAiYmxhY2siKSArCiAgZ2VvbV90ZXh0KGFlcyhsYWJlbCA9IHRvdGFscyksIHZqdXN0ID0gLTAuMywgc2l6ZSA9IDMuNSkgKwogIGxhYnModGl0bGUgPSAiVG90YWwgTnVtYmVyIG9mIG1pUk5BIFRyYW5zY3JpcHRzIHBlciBTYW1wbGUiLAogICAgICAgeCA9ICJTYW1wbGUiLAogICAgICAgeSA9ICJUb3RhbCBUcmFuc2NyaXB0cyIpICsKICB0aGVtZV9taW5pbWFsKCkgKwogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUsIGhqdXN0ID0gMSkpICAjIFJvdGF0ZSB4LWF4aXMgbGFiZWxzIGZvciByZWFkYWJpbGl0eQpgYGAKCiMjIEhlYXRtYXAKCk9mIGFsbCBtaVJOQXMKCmBgYHtyIG1pUk5BLWhlYXRtYXBzLXZzZH0KaGVhdF9jb2xvcnMgPC0gcmV2KGJyZXdlci5wYWwoMTIsICJSZFlsQnUiKSkKcGhlYXRtYXAoYXMubWF0cml4KEFwdWxfY291bnRzX21pUk5BX3ZzZFthcHBseShBcHVsX2NvdW50c19taVJOQV92c2QsIDEsIHZhcikgPiAwLCBdKSwgCiAgICAgICAgIGNsdXN0ZXJfcm93cyA9IFRSVUUsCiAgICAgICAgIGNsdXN0ZXJfY29scyA9IFRSVUUsCiAgICAgICAgIHNob3dfcm93bmFtZXMgPSBUUlVFLAogICAgICAgICBzaG93X2NvbG5hbWVzID0gVFJVRSwKICAgICAgICAgY29sb3IgPSBoZWF0X2NvbG9ycywKICAgICAgICAgc2NhbGU9InJvdyIpCgpgYGAKCgojIE5vcm1hbGl6ZWQgc2lSTkEgY291bnRzCgojIyBJc29sYXRlIG5vcm1hbGl6ZWQvdnNkIHNpUk5BCmBgYHtyIHNpUk5BLW5vcm1hbGl6ZWQtc2lSTkF9CgpBcHVsX2NvdW50c19zUk5BX25vcm0kTmFtZSA8LSByb3duYW1lcyhBcHVsX2NvdW50c19zUk5BX25vcm0pCkFwdWxfY291bnRzX3NSTkFfdnNkJE5hbWUgPC0gcm93bmFtZXMoQXB1bF9jb3VudHNfc1JOQV92c2QpCgpBcHVsX2NvdW50c19zaVJOQV9uYW1lc2RmIDwtIGRhdGEuZnJhbWUoTmFtZSA9IHJvd25hbWVzKEFwdWxfY291bnRzX3NpUk5BKSkgCgpBcHVsX2NvdW50c19zaVJOQV9ub3JtIDwtIGxlZnRfam9pbihBcHVsX2NvdW50c19zaVJOQV9uYW1lc2RmLCBBcHVsX2NvdW50c19zUk5BX25vcm0sIGJ5ID0gYygiTmFtZSIgPSAiTmFtZSIpKSAlPiUKICBjb2x1bW5fdG9fcm93bmFtZXModmFyID0gIk5hbWUiKQp3cml0ZS50YWJsZShBcHVsX2NvdW50c19zaVJOQV9ub3JtLCBmaWxlID0gIi4uL291dHB1dC8wMy4xLUFwdWwtc1JOQS1zdW1tYXJ5L0FwdWxfY291bnRzX3NpUk5BX25vcm1hbGl6ZWQudHh0Iiwgc2VwID0gIlx0Iiwgcm93Lm5hbWVzID0gVFJVRSwgY29sLm5hbWVzID0gVFJVRSxxdW90ZSA9IEZBTFNFKQoKQXB1bF9jb3VudHNfc2lSTkFfdnNkIDwtIGxlZnRfam9pbihBcHVsX2NvdW50c19zaVJOQV9uYW1lc2RmLCBBcHVsX2NvdW50c19zUk5BX3ZzZCwgYnkgPSBjKCJOYW1lIiA9ICJOYW1lIikpICU+JQogIGNvbHVtbl90b19yb3duYW1lcyh2YXIgPSAiTmFtZSIpCndyaXRlLnRhYmxlKEFwdWxfY291bnRzX3NpUk5BX3ZzZCwgZmlsZSA9ICIuLi9vdXRwdXQvMDMuMS1BcHVsLXNSTkEtc3VtbWFyeS9BcHVsX2NvdW50c19zaVJOQV92YXJpYW5jZXN0YWJpbGl6ZWQudHh0Iiwgc2VwID0gIlx0Iiwgcm93Lm5hbWVzID0gVFJVRSwgY29sLm5hbWVzID0gVFJVRSxxdW90ZSA9IEZBTFNFKQpgYGAKCiMjIE5vcm1hbGl6ZWQgZXhwcmVzc2lvbiBsZXZlbHMKClBsb3QgaGlzdG9ncmFtcyBvZiB0aGUgbm9ybWFsaXplZCBleHByZXNzaW9uIGxldmVscyBpbiBlYWNoIHNhbXBsZQoKYGBge3Igc2lSTkEtbm9ybS1leHByZXNzaW9uLWxldmVsLWhpc3RvZ3JhbXN9CiMgTWVsdCB0aGUgY291bnQgbWF0cml4IGludG8gbG9uZyBmb3JtYXQKQXB1bF9jb3VudHNfc2lSTkFfbm9ybV9tZWx0ZWQgPC0gbWVsdChBcHVsX2NvdW50c19zaVJOQV9ub3JtLCB2YXJpYWJsZS5uYW1lID0gInNhbXBsZSIsIHZhbHVlLm5hbWUgPSAiY291bnRzIikKCiMgUGxvdCB0aGUgZXhwcmVzc2lvbiBsZXZlbCBoaXN0b2dyYW1zIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QoQXB1bF9jb3VudHNfc2lSTkFfbm9ybV9tZWx0ZWQsIGFlcyh4ID0gY291bnRzKSkgKwogIGdlb21faGlzdG9ncmFtKGJpbndpZHRoID0gMSwgZmlsbCA9ICIjNDA4RUM2IiwgY29sb3IgPSAiYmxhY2siKSArCiAgc2NhbGVfeF9sb2cxMCgpICsgICMgT3B0aW9uYWw6IExvZy10cmFuc2Zvcm0gdGhlIHgtYXhpcyBmb3IgYmV0dGVyIHZpc3VhbGl6YXRpb24KICBmYWNldF93cmFwKH5zYW1wbGUsIHNjYWxlcyA9ICJmcmVlX3kiKSArCiAgbGFicyh0aXRsZSA9ICJHZW5lIEV4cHJlc3Npb24gTGV2ZWwgSGlzdG9ncmFtIGZvciBFYWNoIFNhbXBsZSIsCiAgICAgICB4ID0gIkV4cHJlc3Npb24gTGV2ZWwgKENvdW50cykiLAogICAgICAgeSA9ICJGcmVxdWVuY3kiKSArCiAgdGhlbWVfbWluaW1hbCgpCmBgYAoKIyMgTm9ybWFsaXplZCB0cmFuc2NyaXB0IGNvdW50cwoKQ2hlY2sgdGhlIHRvdGFsIG51bWJlciBvZiB0cmFuc2NyaXB0cyBpbiBlYWNoIHNhbXBsZSAtLSBub3cgdGhhdCB3ZSd2ZSBub3JtYWxpemVkIHRoZSBkYXRhIHRoZXNlIHRvdGFscyBzaG91bGQgYmUgc2ltaWxhcgpgYGB7ciBzaVJOQS1ub3JtLXRyYW5zY3JpcHQtY291bnRzLXBsb3R9CiMgQ2FsY3VsYXRlIHRoZSB0b3RhbCBudW1iZXIgb2YgdHJhbnNjcmlwdHMgZm9yIGVhY2ggc2FtcGxlCnRvdGFsX3RyYW5zY3JpcHRzX3NpUk5BX25vcm0gPC0gY29sU3VtcyhBcHVsX2NvdW50c19zaVJOQV9ub3JtKQoKIyBDcmVhdGUgYSBkYXRhIGZyYW1lIGZvciBwbG90dGluZwp0b3RhbF90cmFuc2NyaXB0c19zaVJOQV9ub3JtX2RmIDwtIGRhdGEuZnJhbWUoc2FtcGxlID0gbmFtZXModG90YWxfdHJhbnNjcmlwdHNfc2lSTkFfbm9ybSksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdG90YWxzID0gdG90YWxfdHJhbnNjcmlwdHNfc2lSTkFfbm9ybSkKCiMgUGxvdCB0aGUgdG90YWwgbnVtYmVyIG9mIHRyYW5zY3JpcHRzIGZvciBlYWNoIHNhbXBsZQpnZ3Bsb3QodG90YWxfdHJhbnNjcmlwdHNfc2lSTkFfbm9ybV9kZiwgYWVzKHggPSBzYW1wbGUsIHkgPSB0b3RhbHMpKSArCiAgZ2VvbV9iYXIoc3RhdCA9ICJpZGVudGl0eSIsIGZpbGwgPSAiIzQwOEVDNiIsIGNvbG9yID0gImJsYWNrIikgKwogIGdlb21fdGV4dChhZXMobGFiZWwgPSB0b3RhbHMpLCB2anVzdCA9IC0wLjMsIHNpemUgPSAzLjUpICsKICBsYWJzKHRpdGxlID0gIlRvdGFsIE51bWJlciBvZiBzaVJOQSBUcmFuc2NyaXB0cyBwZXIgU2FtcGxlIiwKICAgICAgIHggPSAiU2FtcGxlIiwKICAgICAgIHkgPSAiVG90YWwgVHJhbnNjcmlwdHMiKSArCiAgdGhlbWVfbWluaW1hbCgpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSAgIyBSb3RhdGUgeC1heGlzIGxhYmVscyBmb3IgcmVhZGFiaWxpdHkKYGBgCgojIyBIZWF0bWFwCgpPZiBhbGwgc2lSTkFzCgpgYGB7ciBzaVJOQS1oZWF0bWFwcy12c2R9CmhlYXRfY29sb3JzIDwtIHJldihicmV3ZXIucGFsKDEyLCAiUmRZbEJ1IikpCnBoZWF0bWFwKGFzLm1hdHJpeChBcHVsX2NvdW50c19zaVJOQV92c2RbYXBwbHkoQXB1bF9jb3VudHNfc2lSTkFfdnNkLCAxLCB2YXIpID4gMCwgXSksIAogICAgICAgICBjbHVzdGVyX3Jvd3MgPSBUUlVFLAogICAgICAgICBjbHVzdGVyX2NvbHMgPSBUUlVFLAogICAgICAgICBzaG93X3Jvd25hbWVzID0gVFJVRSwKICAgICAgICAgc2hvd19jb2xuYW1lcyA9IFRSVUUsCiAgICAgICAgIGNvbG9yID0gaGVhdF9jb2xvcnMsCiAgICAgICAgIHNjYWxlPSJyb3ciKQoKYGBgCgoK