El presente tutorial está basado en la publicación del blog variance explained de David Robinson. El material ha sido actualizado y adaptado al objetivo del curso.

Objetivo

  • Aplicar las herramientas y estilo del tidyverse en la limpieza, exploración y visualización de datos genómicos.

Nota: Todos los outputs deben verse como líneas corridas. Disminuir el zoom de ser necesario.

Dependencies

library(tidyverse)
Loading tidyverse: ggplot2
Loading tidyverse: tibble
Loading tidyverse: tidyr
Loading tidyverse: readr
Loading tidyverse: purrr
Loading tidyverse: dplyr
Conflicts with tidy packages -------------------------------------------------------------
filter(): dplyr, stats
lag():    dplyr, stats

Contexto: Expresión de genes en inanición

Through the process of gene regulation, a cell can control which genes are transcribed from DNA to RNA- what we call being “expressed”. (If a gene is never turned into RNA, it may as well not be there at all). This provides a sort of “cellular switchboard” that can activate some systems and deactivate others, which can speed up or slow down growth, switch what nutrients are transported into or out of the cell, and respond to other stimuli. A gene expression microarray lets us measure how much of each gene is expressed in a particular condition. We can use this to figure out the function of a specific gene (based on when it turns on and off), or to get an overall picture of the cell’s activity.

Brauer 2008 used microarrays to test the effect of starvation and growth rate on baker’s yeast (S. cerevisiae, a popular model organism for studying molecular genomics because of its simplicity)1. Basically, if you give yeast plenty of nutrients (a rich media), except that you sharply restrict its supply of one nutrient, you can control the growth rate to whatever level you desire (we do this with a tool called a chemostat). For example, you could limit the yeast’s supply of glucose (sugar, which the cell metabolizes to get energy and carbon), of leucine (an essential amino acid), or of ammonium (a source of nitrogen).

“Starving” the yeast of these nutrients lets us find genes that:

  • Raise or lower their activity in response to growth rate. Growth-rate dependent expression patterns can tell us a lot about cell cycle control, and how the cell responds to stress.
  • Respond differently when different nutrients are being limited. These genes may be involved in the transport or metabolism of those nutrients.

Sounds pretty cool, right? So let’s get started!

Importar

original_data <- readRDS("data-raw/tidymicro.rds")

Reconocer variables

Each of those columns like G0.05, N0.3 and so on represents gene expression values for that sample, as measured by the microarray. The column titles show the condition: G0.05, for instance, means the limiting nutrient was glucose and the growth rate was .05. A higher value means the gene was more expressed in that sample, lower means the gene was less expressed. In total the yeast was grown with six limiting nutrients and six growth rates, which makes 36 samples, and therefore 36 columns, of gene expression data.

Diseño experimental:

  • 6 nutrientes limitantes
  • 6 tasas de crecimiento

Columnas:

  • 36 con tasa de crecimiento por nutriente limitado
  • 4 con identificadores
dim(original_data)
[1] 5537   40
head(original_data)
glimpse(original_data)
Observations: 5,537
Variables: 40
$ GID     <chr> "GENE1331X", "GENE4924X", "GENE4690X", "GENE1177X", "GENE511X", "GENE...
$ YORF    <chr> "A_06_P5820", "A_06_P5866", "A_06_P1834", "A_06_P4928", "A_06_P5620",...
$ NAME    <chr> "SFB2       || ER to Golgi transport || molecular function unknown ||...
$ GWEIGHT <int> 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, ...
$ G0.05   <dbl> -0.24, 0.28, -0.02, -0.33, 0.05, -0.69, -0.55, -0.75, -0.24, -0.16, -...
$ G0.1    <dbl> -0.13, 0.13, -0.27, -0.41, 0.02, -0.03, -0.30, -0.12, -0.22, -0.38, -...
$ G0.15   <dbl> -0.21, -0.40, -0.27, -0.24, 0.40, 0.23, -0.12, -0.07, 0.14, 0.05, 0.2...
$ G0.2    <dbl> -0.15, -0.48, -0.02, -0.03, 0.34, 0.20, -0.03, 0.02, 0.06, 0.14, 0.18...
$ G0.25   <dbl> -0.05, -0.11, 0.24, -0.03, -0.13, 0.00, -0.16, -0.32, 0.00, -0.04, 0....
$ G0.3    <dbl> -0.05, 0.17, 0.25, 0.00, -0.14, -0.27, -0.11, -0.41, -0.13, -0.01, -0...
$ N0.05   <dbl> 0.20, 0.31, 0.23, 0.20, -0.35, 0.17, 0.04, 0.11, 0.30, 0.39, 0.26, -0...
$ N0.1    <dbl> 0.24, 0.00, 0.06, -0.25, -0.09, -0.40, 0.00, -0.16, 0.07, 0.20, 0.15,...
$ N0.15   <dbl> -0.20, -0.63, -0.66, -0.49, -0.08, -0.54, -0.63, -0.26, -0.30, 0.27, ...
$ N0.2    <dbl> -0.42, -0.44, -0.40, -0.49, -0.58, -1.19, -0.51, -0.42, -0.01, 0.19, ...
$ N0.25   <dbl> -0.14, -0.26, -0.46, -0.43, -0.14, -0.42, -0.37, 0.18, 0.15, 0.20, 0....
$ N0.3    <dbl> 0.09, 0.21, -0.43, -0.26, -0.12, 1.89, -0.24, 0.13, 0.13, 0.06, -0.08...
$ P0.05   <dbl> -0.26, -0.09, 0.18, 0.05, -0.16, -0.32, -0.35, -0.19, -0.26, -0.23, -...
$ P0.1    <dbl> -0.20, -0.04, 0.22, 0.04, 0.18, -0.06, -0.32, -0.25, -0.20, -0.20, -0...
$ P0.15   <dbl> -0.22, -0.10, 0.33, 0.03, 0.21, -0.62, -0.39, -0.25, -0.22, -0.07, -0...
$ P0.2    <dbl> -0.31, 0.15, 0.34, -0.04, 0.08, -0.50, -0.60, -0.47, -0.17, -0.13, -0...
$ P0.25   <dbl> 0.04, 0.20, 0.13, 0.08, 0.23, -0.37, -0.29, -0.24, -0.23, -0.14, -0.4...
$ P0.3    <dbl> 0.34, 0.63, 0.44, 0.21, -0.29, NA, -0.25, -0.49, -0.38, -0.42, -0.63,...
$ S0.05   <dbl> -0.51, 0.53, 1.29, 0.41, -0.70, NA, -0.14, 0.09, -0.35, -0.38, 0.25, ...
$ S0.1    <dbl> -0.12, 0.15, -0.32, -0.43, 0.05, -0.20, -0.50, 0.13, -0.14, -0.14, 0....
$ S0.15   <dbl> 0.09, -0.01, -0.47, -0.21, 0.10, -0.09, -0.19, 0.15, 0.10, 0.00, 0.25...
$ S0.2    <dbl> 0.09, 0.12, -0.50, -0.33, -0.07, 0.06, -0.13, -0.02, -0.04, -0.06, 0....
$ S0.25   <dbl> 0.20, -0.15, -0.42, -0.05, -0.10, -0.19, -0.01, 0.24, 0.22, 0.16, 0.2...
$ S0.3    <dbl> 0.08, 0.32, -0.33, -0.24, -0.32, -0.14, -0.04, -0.08, 0.02, -0.15, -0...
$ L0.05   <dbl> 0.18, 0.16, -0.30, -0.27, -0.59, -0.17, -0.02, -0.11, 0.12, -0.20, -0...
$ L0.1    <dbl> 0.18, 0.09, 0.02, -0.28, -0.13, -0.07, -0.05, -0.01, -0.01, -0.18, 0....
$ L0.15   <dbl> 0.13, 0.02, -0.07, -0.05, 0.00, 0.25, 0.27, 0.15, 0.17, 0.11, -0.04, ...
$ L0.2    <dbl> 0.20, 0.04, -0.05, 0.02, -0.11, -0.21, 0.24, 0.15, 0.07, 0.00, -0.13,...
$ L0.25   <dbl> 0.17, 0.03, -0.13, 0.00, 0.04, 0.12, 0.05, 0.00, 0.10, 0.02, -0.08, -...
$ L0.3    <dbl> 0.11, 0.01, -0.04, 0.08, 0.01, -0.11, 0.19, 0.03, 0.11, 0.09, 0.10, -...
$ U0.05   <dbl> -0.06, -1.02, -0.91, -0.53, -0.45, NA, 0.07, -0.40, 0.01, -0.26, -0.0...
$ U0.1    <dbl> -0.26, -0.91, -0.94, -0.51, -0.09, -0.65, -0.31, -0.02, -0.16, -0.13,...
$ U0.15   <dbl> -0.05, -0.59, -0.42, -0.26, -0.13, 0.09, -0.08, 0.26, 0.07, -0.10, 0....
$ U0.2    <dbl> -0.28, -0.61, -0.36, 0.05, 0.02, 0.06, 0.12, 0.31, 0.20, 0.07, 0.02, ...
$ U0.25   <dbl> -0.19, -0.17, -0.49, -0.14, -0.09, -0.07, 0.05, 0.14, 0.02, -0.04, -0...
$ U0.3    <dbl> 0.09, 0.18, -0.47, -0.01, -0.03, -0.10, 0.06, 0.11, 0.10, -0.12, -0.2...

Limpieza de datos

Usando dplyr y tidyr

  • Una data “limpia” o “Tidy data” o sigue las siguientes reglas:

    1. Each variable forms a column.
    2. Each observation forms a row.
    3. Each type of observational unit forms a table.
  • ¿Qué no está limpio (“untidy”) en la data?

    1. Column headers are values, not variable names. Our column names contain the values of two variables: nutrient (G, N, P, etc) and growth rate (0.05-0.3). For this reason, we end up with not one observation per row, but 36! This is a very common issue in biological datasets: you often see one-row-per-gene and one-column-per-sample, rather than one-row-per-gene-per-sample.

    2. Multiple variables are stored in one column. The NAME column contains lots of information, split up by ||’s. If we examine one of the names, it looks like

      SFB2 || ER to Golgi transport || molecular function unknown || YNL049C || 1082129

      which seems to have both some systematic IDs and some biological information about the gene. If we’re going to use this programmatically, we need to split up the information into multiple columns.

  • “The more effort you put up front into tidying your data, the easier it will be to explore interactively.”
    • Since the analysis steps are where you’ll actually be answering questions about your data, it’s worth putting up this effort!

Múltiples variables están guardadas en una misma columna

original_data$NAME[1:2]
[1] "SFB2       || ER to Golgi transport || molecular function unknown || YNL049C || 1082129"    
[2] "          || biological process unknown || molecular function unknown || YNL095C || 1086222"
SFB2       || ER to Golgi transport || molecular function unknown || YNL049C || 1082129

          || biological process unknown || molecular function unknown || YNL095C || 1086222
  • The details of each of these fields isn’t annotated in the paper, but we can figure out most of it. It contains:

    • Gene name e.g. SFB2. Note that not all genes have a name.
    • Biological process e.g. “proteolysis and peptidolysis”
    • Molecular function e.g. “metalloendopeptidase activity”
    • Systematic ID e.g. YNL049C. Unlike a gene name, every gene in this dataset has a systematic ID.
    • Another ID number e.g. 1082129. I don’t know what this number means, and it’s not annotated in the paper. Oh, well.
  • Having all give of these in the same column is very inconvenient. For example, if I have another dataset with information about each gene, I can’t merge the two. Luckily, the tidyr package provides the separate function for exactly this case.

library(dplyr)
library(tidyr)

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|")

head(cleaned_data)

Two more things. First, when we split by ||, we ended up with whitespace at the start and end of some of the columns, which is inconvenient:

head(cleaned_data$BP)
[1] " ER to Golgi transport "        " biological process unknown "  
[3] " proteolysis and peptidolysis " " mRNA polyadenylylation* "     
[5] " vesicle fusion* "              " biological process unknown "  
 ER to Golgi transport 

 biological process unknown 

 proteolysis and peptidolysis 

 mRNA polyadenylylation* 

 vesicle fusion* 

 biological process unknown 

We’ll solve that with dplyr’s mutate_each, along with the built-in trimws (“trim whitespace”) function.

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws))

head(cleaned_data$BP)
[1] "ER to Golgi transport"        "biological process unknown"  
[3] "proteolysis and peptidolysis" "mRNA polyadenylylation*"     
[5] "vesicle fusion*"              "biological process unknown"  
ER to Golgi transport

biological process unknown

proteolysis and peptidolysis

mRNA polyadenylylation*

vesicle fusion*

biological process unknown

Finally, we don’t even know what the number column represents (if you can figure it out, let me know!) And while we’re at it, we’re not going to use the GID, YORF or GWEIGHT columns in this analysis either. We may as well drop them, which we can do with dplyr’s select.

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) 

head(cleaned_data)

Los titulos de columnas son observaciones, no variables

  • Let’s take a closer look at all those column headers like G0.05, N0.2 and P0.15.

    • Limiting nutrient. This has six possible values: glucose (G), ammonium (N), sulfate (S), phosphate (P), uracil (U) or leucine (L).
    • Growth rate: A number, ranging from .05 to .3. .05 means slow growth (the yeast were being starved hard of that nutrient) while .3 means fast growth. (Technically, this value measures the dilution rate from the chemostat).
    • Expression level. These are the values currently stored in those columns, as measured by the microarray. (Note that the paper already did some pre-processing and normalization on these values, which we’re ignoring here).
  • The rules of tidy data specify that each variable forms one column, and this is not even remotely the case- we have 36 columns when we should have 3. That means our data is trapped in our column names. If you don’t see why this is a problem, consider: how would you put growth rate on the x-axis of a graph? How would you filter to look only the glucose condition?

  • Luckily, the tidyr package has a solution ready. The documentation for gather notes (emphasis mine):

    Gather takes multiple columns and collapses into key-value pairs, duplicating all other columns as needed. You use gather() when you notice that you have columns that are not variables.

  • Hey, that’s us! So let’s apply gather as our next step:

  • OJO: Experimentar con el comando dplyr::count(sample) luego de aplicar gather() para confirmar la frecuencia de datos por observación.

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) %>% 
  gather(sample, expression, G0.05:U0.3)

#cleaned_data
cleaned_data %>% dplyr::count(sample)
  • Notice that the dataset no longer consists of one-row-per-gene: it’s one-row-per-gene-per-sample. This has previously been called “melting” a dataset, or turning it into “long” format. But I like the term “gather”: it shows that we’re taking these 36 columns and pulling them together.

  • One last problem. That sample column really contains two variables, nutrient and rate. We already learned what to do when we have two variables in one column: use separate:

library(dplyr)
library(tidyr)

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) %>%
  gather(sample, expression, G0.05:U0.3) %>% #dplyr::count(sample)
  separate(sample, c("nutrient", "rate"), sep = 1, convert = TRUE) 

head(cleaned_data)
  • This time, instead of telling separate to split the strings based on a particular delimiter, we told it to separate it after the first character (that is, after G/P/S/N/L/U). We also told it convert = TRUE to tell it that it should notice the 0.05/0.1/etc value is a number and convert it.

  • Take a look at those six lines of code, a mini-sonnet of data cleaning. Doesn’t it read less like code and more like instructions? (“First we separated the NAME column into its five parts, and trimmed each. We selected out columns we didn’t need…”) That’s the beauty of the %>% operator and the dplyr/tidyr verbs.

Visualizaciones con ggplot

¿Por qué limpiar la data?

  • So we went through this effort to get this dataset into this structure, and you’re probably wondering why. In particular, why did we have to bother gathering those expression columns into one-row-per-gene-per-sample?

  • Well, suppose we have a single yeast gene we’re interested in. Let’s say LEU1, a gene involved in the leucine synthesis pathway.

cleaned_data %>%
  filter(name == "LEU1") %>%
  glimpse()
Observations: 36
Variables: 7
$ name            <chr> "LEU1", "LEU1", "LEU1", "LEU1", "LEU1", "LEU1", "LEU1", "LEU1...
$ BP              <chr> "leucine biosynthesis", "leucine biosynthesis", "leucine bios...
$ MF              <chr> "3-isopropylmalate dehydratase activity", "3-isopropylmalate ...
$ systematic_name <chr> "YGL009C", "YGL009C", "YGL009C", "YGL009C", "YGL009C", "YGL00...
$ nutrient        <chr> "G", "G", "G", "G", "G", "G", "N", "N", "N", "N", "N", "N", "...
$ rate            <dbl> 0.05, 0.10, 0.15, 0.20, 0.25, 0.30, 0.05, 0.10, 0.15, 0.20, 0...
$ expression      <dbl> -1.12, -0.77, -0.67, -0.59, -0.20, 0.03, -0.76, -1.17, -1.20,...

We now have 36 data points (six conditions, six growth rates), and for each we have a limiting nutrient, a growth rate, and the resulting expresion. We’re probably interested in how both the growth rate and the limiting nutrient affect the gene’s expression.

36 points is too many to look at manually. So it’s time to bring in some visualization. To do that, we simply pipe the results of our filtering right into ggplot2:

cleaned_data %>% 
  filter(name=="LEU1") %>% 
  ggplot(aes(rate,expression,colour=nutrient))+
  geom_line()

  • What a story this single gene tells! The gene’s expression is far higher (more “turned on”) when the cell is being starved of leucine than in any other condition, because in that case the cell has to synthesize its own leucine. And as the amount of leucine in the environment (the growth rate) increases, the cell can focus less on leucine production, and the expression of those genes go down. We’ve just gotten one snapshot of our gene’s regulatory network, and how it responds to external stimuli.

  • We don’t have to choose one gene to visualize- LEU1 is just one gene in the leucine biosynthesis process. Recall that we have that information in the BP column, so we can filter for all genes in that process, and then facet to create sub-plots for each.

cleaned_data %>%
  filter(BP == "leucine biosynthesis") %>%
  ggplot(aes(rate, expression, color = nutrient)) +
  geom_line() +
  facet_wrap(~name)

  • LEU1, LEU2, and LEU4 all show a similar pattern, where starvation of leucine causes higher gene expression. (Interestingly, LEU4 responds to glucose starvation as well. Any geneticists have some idea why?). LEU9 is a little more ambiguous but is still highest expressed under leucine starvation. We already know what these genes do, but this hints at how we might be able to find other genes that are involved in leucine synthesis, including ones we don’t yet know.

  • Let’s play with graph a little more. These trends look vaguely linear. Maybe we should show best points with best fit lines instead:

cleaned_data %>%
  filter(BP == "leucine biosynthesis") %>%
  ggplot(aes(rate, expression, color = nutrient)) +
  geom_point() +
  geom_smooth(method = "lm", se = FALSE) +
  facet_wrap(~name)

  • The options for exploratory data analysis are endless. We could instead look at sulfur metabolism:
cleaned_data %>%
  filter(BP == "sulfur metabolism") %>%
  ggplot(aes(rate, expression, color = nutrient)) +
  geom_point() +
  geom_smooth(method = "lm", se = FALSE) +
  facet_wrap(~name + systematic_name, scales = "free_y")
Warning: Removed 3 rows containing non-finite values (stat_smooth).
Warning: Removed 3 rows containing missing values (geom_point).

(Notice that we have to facet by the systematic_name here, since not all genes in this process have traditional names).

  • If you were an interested molecular biologist, you could go a long way just by examining various biological processes, or looking at lists of genes you were interested in. This is a great way to explore a dataset interactively, and to hint at methods of further analysis. For example, we notice that we can fit linear models to many of these expression profiles that will come in handy in the next part.

Conclusión: data específica, herramientas generalizadas

I earlier pointed to a list of available workflows from Bioconductor, which teach ways to analyze many kinds of genomic data using packages specialized for those purposes. These kinds of guides are incredibly valuable, and Bioconductor has built up an excellent set of tools for analyzing genomic data, many of which come with their own data processing and visualization methods.

So why bother teaching a dplyr/ggplot2 approach? Because these tools are useful everywhere. Consider the dozen lines of code we used to clean and visualize our data:

library(dplyr)
library(tidyr)

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) %>%
  gather(sample, expression, G0.05:U0.3) %>% #dplyr::count(sample)
  separate(sample, c("nutrient", "rate"), sep = 1, convert = TRUE)

cleaned_data %>%
  filter(BP == "leucine biosynthesis") %>%
  ggplot(aes(rate, expression, color = nutrient)) +
  geom_point() +
  geom_smooth(method = "lm", se = FALSE) +
  facet_wrap(~name)

With this code we’re able to go from published data to a new conclusion about leucine biosynthesis (or sulfate metabolism, or whatever biological process or genes we’re interested in). But the functions we use aren’t specific to our dataset or data format: in fact, they’re not related to biology at all. Instead, these tools are building blocks, or “atoms,” in a grammar of data manipulation and visualization that applies to nearly every kind of data.

This isn’t meant to disparage Bioconductor in any way: scientists should use whatever tool gets their job done. But educators can and should focus on teaching tools that can be universally applied. In turn, students can take these tools and build new packages, for Bioconductor and elsewhere, that analyze novel forms of data.

Más limpieza

Tres pasos más de limpieza:

  • First, I spelled out the full names of the nutrients- “Glucose” instead of just “G”, for example.
  • Second, I filtered out missing values from the expression column, as well as genes that have no systematic ID
  • Tercero, filtrar las genes con lecturas completas para tasa de crecimiento por nutriente de cultivo limitado.

retirar observaciones NA’s

¿Cómo revisar la presencia de NA?

# tipo de variable: character <chr>
m <- cleaned_data %>% 
  filter(systematic_name=="") %>% 
  dplyr::count()
# tipo de variable: double <dbl>
n <- cleaned_data %>% 
  filter(is.na(expression)) %>% 
  dplyr::count(expression)
  • la variable systematic_name posee 36 NA’s
  • la variable expression posee 866 NA’s

cambiar el nombre a factores

Ejecutar la limpieza

  1. cambiar de nombre a los fatores de la variable nutrient
  2. retirar a los elementos con:
    • systematic_namescon NA’s (porque no habría cómo identificar al gen)
    • valores de expression con NA’s
library(forcats)
cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) %>%
  gather(sample, expression, G0.05:U0.3) %>% #dplyr::count(sample)
  separate(sample, c("nutrient", "rate"), sep = 1, convert = TRUE) %>% 
  mutate(nutrient=forcats::fct_recode(nutrient, 
                                      "Glucose" = "G", "Leucine" = "L",
                                      "Phosphate" = "P", "Sulfate" = "S",
                                      "Ammonia" = "N", "Uracil" = "U")) %>% 
  filter(!is.na(expression),systematic_name!="") %>% 
  glimpse()
Observations: 198,430
Variables: 7
$ name            <chr> "SFB2", "", "QRI7", "CFT2", "SSO2", "PSP2", "RIB2", "VMA13", ...
$ BP              <chr> "ER to Golgi transport", "biological process unknown", "prote...
$ MF              <chr> "molecular function unknown", "molecular function unknown", "...
$ systematic_name <chr> "YNL049C", "YNL095C", "YDL104C", "YLR115W", "YMR183C", "YML01...
$ nutrient        <fctr> Glucose, Glucose, Glucose, Glucose, Glucose, Glucose, Glucos...
$ rate            <dbl> 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0...
$ expression      <dbl> -0.24, 0.28, -0.02, -0.33, 0.05, -0.69, -0.55, -0.75, -0.24, ...

retirar combinaciones con observaciones incompletas

¿Todos los genes tienen valores de expresión?

y <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) %>%
  gather(sample, expression, G0.05:U0.3) %>% #dplyr::count(sample)
  separate(sample, c("nutrient", "rate"), sep = 1, convert = TRUE) %>% 
  mutate(nutrient=forcats::fct_recode(nutrient, 
                                      "Glucose" = "G", "Leucine" = "L",
                                      "Phosphate" = "P", "Sulfate" = "S",
                                      "Ammonia" = "N", "Uracil" = "U")) %>% 
  filter(!is.na(expression),systematic_name!="") %>% 
  #filter(systematic_name=="Q0140", nutrient=="Phosphate") # ejemplo con solo dos lecturas de gen x nutriente
  group_by(systematic_name, nutrient) %>% dplyr::count() %>% 
  ungroup() %>% count(n) %>% filter(n<6) #%>% summarise(sum(nn))
  • Hay 678 genes por nutriente con menos de 6 observaciones de la variable rate para las combinaciones entre systematic_name (gen) y nutrient (nutriente limitante).

final

Limpieza final

cleaned_data <- original_data %>%
  separate(NAME, c("name", "BP", "MF", "systematic_name", "number"), sep = "\\|\\|") %>%
  mutate_at(vars(name:systematic_name), funs(trimws)) %>%
  select(-number, -GID, -YORF, -GWEIGHT) %>%
  gather(sample, expression, G0.05:U0.3) %>% #dplyr::count(sample)
  separate(sample, c("nutrient", "rate"), sep = 1, convert = TRUE) %>% 
  mutate(nutrient=forcats::fct_recode(nutrient, 
                                      "Glucose" = "G", "Leucine" = "L",
                                      "Phosphate" = "P", "Sulfate" = "S",
                                      "Ammonia" = "N", "Uracil" = "U")) %>% 
  filter(!is.na(expression),systematic_name!="") %>% 
  group_by(systematic_name, nutrient) %>%
  filter(n()==6) %>% 
  ungroup() %>% 
  glimpse()
Observations: 195,222
Variables: 7
$ name            <chr> "SFB2", "", "QRI7", "CFT2", "SSO2", "PSP2", "RIB2", "VMA13", ...
$ BP              <chr> "ER to Golgi transport", "biological process unknown", "prote...
$ MF              <chr> "molecular function unknown", "molecular function unknown", "...
$ systematic_name <chr> "YNL049C", "YNL095C", "YDL104C", "YLR115W", "YMR183C", "YML01...
$ nutrient        <fctr> Glucose, Glucose, Glucose, Glucose, Glucose, Glucose, Glucos...
$ rate            <dbl> 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0.05, 0...
$ expression      <dbl> -0.24, 0.28, -0.02, -0.33, 0.05, -0.69, -0.55, -0.75, -0.24, ...

Aplicación

Tidying the data in this way lets us make graphs like this

cleaned_data %>%
  filter(BP == "leucine biosynthesis") %>%
  ggplot(aes(rate, expression, color = nutrient)) +
  geom_point() +
  geom_smooth(method = "lm", se = FALSE) +
  facet_wrap(~name + systematic_name)

For starters, let’s wrap this useful graph into a function, so that we can make it easily in the rest of the post.

plot_expression_data <- function(expression_data) {
  ggplot(expression_data, aes(rate, expression, color = nutrient)) +
    geom_point() +
    geom_smooth(method = "lm", se = FALSE) +
    facet_wrap(~name + systematic_name, scales = "free_y")
}

At which point we would rewrite the above graph like:

cleaned_data %>%
  filter(BP == "leucine biosynthesis") %>%
  plot_expression_data()

This is a great way to visualize a few genes at a time. But there are so many genes in the dataset. For example, let’s instead filter by the biological process cell wall organization and biogenesis.

cleaned_data %>%
  filter(BP == "cell wall organization and biogenesis") %>%
  plot_expression_data()

OK, that’s 36 genes and it’s already getting a lot harder to understand these plots. And we have 5500 genes in this dataset: no way can we visually interpret all those genes at once. This is where we introduce modeling.

Computer environment

devtools::session_info()
Session info ----------------------------------------------------------------------------
 setting  value                       
 version  R version 3.4.1 (2017-06-30)
 system   x86_64, linux-gnu           
 ui       X11                         
 language en_US                       
 collate  en_US.UTF-8                 
 tz       America/Lima                
 date     2017-08-03                  
Packages --------------------------------------------------------------------------------
 package    * version date       source         
 assertthat   0.2.0   2017-04-11 CRAN (R 3.4.0) 
 backports    1.1.0   2017-05-22 CRAN (R 3.4.1) 
 base       * 3.4.1   2017-07-08 local          
 base64enc    0.1-3   2015-07-28 CRAN (R 3.4.0) 
 bindr        0.1     2016-11-13 cran (@0.1)    
 bindrcpp   * 0.2     2017-06-17 CRAN (R 3.4.1) 
 broom        0.4.2   2017-02-13 CRAN (R 3.4.0) 
 cellranger   1.1.0   2016-07-27 CRAN (R 3.4.0) 
 colorspace   1.3-2   2016-12-14 CRAN (R 3.4.0) 
 compiler     3.4.1   2017-07-08 local          
 datasets   * 3.4.1   2017-07-08 local          
 devtools     1.13.2  2017-06-02 CRAN (R 3.4.1) 
 digest       0.6.12  2017-01-27 CRAN (R 3.4.0) 
 dplyr      * 0.7.2   2017-07-20 CRAN (R 3.4.1) 
 evaluate     0.10.1  2017-06-24 CRAN (R 3.4.1) 
 forcats    * 0.2.0   2017-01-23 CRAN (R 3.4.0) 
 foreign      0.8-69  2017-06-21 CRAN (R 3.4.1) 
 ggplot2    * 2.2.1   2016-12-30 CRAN (R 3.4.0) 
 glue         1.1.1   2017-06-21 CRAN (R 3.4.1) 
 graphics   * 3.4.1   2017-07-08 local          
 grDevices  * 3.4.1   2017-07-08 local          
 grid         3.4.1   2017-07-08 local          
 gtable       0.2.0   2016-02-26 CRAN (R 3.4.0) 
 haven        1.1.0   2017-07-09 CRAN (R 3.4.1) 
 hms          0.3     2016-11-22 CRAN (R 3.4.0) 
 htmltools    0.3.6   2017-04-28 CRAN (R 3.4.0) 
 httr         1.2.1   2016-07-03 CRAN (R 3.4.0) 
 jsonlite     1.5     2017-06-01 cran (@1.5)    
 knitr        1.16    2017-05-18 cran (@1.16)   
 labeling     0.3     2014-08-23 CRAN (R 3.4.0) 
 lattice      0.20-35 2017-03-25 CRAN (R 3.3.3) 
 lazyeval     0.2.0   2016-06-12 CRAN (R 3.4.0) 
 lubridate    1.6.0   2016-09-13 CRAN (R 3.4.0) 
 magrittr     1.5     2014-11-22 CRAN (R 3.4.0) 
 memoise      1.1.0   2017-04-21 CRAN (R 3.4.0) 
 methods    * 3.4.1   2017-07-08 local          
 mnormt       1.5-5   2016-10-15 CRAN (R 3.4.0) 
 modelr       0.1.1   2017-07-24 CRAN (R 3.4.1) 
 munsell      0.4.3   2016-02-13 CRAN (R 3.4.0) 
 nlme         3.1-131 2017-02-06 CRAN (R 3.4.0) 
 parallel     3.4.1   2017-07-08 local          
 pkgconfig    2.0.1   2017-03-21 cran (@2.0.1)  
 plyr         1.8.4   2016-06-08 CRAN (R 3.4.0) 
 psych        1.7.5   2017-05-03 CRAN (R 3.4.0) 
 purrr      * 0.2.2.2 2017-05-11 cran (@0.2.2.2)
 R6           2.2.2   2017-06-17 CRAN (R 3.4.1) 
 Rcpp         0.12.12 2017-07-15 CRAN (R 3.4.1) 
 readr      * 1.1.1   2017-05-16 CRAN (R 3.4.1) 
 readxl       1.0.0   2017-04-18 CRAN (R 3.4.0) 
 reshape2     1.4.2   2016-10-22 CRAN (R 3.4.0) 
 rlang        0.1.1   2017-05-18 cran (@0.1.1)  
 rmarkdown    1.6     2017-06-15 CRAN (R 3.4.1) 
 rprojroot    1.2     2017-01-16 CRAN (R 3.4.0) 
 rvest        0.3.2   2016-06-17 CRAN (R 3.4.0) 
 scales       0.4.1   2016-11-09 CRAN (R 3.4.0) 
 stats      * 3.4.1   2017-07-08 local          
 stringi      1.1.5   2017-04-07 CRAN (R 3.4.0) 
 stringr      1.2.0   2017-02-18 CRAN (R 3.4.0) 
 tibble     * 1.3.3   2017-05-28 cran (@1.3.3)  
 tidyr      * 0.6.3   2017-05-15 CRAN (R 3.4.1) 
 tidyverse  * 1.1.1   2017-01-27 CRAN (R 3.4.0) 
 tools        3.4.1   2017-07-08 local          
 utils      * 3.4.1   2017-07-08 local          
 withr        2.0.0   2017-07-28 CRAN (R 3.4.1) 
 xml2         1.1.1   2017-01-24 CRAN (R 3.4.0) 
 yaml         2.1.14  2016-11-12 CRAN (R 3.4.0) 

References

LS0tCnRpdGxlOiAiQ2FzbyBJOiBMaW1waWV6YSB5IFZpc3VhbGl6YWNpw7NuIGRlIGRhdGEgZ2Vuw7NtaWNhIgphdXRob3I6ICJhdmFsbGVjYW0iCmRhdGU6ICdgciBTeXMuRGF0ZSgpYCcKb3V0cHV0OgogICNodG1sX2RvY3VtZW50OgogICNwZGZfZG9jdW1lbnQ6CiAgaHRtbF9ub3RlYm9vazoKICAgIHRvYzogeWVzCiAgICB0b2NfZGVwdGg6IDMKICAgIHRvY19mbG9hdDoKICAgICAgY29sbGFwc2VkOiB5ZXMKICAgICN0aGVtZTogdW5pdGVkCiAgICAjY29kZV9mb2xkaW5nOiAiaGlkZSIKICAgICNmaWdfY2FwdGlvbjogVFJVRQogICAgI251bWJlcl9zZWN0aW9uczogVFJVRQotLS0KCmBgYHtyIHNldHVwLCBpbmNsdWRlPUZBTFNFfQprbml0cjo6b3B0c19jaHVuayRzZXQoZWNobyA9IFRSVUUpCm9wdGlvbnMod2lkdGggPSA5MCkgIyBleHBhbmQgbGltaXRzIG9mIENPTlNPTEUgb3V0cHV0CmBgYAoKRWwgcHJlc2VudGUgdHV0b3JpYWwgZXN0w6EgYmFzYWRvIGVuIGxhIFtwdWJsaWNhY2nDs25dKGh0dHA6Ly92YXJpYW5jZWV4cGxhaW5lZC5vcmcvci90aWR5LWdlbm9taWNzLykgZGVsIApibG9nIFt2YXJpYW5jZSBleHBsYWluZWRdKGh0dHA6Ly92YXJpYW5jZWV4cGxhaW5lZC5vcmcvKSAKZGUgW0RhdmlkIFJvYmluc29uXShodHRwczovL3R3aXR0ZXIuY29tL2Ryb2IpLgpFbCBtYXRlcmlhbCBoYSBzaWRvIGFjdHVhbGl6YWRvIHkgYWRhcHRhZG8gYWwgb2JqZXRpdm8gZGVsIGN1cnNvLgoKIyMgT2JqZXRpdm8KCi0gQXBsaWNhciBsYXMgaGVycmFtaWVudGFzIHkgZXN0aWxvIGRlbCBgdGlkeXZlcnNlYCBlbiBsYSBsaW1waWV6YSwgZXhwbG9yYWNpw7NuIHkgdmlzdWFsaXphY2nDs24gZGUgZGF0b3MgZ2Vuw7NtaWNvcy4KCl9fTm90YTpfXyBUb2RvcyBsb3Mgb3V0cHV0cyBkZWJlbiB2ZXJzZSBjb21vIGzDrW5lYXMgY29ycmlkYXMuIF9EaXNtaW51aXIgZWwgem9vbSBkZSBzZXIgbmVjZXNhcmlvLl8KCiMjIERlcGVuZGVuY2llcwpgYGB7cn0KbGlicmFyeSh0aWR5dmVyc2UpCmBgYAoKIyMgQ29udGV4dG86IEV4cHJlc2nDs24gZGUgZ2VuZXMgZW4gaW5hbmljacOzbgoKVGhyb3VnaCB0aGUgcHJvY2VzcyBvZiBnZW5lIHJlZ3VsYXRpb24sIGEgY2VsbCBjYW4gY29udHJvbCB3aGljaCBnZW5lcyBhcmUgdHJhbnNjcmliZWQgZnJvbSBETkEgdG8gUk5BLSB3aGF0IHdlIGNhbGwgYmVpbmcg4oCcZXhwcmVzc2Vk4oCdLiAoSWYgYSBnZW5lIGlzIG5ldmVyIHR1cm5lZCBpbnRvIFJOQSwgaXQgbWF5IGFzIHdlbGwgbm90IGJlIHRoZXJlIGF0IGFsbCkuIFRoaXMgcHJvdmlkZXMgYSBzb3J0IG9mIOKAnGNlbGx1bGFyIHN3aXRjaGJvYXJk4oCdIHRoYXQgY2FuIGFjdGl2YXRlIHNvbWUgc3lzdGVtcyBhbmQgZGVhY3RpdmF0ZSBvdGhlcnMsIHdoaWNoIGNhbiBzcGVlZCB1cCBvciBzbG93IGRvd24gZ3Jvd3RoLCBzd2l0Y2ggd2hhdCBudXRyaWVudHMgYXJlIHRyYW5zcG9ydGVkIGludG8gb3Igb3V0IG9mIHRoZSBjZWxsLCBhbmQgcmVzcG9uZCB0byBvdGhlciBzdGltdWxpLiBBIGdlbmUgZXhwcmVzc2lvbiBtaWNyb2FycmF5IGxldHMgdXMgbWVhc3VyZSBob3cgbXVjaCBvZiBlYWNoIGdlbmUgaXMgZXhwcmVzc2VkIGluIGEgcGFydGljdWxhciBjb25kaXRpb24uIFdlIGNhbiB1c2UgdGhpcyB0byBmaWd1cmUgb3V0IHRoZSBmdW5jdGlvbiBvZiBhIHNwZWNpZmljIGdlbmUgKGJhc2VkIG9uIHdoZW4gaXQgdHVybnMgb24gYW5kIG9mZiksIG9yIHRvIGdldCBhbiBvdmVyYWxsIHBpY3R1cmUgb2YgdGhlIGNlbGzigJlzIGFjdGl2aXR5LgoKQnJhdWVyIDIwMDggdXNlZCBtaWNyb2FycmF5cyB0byB0ZXN0IHRoZSBlZmZlY3Qgb2Ygc3RhcnZhdGlvbiBhbmQgZ3Jvd3RoIHJhdGUgb24gYmFrZXLigJlzIHllYXN0IChTLiBjZXJldmlzaWFlLCBhIHBvcHVsYXIgbW9kZWwgb3JnYW5pc20gZm9yIHN0dWR5aW5nIG1vbGVjdWxhciBnZW5vbWljcyBiZWNhdXNlIG9mIGl0cyBzaW1wbGljaXR5KTEuIEJhc2ljYWxseSwgaWYgeW91IGdpdmUgeWVhc3QgcGxlbnR5IG9mIG51dHJpZW50cyAoYSByaWNoIG1lZGlhKSwgZXhjZXB0IHRoYXQgeW91IHNoYXJwbHkgcmVzdHJpY3QgaXRzIHN1cHBseSBvZiBvbmUgbnV0cmllbnQsIHlvdSBjYW4gY29udHJvbCB0aGUgZ3Jvd3RoIHJhdGUgdG8gd2hhdGV2ZXIgbGV2ZWwgeW91IGRlc2lyZSAod2UgZG8gdGhpcyB3aXRoIGEgdG9vbCBjYWxsZWQgYSBjaGVtb3N0YXQpLiBGb3IgZXhhbXBsZSwgeW91IGNvdWxkIGxpbWl0IHRoZSB5ZWFzdOKAmXMgc3VwcGx5IG9mIGdsdWNvc2UgKHN1Z2FyLCB3aGljaCB0aGUgY2VsbCBtZXRhYm9saXplcyB0byBnZXQgZW5lcmd5IGFuZCBjYXJib24pLCBvZiBsZXVjaW5lIChhbiBlc3NlbnRpYWwgYW1pbm8gYWNpZCksIG9yIG9mIGFtbW9uaXVtIChhIHNvdXJjZSBvZiBuaXRyb2dlbikuCgrigJxTdGFydmluZ+KAnSB0aGUgeWVhc3Qgb2YgdGhlc2UgbnV0cmllbnRzIGxldHMgdXMgZmluZCBnZW5lcyB0aGF0OgoKLSBfX1JhaXNlIG9yIGxvd2VyIHRoZWlyIGFjdGl2aXR5IGluIHJlc3BvbnNlIHRvIGdyb3d0aCByYXRlLl9fIEdyb3d0aC1yYXRlIGRlcGVuZGVudCBleHByZXNzaW9uIHBhdHRlcm5zIGNhbiB0ZWxsIHVzIGEgbG90IGFib3V0IGNlbGwgY3ljbGUgY29udHJvbCwgYW5kIGhvdyB0aGUgY2VsbCByZXNwb25kcyB0byBzdHJlc3MuCi0gX19SZXNwb25kIGRpZmZlcmVudGx5IHdoZW4gZGlmZmVyZW50IG51dHJpZW50cyBhcmUgYmVpbmcgbGltaXRlZC5fXyBUaGVzZSBnZW5lcyBtYXkgYmUgaW52b2x2ZWQgaW4gdGhlIHRyYW5zcG9ydCBvciBtZXRhYm9saXNtIG9mIHRob3NlIG51dHJpZW50cy4KClNvdW5kcyBwcmV0dHkgY29vbCwgcmlnaHQ/IFNvIGxldOKAmXMgZ2V0IHN0YXJ0ZWQhCgojIyBJbXBvcnRhcgoKYGBge3IsIGV2YWw9RkFMU0UsIGVjaG89RkFMU0V9CmxpYnJhcnkocmVhZHIpCm9yaWdpbmFsX2RhdGEgPC0gcmVhZHI6OnJlYWRfZGVsaW0oImh0dHA6Ly92YXJpYW5jZWV4cGxhaW5lZC5vcmcvZmlsZXMvQnJhdWVyMjAwOF9EYXRhU2V0MS50ZHMiLCBkZWxpbSA9ICJcdCIpCnNhdmVSRFMob3JpZ2luYWxfZGF0YSwgZmlsZSA9ICJkYXRhLXJhdy90aWR5bWljcm8ucmRzIikKYGBgCgpgYGB7cn0Kb3JpZ2luYWxfZGF0YSA8LSByZWFkUkRTKCJkYXRhLXJhdy90aWR5bWljcm8ucmRzIikKYGBgCgojIyBSZWNvbm9jZXIgdmFyaWFibGVzCgpFYWNoIG9mIHRob3NlIGNvbHVtbnMgbGlrZSBgRzAuMDVgLCBgTjAuM2AgYW5kIHNvIG9uIHJlcHJlc2VudHMgZ2VuZSBleHByZXNzaW9uIHZhbHVlcyBmb3IgdGhhdCBzYW1wbGUsIGFzIG1lYXN1cmVkIGJ5IHRoZSBtaWNyb2FycmF5LiBUaGUgY29sdW1uIHRpdGxlcyBzaG93IHRoZSBjb25kaXRpb246IF9fYEcwLjA1YF9fLCBmb3IgaW5zdGFuY2UsIG1lYW5zIHRoZSBfX2xpbWl0aW5nIG51dHJpZW50IHdhcyBnbHVjb3NlX18gYW5kIHRoZSBfX2dyb3d0aCByYXRlIHdhcyAuMDVfXy4gQSBoaWdoZXIgdmFsdWUgbWVhbnMgdGhlIGdlbmUgd2FzIG1vcmUgZXhwcmVzc2VkIGluIHRoYXQgc2FtcGxlLCBsb3dlciBtZWFucyB0aGUgZ2VuZSB3YXMgbGVzcyBleHByZXNzZWQuIEluIHRvdGFsIHRoZSB5ZWFzdCB3YXMgZ3Jvd24gd2l0aCBfX3NpeCBsaW1pdGluZyBudXRyaWVudHNfXyBhbmQgX19zaXggZ3Jvd3RoIHJhdGVzX18sIHdoaWNoIG1ha2VzIDM2IHNhbXBsZXMsIGFuZCB0aGVyZWZvcmUgMzYgY29sdW1ucywgb2YgZ2VuZSBleHByZXNzaW9uIGRhdGEuCgpEaXNlw7FvIGV4cGVyaW1lbnRhbDoKCi0gNiBudXRyaWVudGVzIGxpbWl0YW50ZXMKLSA2IHRhc2FzIGRlIGNyZWNpbWllbnRvCgpDb2x1bW5hczoKCi0gMzYgY29uIHRhc2EgZGUgY3JlY2ltaWVudG8gcG9yIG51dHJpZW50ZSBsaW1pdGFkbwotIDQgY29uIGlkZW50aWZpY2Fkb3JlcwoKYGBge3J9CmRpbShvcmlnaW5hbF9kYXRhKQpoZWFkKG9yaWdpbmFsX2RhdGEpCmBgYApgYGB7cn0KZ2xpbXBzZShvcmlnaW5hbF9kYXRhKQpgYGAKCiMjIExpbXBpZXphIGRlIGRhdG9zCgpfX1VzYW5kbyBgZHBseXJgIHkgYHRpZHlyYF9fCgotIFVuYSBkYXRhICJsaW1waWEiIG8gX18iVGlkeSBkYXRhIl9fIG8gc2lndWUgbGFzIHNpZ3VpZW50ZXMgcmVnbGFzOgogICAgCiAgICAxLiBFYWNoIF9fdmFyaWFibGVfXyBmb3JtcyBhIGNvbHVtbi4KICAgIDIuIEVhY2ggX19vYnNlcnZhdGlvbl9fIGZvcm1zIGEgcm93LgogICAgMy4gRWFjaCB0eXBlIG9mIG9ic2VydmF0aW9uYWwgdW5pdCBmb3JtcyBhIHRhYmxlLgogICAgCi0gwr9RdcOpIG5vIGVzdMOhIGxpbXBpbyAoX18idW50aWR5Il9fKSBlbiBsYSBkYXRhPwogICAgCiAgICAxLiBfX0NvbHVtbiBoZWFkZXJzIGFyZSB2YWx1ZXMsIG5vdCB2YXJpYWJsZSBuYW1lcy5fXyAKICAgIE91ciBjb2x1bW4gbmFtZXMgY29udGFpbiB0aGUgdmFsdWVzIG9mIHR3byB2YXJpYWJsZXM6IG51dHJpZW50IChHLCBOLCBQLCBldGMpIGFuZCBncm93dGggcmF0ZSAoMC4wNS0wLjMpLiBGb3IgdGhpcyByZWFzb24sIHdlIGVuZCB1cCB3aXRoIF9fbm90IG9uZSBvYnNlcnZhdGlvbiBwZXIgcm93LCBidXQgMzYhX18gVGhpcyBpcyBhIHZlcnkgY29tbW9uIGlzc3VlIGluIGJpb2xvZ2ljYWwgZGF0YXNldHM6IHlvdSBvZnRlbiBzZWUgX19vbmUtcm93LXBlci1nZW5lIGFuZCBvbmUtY29sdW1uLXBlci1zYW1wbGVfXywgcmF0aGVyIHRoYW4gb25lLXJvdy1wZXItZ2VuZS1wZXItc2FtcGxlLgogICAgCiAgICAyLiBfX011bHRpcGxlIHZhcmlhYmxlcyBhcmUgc3RvcmVkIGluIG9uZSBjb2x1bW4uX18gCiAgICBUaGUgTkFNRSBjb2x1bW4gY29udGFpbnMgbG90cyBvZiBpbmZvcm1hdGlvbiwgX19zcGxpdCB1cCBieSBgfHxg4oCZc19fLiAKICAgIElmIHdlIGV4YW1pbmUgb25lIG9mIHRoZSBuYW1lcywgaXQgbG9va3MgbGlrZQogICAgCiAgICAgICAgPiBTRkIyIHx8IEVSIHRvIEdvbGdpIHRyYW5zcG9ydCB8fCBtb2xlY3VsYXIgZnVuY3Rpb24gdW5rbm93biB8fCBZTkwwNDlDIHx8IDEwODIxMjkKICAgIAogICAgICAgIHdoaWNoIHNlZW1zIHRvIGhhdmUgYm90aCBzb21lIHN5c3RlbWF0aWMgSURzIGFuZCBzb21lIGJpb2xvZ2ljYWwgaW5mb3JtYXRpb24gYWJvdXQgCiAgICB0aGUgZ2VuZS4gSWYgd2XigJlyZSBnb2luZyB0byB1c2UgdGhpcyBwcm9ncmFtbWF0aWNhbGx5LCB3ZSBuZWVkIHRvIHNwbGl0IHVwIAogICAgdGhlIGluZm9ybWF0aW9uIGludG8gbXVsdGlwbGUgY29sdW1ucy4KICAgIAotIF9fIlRoZSBtb3JlIGVmZm9ydCB5b3UgcHV0IHVwIGZyb250IGludG8gdGlkeWluZyB5b3VyIGRhdGEsIHRoZSBlYXNpZXIgaXQgd2lsbCBiZSAKdG8gZXhwbG9yZSBpbnRlcmFjdGl2ZWx5LiJfXwogICAgKyBTaW5jZSB0aGUgYW5hbHlzaXMgc3RlcHMgYXJlIHdoZXJlIHlvdeKAmWxsIGFjdHVhbGx5IGJlIGFuc3dlcmluZyBxdWVzdGlvbnMgCiAgICBhYm91dCB5b3VyIGRhdGEsIGl04oCZcyB3b3J0aCBwdXR0aW5nIHVwIHRoaXMgZWZmb3J0IQoKIyMjIE3Dumx0aXBsZXMgdmFyaWFibGVzIGVzdMOhbiBndWFyZGFkYXMgZW4gdW5hIG1pc21hIGNvbHVtbmEKYGBge3J9Cm9yaWdpbmFsX2RhdGEkTkFNRVsxOjJdCmBgYAoKLSBUaGUgZGV0YWlscyBvZiBlYWNoIG9mIHRoZXNlIGZpZWxkcyBpc27igJl0IGFubm90YXRlZCBpbiB0aGUgcGFwZXIsIGJ1dCB3ZSBjYW4gZmlndXJlIG91dCBtb3N0IG9mIGl0LiBJdCBjb250YWluczoKICAgIAogICAgLSBfX0dlbmUgbmFtZV9fIGUuZy4gU0ZCMi4gTm90ZSB0aGF0IG5vdCBhbGwgZ2VuZXMgaGF2ZSBhIG5hbWUuCiAgICAtIF9fQmlvbG9naWNhbCBwcm9jZXNzX18gZS5nLiDigJxwcm90ZW9seXNpcyBhbmQgcGVwdGlkb2x5c2lz4oCdCiAgICAtIF9fTW9sZWN1bGFyIGZ1bmN0aW9uX18gZS5nLiDigJxtZXRhbGxvZW5kb3BlcHRpZGFzZSBhY3Rpdml0eeKAnQogICAgLSBfX1N5c3RlbWF0aWMgSURfXyBlLmcuIFlOTDA0OUMuIFVubGlrZSBhIGdlbmUgbmFtZSwgZXZlcnkgZ2VuZSBpbiB0aGlzIGRhdGFzZXQgaGFzIGEgc3lzdGVtYXRpYyBJRC4KICAgIC0gX19Bbm90aGVyIElEIG51bWJlcl9fIGUuZy4gMTA4MjEyOS4gSSBkb27igJl0IGtub3cgd2hhdCB0aGlzIG51bWJlciBtZWFucywgCmFuZCBpdOKAmXMgbm90IGFubm90YXRlZCBpbiB0aGUgcGFwZXIuIE9oLCB3ZWxsLgoKLSBIYXZpbmcgYWxsIGdpdmUgb2YgdGhlc2UgaW4gdGhlIHNhbWUgY29sdW1uIGlzIHZlcnkgaW5jb252ZW5pZW50LiBGb3IgZXhhbXBsZSwgaWYgSSBoYXZlIGFub3RoZXIgZGF0YXNldCB3aXRoIGluZm9ybWF0aW9uIGFib3V0IGVhY2ggZ2VuZSwgSSBjYW7igJl0IG1lcmdlIHRoZSB0d28uIEx1Y2tpbHksIHRoZSBgdGlkeXJgIHBhY2thZ2UgcHJvdmlkZXMgdGhlIF9fYHNlcGFyYXRlYCBmdW5jdGlvbl9fIGZvciBleGFjdGx5IHRoaXMgY2FzZS4KCmBgYHtyfQpsaWJyYXJ5KGRwbHlyKQpsaWJyYXJ5KHRpZHlyKQoKY2xlYW5lZF9kYXRhIDwtIG9yaWdpbmFsX2RhdGEgJT4lCiAgc2VwYXJhdGUoTkFNRSwgYygibmFtZSIsICJCUCIsICJNRiIsICJzeXN0ZW1hdGljX25hbWUiLCAibnVtYmVyIiksIHNlcCA9ICJcXHxcXHwiKQoKaGVhZChjbGVhbmVkX2RhdGEpCmBgYAoKVHdvIG1vcmUgdGhpbmdzLiBGaXJzdCwgd2hlbiB3ZSBfX3NwbGl0IGJ5IHx8X18sIHdlIGVuZGVkIHVwIHdpdGggX193aGl0ZXNwYWNlIGF0IHRoZSBzdGFydCBhbmQgZW5kIG9mIHNvbWUgb2YgdGhlIGNvbHVtbnNfXywgd2hpY2ggaXMgaW5jb252ZW5pZW50OgpgYGB7cn0KaGVhZChjbGVhbmVkX2RhdGEkQlApCmBgYAoKV2XigJlsbCBzb2x2ZSB0aGF0IHdpdGggX19kcGx5cuKAmXMgYG11dGF0ZV9lYWNoYF9fLCBhbG9uZyB3aXRoIHRoZSBidWlsdC1pbiBgdHJpbXdzYCAo4oCcdHJpbSB3aGl0ZXNwYWNl4oCdKSBmdW5jdGlvbi4KCmBgYHtyfQpjbGVhbmVkX2RhdGEgPC0gb3JpZ2luYWxfZGF0YSAlPiUKICBzZXBhcmF0ZShOQU1FLCBjKCJuYW1lIiwgIkJQIiwgIk1GIiwgInN5c3RlbWF0aWNfbmFtZSIsICJudW1iZXIiKSwgc2VwID0gIlxcfFxcfCIpICU+JQogIG11dGF0ZV9hdCh2YXJzKG5hbWU6c3lzdGVtYXRpY19uYW1lKSwgZnVucyh0cmltd3MpKQoKaGVhZChjbGVhbmVkX2RhdGEkQlApCmBgYAoKRmluYWxseSwgd2UgZG9u4oCZdCBldmVuIGtub3cgd2hhdCB0aGUgbnVtYmVyIGNvbHVtbiByZXByZXNlbnRzIChpZiB5b3UgY2FuIGZpZ3VyZSBpdCBvdXQsIGxldCBtZSBrbm93ISkgQW5kIHdoaWxlIHdl4oCZcmUgYXQgaXQsIF9fd2XigJlyZSBub3QgZ29pbmcgdG8gdXNlIHRoZSBHSUQsIFlPUkYgb3IgR1dFSUdIVCBjb2x1bW5zX18gaW4gdGhpcyBhbmFseXNpcyBlaXRoZXIuIFdlIG1heSBhcyB3ZWxsIGRyb3AgdGhlbSwgd2hpY2ggd2UgY2FuIGRvIHdpdGggX19kcGx5cuKAmXMgYHNlbGVjdGBfXy4KCmBgYHtyfQpjbGVhbmVkX2RhdGEgPC0gb3JpZ2luYWxfZGF0YSAlPiUKICBzZXBhcmF0ZShOQU1FLCBjKCJuYW1lIiwgIkJQIiwgIk1GIiwgInN5c3RlbWF0aWNfbmFtZSIsICJudW1iZXIiKSwgc2VwID0gIlxcfFxcfCIpICU+JQogIG11dGF0ZV9hdCh2YXJzKG5hbWU6c3lzdGVtYXRpY19uYW1lKSwgZnVucyh0cmltd3MpKSAlPiUKICBzZWxlY3QoLW51bWJlciwgLUdJRCwgLVlPUkYsIC1HV0VJR0hUKSAKCmhlYWQoY2xlYW5lZF9kYXRhKQpgYGAKCiMjIyBMb3MgdGl0dWxvcyBkZSBjb2x1bW5hcyBzb24gb2JzZXJ2YWNpb25lcywgbm8gdmFyaWFibGVzCgotIExldOKAmXMgdGFrZSBhIGNsb3NlciBsb29rIGF0IGFsbCB0aG9zZSBfX2NvbHVtbiBoZWFkZXJzX18gbGlrZSBgRzAuMDVgLCBgTjAuMmAgYW5kIGBQMC4xNWAuCiAgICAKICAgIC0gX19MaW1pdGluZyBudXRyaWVudF9fLiBUaGlzIGhhcyBzaXggcG9zc2libGUgdmFsdWVzOiBnbHVjb3NlIChHKSwgYW1tb25pdW0gKE4pLCBzdWxmYXRlIChTKSwgcGhvc3BoYXRlIChQKSwgdXJhY2lsIChVKSBvciBsZXVjaW5lIChMKS4KICAgIC0gX19Hcm93dGggcmF0ZV9fOiBBIG51bWJlciwgcmFuZ2luZyBmcm9tIC4wNSB0byAuMy4gLjA1IG1lYW5zIHNsb3cgZ3Jvd3RoICh0aGUgeWVhc3Qgd2VyZSBiZWluZyBzdGFydmVkIGhhcmQgb2YgdGhhdCBudXRyaWVudCkgd2hpbGUgLjMgbWVhbnMgZmFzdCBncm93dGguIChUZWNobmljYWxseSwgdGhpcyB2YWx1ZSBtZWFzdXJlcyB0aGUgZGlsdXRpb24gcmF0ZSBmcm9tIHRoZSBjaGVtb3N0YXQpLgogICAgLSBfX0V4cHJlc3Npb24gbGV2ZWxfXy4gVGhlc2UgYXJlIHRoZSB2YWx1ZXMgY3VycmVudGx5IHN0b3JlZCBpbiB0aG9zZSBjb2x1bW5zLCBhcyBtZWFzdXJlZCBieSB0aGUgbWljcm9hcnJheS4gKE5vdGUgdGhhdCB0aGUgcGFwZXIgYWxyZWFkeSBkaWQgc29tZSBfX3ByZS1wcm9jZXNzaW5nX18gYW5kIF9fbm9ybWFsaXphdGlvbl9fIG9uIHRoZXNlIHZhbHVlcywgd2hpY2ggd2XigJlyZSBpZ25vcmluZyBoZXJlKS4KCi0gVGhlIHJ1bGVzIG9mIHRpZHkgZGF0YSBzcGVjaWZ5IHRoYXQgX19lYWNoIHZhcmlhYmxlIGZvcm1zIG9uZSBjb2x1bW5fXywgYW5kIHRoaXMgaXMgbm90IGV2ZW4gcmVtb3RlbHkgdGhlIGNhc2UtIHdlIGhhdmUgMzYgY29sdW1ucyB3aGVuIHdlIHNob3VsZCBoYXZlIDMuIFRoYXQgbWVhbnMgb3VyIGRhdGEgaXMgdHJhcHBlZCBpbiBvdXIgY29sdW1uIG5hbWVzLiBfX0lmIHlvdSBkb27igJl0IHNlZSB3aHkgdGhpcyBpcyBhIHByb2JsZW0sIGNvbnNpZGVyX186IGhvdyB3b3VsZCB5b3UgcHV0IGdyb3d0aCByYXRlIG9uIHRoZSB4LWF4aXMgb2YgYSBncmFwaD8gSG93IHdvdWxkIHlvdSBmaWx0ZXIgdG8gbG9vayBvbmx5IHRoZSBnbHVjb3NlIGNvbmRpdGlvbj8KCi0gTHVja2lseSwgdGhlIGB0aWR5cmAgcGFja2FnZSBoYXMgYSBzb2x1dGlvbiByZWFkeS4gVGhlIGRvY3VtZW50YXRpb24gZm9yIF9fYGdhdGhlcmBfXyBub3RlcyAoZW1waGFzaXMgbWluZSk6CiAgICAKICAgID4gR2F0aGVyIHRha2VzIG11bHRpcGxlIGNvbHVtbnMgYW5kIGNvbGxhcHNlcyBpbnRvIGtleS12YWx1ZSBwYWlycywgCiAgICBkdXBsaWNhdGluZyBhbGwgb3RoZXIgY29sdW1ucyBhcyBuZWVkZWQuIF9fWW91IHVzZSBnYXRoZXIoKSB3aGVuIHlvdSBub3RpY2UgdGhhdCB5b3UgaGF2ZSAKICAgIGNvbHVtbnMgdGhhdCBhcmUgbm90IHZhcmlhYmxlcy5fXwoKLSBIZXksIHRoYXTigJlzIHVzISBTbyBsZXTigJlzIGFwcGx5IGdhdGhlciBhcyBvdXIgbmV4dCBzdGVwOgoKKiBfX09KTzpfXyBFeHBlcmltZW50YXIgY29uIGVsIGNvbWFuZG8gYGRwbHlyOjpjb3VudChzYW1wbGUpYCBsdWVnbyBkZSBhcGxpY2FyIGBnYXRoZXIoKWAgcGFyYSBjb25maXJtYXIgbGEgX19mcmVjdWVuY2lhX18gZGUgZGF0b3MgcG9yIG9ic2VydmFjacOzbi4KCmBgYHtyfQpjbGVhbmVkX2RhdGEgPC0gb3JpZ2luYWxfZGF0YSAlPiUKICBzZXBhcmF0ZShOQU1FLCBjKCJuYW1lIiwgIkJQIiwgIk1GIiwgInN5c3RlbWF0aWNfbmFtZSIsICJudW1iZXIiKSwgc2VwID0gIlxcfFxcfCIpICU+JQogIG11dGF0ZV9hdCh2YXJzKG5hbWU6c3lzdGVtYXRpY19uYW1lKSwgZnVucyh0cmltd3MpKSAlPiUKICBzZWxlY3QoLW51bWJlciwgLUdJRCwgLVlPUkYsIC1HV0VJR0hUKSAlPiUgCiAgZ2F0aGVyKHNhbXBsZSwgZXhwcmVzc2lvbiwgRzAuMDU6VTAuMykKCiNjbGVhbmVkX2RhdGEKY2xlYW5lZF9kYXRhICU+JSBkcGx5cjo6Y291bnQoc2FtcGxlKQpgYGAKCi0gTm90aWNlIHRoYXQgdGhlIGRhdGFzZXQgbm8gbG9uZ2VyIGNvbnNpc3RzIG9mIF9fb25lLXJvdy1wZXItZ2VuZV9fOiBpdOKAmXMgX19vbmUtcm93LXBlci1nZW5lLXBlci1zYW1wbGVfXy4gVGhpcyBoYXMgcHJldmlvdXNseSBiZWVuIGNhbGxlZCDigJxtZWx0aW5n4oCdIGEgZGF0YXNldCwgb3IgdHVybmluZyBpdCBpbnRvIOKAnGxvbmfigJ0gZm9ybWF0LiBCdXQgSSBsaWtlIHRoZSB0ZXJtIF9f4oCcZ2F0aGVy4oCdX186IGl0IHNob3dzIHRoYXQgd2XigJlyZSB0YWtpbmcgdGhlc2UgMzYgY29sdW1ucyBhbmQgcHVsbGluZyB0aGVtIHRvZ2V0aGVyLgoKLSBPbmUgbGFzdCBwcm9ibGVtLiBUaGF0IF9fc2FtcGxlIGNvbHVtbiByZWFsbHkgY29udGFpbnMgdHdvIHZhcmlhYmxlcywgbnV0cmllbnQgYW5kIHJhdGVfXy4gV2UgYWxyZWFkeSBsZWFybmVkIHdoYXQgdG8gZG8gd2hlbiB3ZSBoYXZlIHR3byB2YXJpYWJsZXMgaW4gb25lIGNvbHVtbjogdXNlIGBzZXBhcmF0ZWA6CgpgYGB7cn0KbGlicmFyeShkcGx5cikKbGlicmFyeSh0aWR5cikKCmNsZWFuZWRfZGF0YSA8LSBvcmlnaW5hbF9kYXRhICU+JQogIHNlcGFyYXRlKE5BTUUsIGMoIm5hbWUiLCAiQlAiLCAiTUYiLCAic3lzdGVtYXRpY19uYW1lIiwgIm51bWJlciIpLCBzZXAgPSAiXFx8XFx8IikgJT4lCiAgbXV0YXRlX2F0KHZhcnMobmFtZTpzeXN0ZW1hdGljX25hbWUpLCBmdW5zKHRyaW13cykpICU+JQogIHNlbGVjdCgtbnVtYmVyLCAtR0lELCAtWU9SRiwgLUdXRUlHSFQpICU+JQogIGdhdGhlcihzYW1wbGUsIGV4cHJlc3Npb24sIEcwLjA1OlUwLjMpICU+JSAjZHBseXI6OmNvdW50KHNhbXBsZSkKICBzZXBhcmF0ZShzYW1wbGUsIGMoIm51dHJpZW50IiwgInJhdGUiKSwgc2VwID0gMSwgY29udmVydCA9IFRSVUUpIAoKaGVhZChjbGVhbmVkX2RhdGEpCmBgYAoKLSBUaGlzIHRpbWUsIGluc3RlYWQgb2YgdGVsbGluZyBzZXBhcmF0ZSB0byBzcGxpdCB0aGUgc3RyaW5ncyBiYXNlZCBvbiBhIHBhcnRpY3VsYXIgZGVsaW1pdGVyLCB3ZSB0b2xkIGl0IHRvIF9fc2VwYXJhdGUgaXQgYWZ0ZXIgdGhlIGZpcnN0IGNoYXJhY3Rlcl9fICh0aGF0IGlzLCBhZnRlciBHL1AvUy9OL0wvVSkuIFdlIGFsc28gdG9sZCBpdCBgY29udmVydCA9IFRSVUVgIHRvIHRlbGwgaXQgdGhhdCBpdCBzaG91bGQgbm90aWNlIHRoZSAwLjA1LzAuMS9ldGMgdmFsdWUgaXMgYSBudW1iZXIgYW5kIGNvbnZlcnQgaXQuCgotIF9fVGFrZSBhIGxvb2sgYXQgdGhvc2Ugc2l4IGxpbmVzIG9mIGNvZGUsIGEgbWluaS1zb25uZXQgb2YgZGF0YSBjbGVhbmluZ19fLiBEb2VzbuKAmXQgaXQgcmVhZCBfX2xlc3MgbGlrZSBjb2RlIGFuZCBtb3JlIGxpa2UgaW5zdHJ1Y3Rpb25zP19fICjigJxGaXJzdCB3ZSBzZXBhcmF0ZWQgdGhlIE5BTUUgY29sdW1uIGludG8gaXRzIGZpdmUgcGFydHMsIGFuZCB0cmltbWVkIGVhY2guIFdlIHNlbGVjdGVkIG91dCBjb2x1bW5zIHdlIGRpZG7igJl0IG5lZWTigKbigJ0pIFRoYXTigJlzIHRoZSBiZWF1dHkgb2YgdGhlIGAlPiVgIG9wZXJhdG9yIGFuZCB0aGUgX19kcGx5ci90aWR5ciB2ZXJic19fLgoKIyMgVmlzdWFsaXphY2lvbmVzIGNvbiBnZ3Bsb3QKCl9fwr9Qb3IgcXXDqSBsaW1waWFyIGxhIGRhdGE/X18KCi0gU28gd2Ugd2VudCB0aHJvdWdoIHRoaXMgZWZmb3J0IHRvIGdldCB0aGlzIGRhdGFzZXQgaW50byB0aGlzIHN0cnVjdHVyZSwgYW5kIHlvdeKAmXJlIHByb2JhYmx5IHdvbmRlcmluZyB3aHkuIEluIHBhcnRpY3VsYXIsIHdoeSBkaWQgd2UgaGF2ZSB0byBib3RoZXIgZ2F0aGVyaW5nIHRob3NlIGV4cHJlc3Npb24gY29sdW1ucyBpbnRvIG9uZS1yb3ctcGVyLWdlbmUtcGVyLXNhbXBsZT8KCi0gV2VsbCwgc3VwcG9zZSB3ZSBoYXZlIGEgc2luZ2xlIHllYXN0IGdlbmUgd2XigJlyZSBpbnRlcmVzdGVkIGluLiBMZXTigJlzIHNheSBMRVUxLCBhIGdlbmUgaW52b2x2ZWQgaW4gdGhlIGxldWNpbmUgc3ludGhlc2lzIHBhdGh3YXkuCmBgYHtyfQpjbGVhbmVkX2RhdGEgJT4lCiAgZmlsdGVyKG5hbWUgPT0gIkxFVTEiKSAlPiUKICBnbGltcHNlKCkKYGBgCgpXZSBub3cgaGF2ZSBfXzM2IGRhdGEgcG9pbnRzIChzaXggY29uZGl0aW9ucywgc2l4IGdyb3d0aCByYXRlcylfXywgYW5kIGZvciBlYWNoIHdlIGhhdmUgYSBsaW1pdGluZyBudXRyaWVudCwgYSBncm93dGggcmF0ZSwgYW5kIHRoZSByZXN1bHRpbmcgZXhwcmVzaW9uLiBXZeKAmXJlIHByb2JhYmx5IGludGVyZXN0ZWQgaW4gX19ob3cgYm90aCB0aGUgZ3Jvd3RoIHJhdGUgYW5kIHRoZSBsaW1pdGluZyBudXRyaWVudCBhZmZlY3QgdGhlIGdlbmXigJlzIGV4cHJlc3Npb25fXy4KCjM2IHBvaW50cyBpcyB0b28gbWFueSB0byBsb29rIGF0IG1hbnVhbGx5LiBTbyBpdOKAmXMgdGltZSB0byBicmluZyBpbiBzb21lIHZpc3VhbGl6YXRpb24uIFRvIGRvIHRoYXQsIHdlIHNpbXBseSBfX3BpcGUgdGhlIHJlc3VsdHMgb2Ygb3VyIGZpbHRlcmluZyByaWdodCBpbnRvIGBnZ3Bsb3QyYF9fOgoKYGBge3J9CmNsZWFuZWRfZGF0YSAlPiUgCiAgZmlsdGVyKG5hbWU9PSJMRVUxIikgJT4lIAogIGdncGxvdChhZXMocmF0ZSxleHByZXNzaW9uLGNvbG91cj1udXRyaWVudCkpKwogIGdlb21fbGluZSgpCmBgYAoKLSBXaGF0IGEgc3RvcnkgdGhpcyBzaW5nbGUgZ2VuZSB0ZWxscyEgVGhlIGdlbmXigJlzIGV4cHJlc3Npb24gaXMgZmFyIGhpZ2hlciAobW9yZSDigJx0dXJuZWQgb27igJ0pIHdoZW4gdGhlIGNlbGwgaXMgYmVpbmcgc3RhcnZlZCBvZiBsZXVjaW5lIHRoYW4gaW4gYW55IG90aGVyIGNvbmRpdGlvbiwgYmVjYXVzZSBpbiB0aGF0IGNhc2UgdGhlIGNlbGwgaGFzIHRvIHN5bnRoZXNpemUgaXRzIG93biBsZXVjaW5lLiBBbmQgYXMgdGhlIGFtb3VudCBvZiBsZXVjaW5lIGluIHRoZSBlbnZpcm9ubWVudCAodGhlIGdyb3d0aCByYXRlKSBpbmNyZWFzZXMsIHRoZSBjZWxsIGNhbiBmb2N1cyBsZXNzIG9uIGxldWNpbmUgcHJvZHVjdGlvbiwgYW5kIHRoZSBleHByZXNzaW9uIG9mIHRob3NlIGdlbmVzIGdvIGRvd24uIFdl4oCZdmUganVzdCBnb3R0ZW4gb25lIF9fc25hcHNob3Qgb2Ygb3VyIGdlbmXigJlzIHJlZ3VsYXRvcnkgbmV0d29ya19fLCBhbmQgaG93IGl0IHJlc3BvbmRzIHRvIGV4dGVybmFsIHN0aW11bGkuCgotIFdlIGRvbuKAmXQgaGF2ZSB0byBjaG9vc2Ugb25lIGdlbmUgdG8gdmlzdWFsaXplLSBMRVUxIGlzIGp1c3Qgb25lIGdlbmUgaW4gdGhlIGxldWNpbmUgYmlvc3ludGhlc2lzIHByb2Nlc3MuIFJlY2FsbCB0aGF0IHdlIGhhdmUgdGhhdCBpbmZvcm1hdGlvbiBpbiB0aGUgQlAgY29sdW1uLCBzbyB3ZSBjYW4gX19maWx0ZXIgZm9yIGFsbCBnZW5lcyBpbiB0aGF0IHByb2Nlc3MsIGFuZCB0aGVuIGZhY2V0IHRvIGNyZWF0ZSBzdWItcGxvdHMgZm9yIGVhY2hfXy4KYGBge3J9CmNsZWFuZWRfZGF0YSAlPiUKICBmaWx0ZXIoQlAgPT0gImxldWNpbmUgYmlvc3ludGhlc2lzIikgJT4lCiAgZ2dwbG90KGFlcyhyYXRlLCBleHByZXNzaW9uLCBjb2xvciA9IG51dHJpZW50KSkgKwogIGdlb21fbGluZSgpICsKICBmYWNldF93cmFwKH5uYW1lKQpgYGAKCi0gTEVVMSwgTEVVMiwgYW5kIExFVTQgYWxsIHNob3cgYSBzaW1pbGFyIHBhdHRlcm4sIHdoZXJlIHN0YXJ2YXRpb24gb2YgbGV1Y2luZSBjYXVzZXMgaGlnaGVyIGdlbmUgZXhwcmVzc2lvbi4gKEludGVyZXN0aW5nbHksIExFVTQgcmVzcG9uZHMgdG8gZ2x1Y29zZSBzdGFydmF0aW9uIGFzIHdlbGwuIEFueSBnZW5ldGljaXN0cyBoYXZlIHNvbWUgaWRlYSB3aHk/KS4gTEVVOSBpcyBhIGxpdHRsZSBtb3JlIGFtYmlndW91cyBidXQgaXMgc3RpbGwgaGlnaGVzdCBleHByZXNzZWQgdW5kZXIgbGV1Y2luZSBzdGFydmF0aW9uLiBXZSBhbHJlYWR5IGtub3cgd2hhdCB0aGVzZSBnZW5lcyBkbywgYnV0IHRoaXMgaGludHMgYXQgaG93IHdlIG1pZ2h0IGJlIGFibGUgdG8gZmluZCBvdGhlciBnZW5lcyB0aGF0IGFyZSBpbnZvbHZlZCBpbiBsZXVjaW5lIHN5bnRoZXNpcywgaW5jbHVkaW5nIG9uZXMgd2UgZG9u4oCZdCB5ZXQga25vdy4KCi0gTGV04oCZcyBwbGF5IHdpdGggZ3JhcGggYSBsaXR0bGUgbW9yZS4gVGhlc2UgdHJlbmRzIGxvb2sgdmFndWVseSBsaW5lYXIuIE1heWJlIHdlIHNob3VsZCBzaG93IGJlc3QgcG9pbnRzIHdpdGggYmVzdCBmaXQgbGluZXMgaW5zdGVhZDoKYGBge3J9CmNsZWFuZWRfZGF0YSAlPiUKICBmaWx0ZXIoQlAgPT0gImxldWNpbmUgYmlvc3ludGhlc2lzIikgJT4lCiAgZ2dwbG90KGFlcyhyYXRlLCBleHByZXNzaW9uLCBjb2xvciA9IG51dHJpZW50KSkgKwogIGdlb21fcG9pbnQoKSArCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGQUxTRSkgKwogIGZhY2V0X3dyYXAofm5hbWUpCmBgYAoKLSBfX1RoZSBvcHRpb25zIGZvciBleHBsb3JhdG9yeSBkYXRhIGFuYWx5c2lzIGFyZSBlbmRsZXNzX18uIFdlIGNvdWxkIGluc3RlYWQgbG9vayBhdCBzdWxmdXIgbWV0YWJvbGlzbToKCmBgYHtyfQpjbGVhbmVkX2RhdGEgJT4lCiAgZmlsdGVyKEJQID09ICJzdWxmdXIgbWV0YWJvbGlzbSIpICU+JQogIGdncGxvdChhZXMocmF0ZSwgZXhwcmVzc2lvbiwgY29sb3IgPSBudXRyaWVudCkpICsKICBnZW9tX3BvaW50KCkgKwogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRkFMU0UpICsKICBmYWNldF93cmFwKH5uYW1lICsgc3lzdGVtYXRpY19uYW1lLCBzY2FsZXMgPSAiZnJlZV95IikKYGBgCgooTm90aWNlIHRoYXQgd2UgaGF2ZSB0byBmYWNldCBieSB0aGUgX19zeXN0ZW1hdGljX25hbWVfXyBoZXJlLCBzaW5jZSBub3QgYWxsIGdlbmVzIGluIHRoaXMgcHJvY2VzcyBoYXZlIHRyYWRpdGlvbmFsIG5hbWVzKS4KCi0gSWYgeW91IHdlcmUgYW4gaW50ZXJlc3RlZCBtb2xlY3VsYXIgYmlvbG9naXN0LCB5b3UgY291bGQgZ28gYSBsb25nIHdheSBqdXN0IGJ5IGV4YW1pbmluZyB2YXJpb3VzIGJpb2xvZ2ljYWwgcHJvY2Vzc2VzLCBvciBsb29raW5nIGF0IGxpc3RzIG9mIGdlbmVzIHlvdSB3ZXJlIGludGVyZXN0ZWQgaW4uIFRoaXMgaXMgYSBncmVhdCB3YXkgdG8gZXhwbG9yZSBhIGRhdGFzZXQgaW50ZXJhY3RpdmVseSwgYW5kIHRvIGhpbnQgYXQgbWV0aG9kcyBvZiBmdXJ0aGVyIGFuYWx5c2lzLiBGb3IgZXhhbXBsZSwgd2Ugbm90aWNlIHRoYXQgd2UgY2FuIF9fZml0IGxpbmVhciBtb2RlbHMgdG8gbWFueSBvZiB0aGVzZSBleHByZXNzaW9uIHByb2ZpbGVzX18gdGhhdCB3aWxsIGNvbWUgaW4gaGFuZHkgaW4gdGhlIG5leHQgcGFydC4KCiMjIENvbmNsdXNpw7NuOiBkYXRhIGVzcGVjw61maWNhLCBoZXJyYW1pZW50YXMgZ2VuZXJhbGl6YWRhcwoKSSBlYXJsaWVyIHBvaW50ZWQgdG8gYSBsaXN0IG9mIGF2YWlsYWJsZSB3b3JrZmxvd3MgZnJvbSBgQmlvY29uZHVjdG9yYCwgd2hpY2ggdGVhY2ggd2F5cyB0byBhbmFseXplIG1hbnkga2luZHMgb2YgZ2Vub21pYyBkYXRhIHVzaW5nIHBhY2thZ2VzIHNwZWNpYWxpemVkIGZvciB0aG9zZSBwdXJwb3Nlcy4gVGhlc2Uga2luZHMgb2YgZ3VpZGVzIGFyZSBpbmNyZWRpYmx5IHZhbHVhYmxlLCBhbmQgYEJpb2NvbmR1Y3RvcmAgaGFzIGJ1aWx0IHVwIGFuIGV4Y2VsbGVudCBzZXQgb2YgdG9vbHMgZm9yIGFuYWx5emluZyBnZW5vbWljIGRhdGEsIG1hbnkgb2Ygd2hpY2ggY29tZSB3aXRoIHRoZWlyIG93biBkYXRhIHByb2Nlc3NpbmcgYW5kIHZpc3VhbGl6YXRpb24gbWV0aG9kcy4KCl9fU28gd2h5IGJvdGhlciB0ZWFjaGluZyBhIGRwbHlyL2dncGxvdDIgYXBwcm9hY2g/X18gQmVjYXVzZSB0aGVzZSB0b29scyBhcmUgdXNlZnVsIF9ldmVyeXdoZXJlXy4gQ29uc2lkZXIgdGhlIGRvemVuIGxpbmVzIG9mIGNvZGUgd2UgdXNlZCB0byBjbGVhbiBhbmQgdmlzdWFsaXplIG91ciBkYXRhOgpgYGB7ciwgZXZhbD1GQUxTRX0KbGlicmFyeShkcGx5cikKbGlicmFyeSh0aWR5cikKCmNsZWFuZWRfZGF0YSA8LSBvcmlnaW5hbF9kYXRhICU+JQogIHNlcGFyYXRlKE5BTUUsIGMoIm5hbWUiLCAiQlAiLCAiTUYiLCAic3lzdGVtYXRpY19uYW1lIiwgIm51bWJlciIpLCBzZXAgPSAiXFx8XFx8IikgJT4lCiAgbXV0YXRlX2F0KHZhcnMobmFtZTpzeXN0ZW1hdGljX25hbWUpLCBmdW5zKHRyaW13cykpICU+JQogIHNlbGVjdCgtbnVtYmVyLCAtR0lELCAtWU9SRiwgLUdXRUlHSFQpICU+JQogIGdhdGhlcihzYW1wbGUsIGV4cHJlc3Npb24sIEcwLjA1OlUwLjMpICU+JSAjZHBseXI6OmNvdW50KHNhbXBsZSkKICBzZXBhcmF0ZShzYW1wbGUsIGMoIm51dHJpZW50IiwgInJhdGUiKSwgc2VwID0gMSwgY29udmVydCA9IFRSVUUpCgpjbGVhbmVkX2RhdGEgJT4lCiAgZmlsdGVyKEJQID09ICJsZXVjaW5lIGJpb3N5bnRoZXNpcyIpICU+JQogIGdncGxvdChhZXMocmF0ZSwgZXhwcmVzc2lvbiwgY29sb3IgPSBudXRyaWVudCkpICsKICBnZW9tX3BvaW50KCkgKwogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRkFMU0UpICsKICBmYWNldF93cmFwKH5uYW1lKQpgYGAKCldpdGggdGhpcyBjb2RlIHdl4oCZcmUgYWJsZSB0byBfX2dvIGZyb20gcHVibGlzaGVkIGRhdGEgdG8gYSBuZXcgY29uY2x1c2lvbiBhYm91dCBsZXVjaW5lIGJpb3N5bnRoZXNpc19fIChvciBzdWxmYXRlIG1ldGFib2xpc20sIG9yIHdoYXRldmVyIGJpb2xvZ2ljYWwgcHJvY2VzcyBvciBnZW5lcyB3ZeKAmXJlIGludGVyZXN0ZWQgaW4pLiBCdXQgdGhlIGZ1bmN0aW9ucyB3ZSB1c2UgYXJlbuKAmXQgc3BlY2lmaWMgdG8gb3VyIGRhdGFzZXQgb3IgZGF0YSBmb3JtYXQ6IGluIGZhY3QsIHRoZXnigJlyZSBub3QgcmVsYXRlZCB0byBiaW9sb2d5IGF0IGFsbC4gX19JbnN0ZWFkLCB0aGVzZSB0b29scyBhcmUgYnVpbGRpbmcgYmxvY2tzLCBvciDigJxhdG9tcyzigJ0gaW4gYSBncmFtbWFyIG9mIGRhdGEgbWFuaXB1bGF0aW9uIGFuZCB2aXN1YWxpemF0aW9uIHRoYXQgYXBwbGllcyB0byBuZWFybHkgZXZlcnkga2luZCBvZiBkYXRhX18uCgpUaGlzIGlzbuKAmXQgbWVhbnQgdG8gZGlzcGFyYWdlIEJpb2NvbmR1Y3RvciBpbiBhbnkgd2F5OiBzY2llbnRpc3RzIHNob3VsZCB1c2Ugd2hhdGV2ZXIgdG9vbCBnZXRzIHRoZWlyIGpvYiBkb25lLiBCdXQgZWR1Y2F0b3JzIGNhbiBhbmQgc2hvdWxkIGZvY3VzIG9uIHRlYWNoaW5nIHRvb2xzIHRoYXQgY2FuIGJlIHVuaXZlcnNhbGx5IGFwcGxpZWQuIEluIHR1cm4sIHN0dWRlbnRzIGNhbiB0YWtlIHRoZXNlIHRvb2xzIGFuZCBidWlsZCBuZXcgcGFja2FnZXMsIGZvciBCaW9jb25kdWN0b3IgYW5kIGVsc2V3aGVyZSwgdGhhdCBhbmFseXplIG5vdmVsIGZvcm1zIG9mIGRhdGEuCgojIyBNw6FzIGxpbXBpZXphCgpUcmVzIHBhc29zIG3DoXMgZGUgbGltcGllemE6IAoKLSBGaXJzdCwgSSBfX3NwZWxsZWQgb3V0X18gdGhlIGZ1bGwgbmFtZXMgb2YgdGhlIG51dHJpZW50cy0g4oCcR2x1Y29zZeKAnSBpbnN0ZWFkIG9mIGp1c3Qg4oCcR+KAnSwgZm9yIGV4YW1wbGUuIAotIFNlY29uZCwgSSBfX2ZpbHRlcmVkIG91dF9fIG1pc3NpbmcgdmFsdWVzIGZyb20gdGhlIGV4cHJlc3Npb24gY29sdW1uLCBhcyB3ZWxsIGFzIGdlbmVzIHRoYXQgaGF2ZSBubyBzeXN0ZW1hdGljIElECi0gVGVyY2VybywgX19maWx0cmFyX18gbGFzIGdlbmVzIGNvbiBsZWN0dXJhcyBjb21wbGV0YXMgcGFyYSB0YXNhIGRlIGNyZWNpbWllbnRvIHBvciBudXRyaWVudGUgZGUgY3VsdGl2byBsaW1pdGFkby4KCiMjIyByZXRpcmFyIG9ic2VydmFjaW9uZXMgTkEncwoKX1/Cv0PDs21vIHJldmlzYXIgbGEgcHJlc2VuY2lhIGRlIE5BP19fCgpgYGB7cn0KIyB0aXBvIGRlIHZhcmlhYmxlOiBjaGFyYWN0ZXIgPGNocj4KbSA8LSBjbGVhbmVkX2RhdGEgJT4lIAogIGZpbHRlcihzeXN0ZW1hdGljX25hbWU9PSIiKSAlPiUgCiAgZHBseXI6OmNvdW50KCkKIyB0aXBvIGRlIHZhcmlhYmxlOiBkb3VibGUgPGRibD4KbiA8LSBjbGVhbmVkX2RhdGEgJT4lIAogIGZpbHRlcihpcy5uYShleHByZXNzaW9uKSkgJT4lIAogIGRwbHlyOjpjb3VudChleHByZXNzaW9uKQpgYGAKCi0gbGEgdmFyaWFibGUgYHN5c3RlbWF0aWNfbmFtZWAgcG9zZWUgX19gciBtJG5gIE5BJ3NfXwotIGxhIHZhcmlhYmxlIGBleHByZXNzaW9uYCBwb3NlZSBfX2ByIG4kbmAgTkEnc19fCgojIyMgY2FtYmlhciBlbCBub21icmUgYSBmYWN0b3JlcwoKX19FamVjdXRhciBsYSBsaW1waWV6YV9fCgoxLiBfX2NhbWJpYXJfXyBkZSBub21icmUgYSBsb3MgZmF0b3JlcyBkZSBsYSB2YXJpYWJsZSBgbnV0cmllbnRgCjIuIF9fcmV0aXJhcl9fIGEgbG9zIGVsZW1lbnRvcyBjb246IAogICAgLSBgc3lzdGVtYXRpY19uYW1lcyBgY29uIE5BJ3MgKHBvcnF1ZSBubyBoYWJyw61hIGPDs21vIGlkZW50aWZpY2FyIGFsIGdlbikKICAgIC0gdmFsb3JlcyBkZSBgZXhwcmVzc2lvbmAgY29uIE5BJ3MKCmBgYHtyfQpsaWJyYXJ5KGZvcmNhdHMpCmNsZWFuZWRfZGF0YSA8LSBvcmlnaW5hbF9kYXRhICU+JQogIHNlcGFyYXRlKE5BTUUsIGMoIm5hbWUiLCAiQlAiLCAiTUYiLCAic3lzdGVtYXRpY19uYW1lIiwgIm51bWJlciIpLCBzZXAgPSAiXFx8XFx8IikgJT4lCiAgbXV0YXRlX2F0KHZhcnMobmFtZTpzeXN0ZW1hdGljX25hbWUpLCBmdW5zKHRyaW13cykpICU+JQogIHNlbGVjdCgtbnVtYmVyLCAtR0lELCAtWU9SRiwgLUdXRUlHSFQpICU+JQogIGdhdGhlcihzYW1wbGUsIGV4cHJlc3Npb24sIEcwLjA1OlUwLjMpICU+JSAjZHBseXI6OmNvdW50KHNhbXBsZSkKICBzZXBhcmF0ZShzYW1wbGUsIGMoIm51dHJpZW50IiwgInJhdGUiKSwgc2VwID0gMSwgY29udmVydCA9IFRSVUUpICU+JSAKICBtdXRhdGUobnV0cmllbnQ9Zm9yY2F0czo6ZmN0X3JlY29kZShudXRyaWVudCwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIkdsdWNvc2UiID0gIkciLCAiTGV1Y2luZSIgPSAiTCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIlBob3NwaGF0ZSIgPSAiUCIsICJTdWxmYXRlIiA9ICJTIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiQW1tb25pYSIgPSAiTiIsICJVcmFjaWwiID0gIlUiKSkgJT4lIAogIGZpbHRlcighaXMubmEoZXhwcmVzc2lvbiksc3lzdGVtYXRpY19uYW1lIT0iIikgJT4lIAogIGdsaW1wc2UoKQpgYGAKCiMjIyByZXRpcmFyIGNvbWJpbmFjaW9uZXMgY29uIG9ic2VydmFjaW9uZXMgaW5jb21wbGV0YXMKCl9fwr9Ub2RvcyBsb3MgZ2VuZXMgdGllbmVuIHZhbG9yZXMgZGUgZXhwcmVzacOzbj9fXwoKYGBge3J9CnkgPC0gb3JpZ2luYWxfZGF0YSAlPiUKICBzZXBhcmF0ZShOQU1FLCBjKCJuYW1lIiwgIkJQIiwgIk1GIiwgInN5c3RlbWF0aWNfbmFtZSIsICJudW1iZXIiKSwgc2VwID0gIlxcfFxcfCIpICU+JQogIG11dGF0ZV9hdCh2YXJzKG5hbWU6c3lzdGVtYXRpY19uYW1lKSwgZnVucyh0cmltd3MpKSAlPiUKICBzZWxlY3QoLW51bWJlciwgLUdJRCwgLVlPUkYsIC1HV0VJR0hUKSAlPiUKICBnYXRoZXIoc2FtcGxlLCBleHByZXNzaW9uLCBHMC4wNTpVMC4zKSAlPiUgI2RwbHlyOjpjb3VudChzYW1wbGUpCiAgc2VwYXJhdGUoc2FtcGxlLCBjKCJudXRyaWVudCIsICJyYXRlIiksIHNlcCA9IDEsIGNvbnZlcnQgPSBUUlVFKSAlPiUgCiAgbXV0YXRlKG51dHJpZW50PWZvcmNhdHM6OmZjdF9yZWNvZGUobnV0cmllbnQsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJHbHVjb3NlIiA9ICJHIiwgIkxldWNpbmUiID0gIkwiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJQaG9zcGhhdGUiID0gIlAiLCAiU3VsZmF0ZSIgPSAiUyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIkFtbW9uaWEiID0gIk4iLCAiVXJhY2lsIiA9ICJVIikpICU+JSAKICBmaWx0ZXIoIWlzLm5hKGV4cHJlc3Npb24pLHN5c3RlbWF0aWNfbmFtZSE9IiIpICU+JSAKICAjZmlsdGVyKHN5c3RlbWF0aWNfbmFtZT09IlEwMTQwIiwgbnV0cmllbnQ9PSJQaG9zcGhhdGUiKSAjIGVqZW1wbG8gY29uIHNvbG8gZG9zIGxlY3R1cmFzIGRlIGdlbiB4IG51dHJpZW50ZQogIGdyb3VwX2J5KHN5c3RlbWF0aWNfbmFtZSwgbnV0cmllbnQpICU+JSBkcGx5cjo6Y291bnQoKSAlPiUgCiAgdW5ncm91cCgpICU+JSBjb3VudChuKSAlPiUgZmlsdGVyKG48NikgIyU+JSBzdW1tYXJpc2Uoc3VtKG5uKSkKYGBgCgotIEhheSBfX2ByIHN1bSh5JG5uKWAgZ2VuZXMgcG9yIG51dHJpZW50ZV9fIGNvbiBtZW5vcyBkZSA2IG9ic2VydmFjaW9uZXMgZGUgbGEgdmFyaWFibGUgYHJhdGVgIHBhcmEgbGFzIGNvbWJpbmFjaW9uZXMgZW50cmUgYHN5c3RlbWF0aWNfbmFtZWAgKGdlbikgeSBgbnV0cmllbnRgIChudXRyaWVudGUgbGltaXRhbnRlKS4KCiMjIyBmaW5hbAoKX19MaW1waWV6YSBmaW5hbF9fCgpgYGB7cn0KY2xlYW5lZF9kYXRhIDwtIG9yaWdpbmFsX2RhdGEgJT4lCiAgc2VwYXJhdGUoTkFNRSwgYygibmFtZSIsICJCUCIsICJNRiIsICJzeXN0ZW1hdGljX25hbWUiLCAibnVtYmVyIiksIHNlcCA9ICJcXHxcXHwiKSAlPiUKICBtdXRhdGVfYXQodmFycyhuYW1lOnN5c3RlbWF0aWNfbmFtZSksIGZ1bnModHJpbXdzKSkgJT4lCiAgc2VsZWN0KC1udW1iZXIsIC1HSUQsIC1ZT1JGLCAtR1dFSUdIVCkgJT4lCiAgZ2F0aGVyKHNhbXBsZSwgZXhwcmVzc2lvbiwgRzAuMDU6VTAuMykgJT4lICNkcGx5cjo6Y291bnQoc2FtcGxlKQogIHNlcGFyYXRlKHNhbXBsZSwgYygibnV0cmllbnQiLCAicmF0ZSIpLCBzZXAgPSAxLCBjb252ZXJ0ID0gVFJVRSkgJT4lIAogIG11dGF0ZShudXRyaWVudD1mb3JjYXRzOjpmY3RfcmVjb2RlKG51dHJpZW50LCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiR2x1Y29zZSIgPSAiRyIsICJMZXVjaW5lIiA9ICJMIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiUGhvc3BoYXRlIiA9ICJQIiwgIlN1bGZhdGUiID0gIlMiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJBbW1vbmlhIiA9ICJOIiwgIlVyYWNpbCIgPSAiVSIpKSAlPiUgCiAgZmlsdGVyKCFpcy5uYShleHByZXNzaW9uKSxzeXN0ZW1hdGljX25hbWUhPSIiKSAlPiUgCiAgZ3JvdXBfYnkoc3lzdGVtYXRpY19uYW1lLCBudXRyaWVudCkgJT4lCiAgZmlsdGVyKG4oKT09NikgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgZ2xpbXBzZSgpCmBgYAoKIyMgQXBsaWNhY2nDs24KClRpZHlpbmcgdGhlIGRhdGEgaW4gdGhpcyB3YXkgbGV0cyB1cyBtYWtlIGdyYXBocyBsaWtlIHRoaXMKCmBgYHtyLCBmaWcuaGVpZ2h0PTYsIGZpZy53aWR0aD02fQpjbGVhbmVkX2RhdGEgJT4lCiAgZmlsdGVyKEJQID09ICJsZXVjaW5lIGJpb3N5bnRoZXNpcyIpICU+JQogIGdncGxvdChhZXMocmF0ZSwgZXhwcmVzc2lvbiwgY29sb3IgPSBudXRyaWVudCkpICsKICBnZW9tX3BvaW50KCkgKwogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRkFMU0UpICsKICBmYWNldF93cmFwKH5uYW1lICsgc3lzdGVtYXRpY19uYW1lKQpgYGAKCkZvciBzdGFydGVycywgbGV04oCZcyB3cmFwIHRoaXMgdXNlZnVsIGdyYXBoIGludG8gYSBmdW5jdGlvbiwgc28gdGhhdCB3ZSBjYW4gbWFrZSBpdCBlYXNpbHkgaW4gdGhlIHJlc3Qgb2YgdGhlIHBvc3QuCgpgYGB7cn0KcGxvdF9leHByZXNzaW9uX2RhdGEgPC0gZnVuY3Rpb24oZXhwcmVzc2lvbl9kYXRhKSB7CiAgZ2dwbG90KGV4cHJlc3Npb25fZGF0YSwgYWVzKHJhdGUsIGV4cHJlc3Npb24sIGNvbG9yID0gbnV0cmllbnQpKSArCiAgICBnZW9tX3BvaW50KCkgKwogICAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGQUxTRSkgKwogICAgZmFjZXRfd3JhcCh+bmFtZSArIHN5c3RlbWF0aWNfbmFtZSwgc2NhbGVzID0gImZyZWVfeSIpCn0KYGBgCgpBdCB3aGljaCBwb2ludCB3ZSB3b3VsZCByZXdyaXRlIHRoZSBhYm92ZSBncmFwaCBsaWtlOgoKYGBge3IsIGV2YWw9RkFMU0V9CmNsZWFuZWRfZGF0YSAlPiUKICBmaWx0ZXIoQlAgPT0gImxldWNpbmUgYmlvc3ludGhlc2lzIikgJT4lCiAgcGxvdF9leHByZXNzaW9uX2RhdGEoKQpgYGAKClRoaXMgaXMgYSBncmVhdCB3YXkgdG8gdmlzdWFsaXplIGEgZmV3IGdlbmVzIGF0IGEgdGltZS4gQnV0IHRoZXJlIGFyZSBzbyBtYW55IGdlbmVzIGluIHRoZSBkYXRhc2V0LiBGb3IgZXhhbXBsZSwgbGV04oCZcyBpbnN0ZWFkIGZpbHRlciBieSB0aGUgYmlvbG9naWNhbCBwcm9jZXNzIGNlbGwgd2FsbCBvcmdhbml6YXRpb24gYW5kIGJpb2dlbmVzaXMuCmBgYHtyLCBmaWcuaGVpZ2h0PTEwLCBmaWcud2lkdGg9MTB9CmNsZWFuZWRfZGF0YSAlPiUKICBmaWx0ZXIoQlAgPT0gImNlbGwgd2FsbCBvcmdhbml6YXRpb24gYW5kIGJpb2dlbmVzaXMiKSAlPiUKICBwbG90X2V4cHJlc3Npb25fZGF0YSgpCmBgYAoKT0ssIHRoYXTigJlzIDM2IGdlbmVzIGFuZCBpdOKAmXMgYWxyZWFkeSBnZXR0aW5nIGEgbG90IGhhcmRlciB0byB1bmRlcnN0YW5kIHRoZXNlIHBsb3RzLiBBbmQgd2UgaGF2ZSA1NTAwIGdlbmVzIGluIHRoaXMgZGF0YXNldDogbm8gd2F5IGNhbiB3ZSB2aXN1YWxseSBpbnRlcnByZXQgYWxsIHRob3NlIGdlbmVzIGF0IG9uY2UuIFRoaXMgaXMgd2hlcmUgd2UgaW50cm9kdWNlIG1vZGVsaW5nLgoKCgoKIyMgQ29tcHV0ZXIgZW52aXJvbm1lbnQKYGBge3J9CmRldnRvb2xzOjpzZXNzaW9uX2luZm8oKQpgYGAKCiMjIFJlZmVyZW5jZXM=