1. Introducción


En el tutorial anterior aprendimos a cargar datos en R. Sin embargo, es difícil que en una aplicación real tengamos los datos tal y como los necesitamos para hacer nuestro análisis. Habitualmente tendremos que trabajar los datos para arreglarlos. Este proceso, que en castellano podría llamarse “limpieza” o procesado de datos, se conoce en inglés como data munging or data wrangling.

En el curso vamos a trabajar/manejar los datos usando un conjunto de paquetes asociados asociados con el enfoque conocido como tidyverse. Como puedes ver en la imagen, ya hemos importado los datos y, antes de empezar a hacer el verdadero análisis, tenemos que pasar por 2 etapas más:

  • hacer nuestros datos tidy
  • arreglarlos para que sean útiles para nuestros propósitos



Se suele decir que el procesado/limpieza de los datos suele ocupar un 80% del tiempo de un análisis de datos. Quizás sea una cifra un poco exagerada, pero, en cualquier caso, es una tarea que ocupa tiempo y que puede llegar a ser tediosa y frustrante si no se dispone de las herramientas adecuadas. Incluso datos que parecen que ya están trabajados es bastante fácil que tengamos que trabajarlos para adaptarlos a nuestras necesidades.

Una secuencia "real" de tweets:

0930: How lucky I am to work on clean datasets curated by true professionals.
1330: Huh. Some inconsistencies here. No bigs. Ill just write up some quick and dirty regex to clean this up.

1720: I WILL BURN THIS HERETICAL DATA CENTER AND SCATTER ITS ASHES (traducción: me cago en todo lo que se menea)

En clase utilizamos datos reales, pero la verdad es que suelen ya estar casi limpios del todo. Estamos aprendiendo.

Classroom data are like teddy bears; real data are like a grizzly with salmon blood dripping out its mouth. —- [@JennyBryan]

Como dice Albert Y. Kim en estas transparencias los datos utilizados para aprender a manejar datos tienen que ser realistas pero sin llegar a ser intimidantes.


En este tutorial aprenderemos a limpiar y transformar datos en R. Priorizaremos la nueva forma de hacer las cosas en R (o workflow) conocido como tidyverse. En los últimos años se ha convertido, por varias razones, en el enfoque estándar; a pesar de ello, cada cierto tiempo vuelve a reabrirse el debate sobre cómo enseñar/aprender R y si es apropiado priorizar el tidyverse sobre R-base. Aquí tienes un hilo de twitter donde se debate sobre este tema.


Aquí tenéis un post sobre las diferencias entre las funciones de R-base y las del tidyverse para el procesado de datos, y aquí otro post de un nuevo convencido de las bondades de esta nueva forma de manipular datos en R. Como ejemplo:

Up until last year my R workflow was not dramatically different from when I started using R more than 10 years ago. Thanks to several R package authors, most notably Hadley Wickham, my workflow has changed for the better using dplyr, magrittr, tidyr and ggplot2. Given how much I’ve enjoyed the speed and clarity of the new workflow



Tidyverse

¿Qué es esto del tidyverse?


Con la palabra tidyverse se hace referencia a una “nueva” forma de afrontar el análisis de datos en R en la que se hace uso de un grupo de paquetes que trabajan en armonía porque comparten ciertos principios, como por ejemplo, la forma de estructurar los datos.

La mayoría de estos paquetes han sido desarrollados por (o al menos con la colaboración de) Hadley Wickham. Esta es la página web del tidyverse

No es necesario, pero si quieres conocer un poco mejor qué es el tidyverse, puedes hacerlo leyendo The tidy tools manifesto. Está cita es un buen referente de la filosofía o enfoque del tidyverse

Programs must be written for people to read, and only incidentally for machines to execute – Hal Abelson


Para continuar entendiendo qué es esto del tidyverse, citaré 2 de sus principios:

  • Los scripts deben ser “fácilmente” legibles por las personas

  • Resolver problemas complejos encadenando funciones simples con el operador %>%



The pipe (%>%)

Este operador ocupa un lugar fundamental en el tidyverse. Permite resolver un problema complejo no de una sola vez, sino encadenando llamadas a funciones que se van encadenando con el operador %>%. Este operador facilita mucho la lectura e interpretación del código, ya que se van encadenando operaciones sencillas para, poco a poco, conseguir transformaciones de datos complejas. El operador pipe se lo debemos a Stefan Bache en su pkg magrittr.


En palabras, lo que hace este operador es pasar el elemento que está a su izquierda como un argumento de la función que tiene a la derecha. Así al principio parece complicado.

Con expresiones el operador pipe hace:

f(object, argumentos de la función) ES EQUIVALENTE a object %>% f(argumentos de la función)


Se entiende mejor con ejemplos sencillos. Las siguientes dos instrucciones de R hacen exactamente lo mismo: permiten ver las 4 primeras filas del penguins dataset.

library(palmerpenguins)

head(penguins, n = 4)         #- forma habitual de llamar/usar la función head()

penguins %>% head(. , n = 4)  #- usando el operador pipe

La primera expresión es la manera habitual de usar/llamar a la función head(). La segunda expresión es la sintaxis, la forma que hay que usar, si trabajamos con el operador pipe.

Así, a primera vista, parece que el operador %>% no supone ninguna ventaja, sólo es una forma distinta de ejecutar o llamar a una función, y a primera vista parece complicar las cosas. Sí, eso es cierto, si solo usas una función no tendría mucho sentido usar %>%, pero cuando tienes que hacer una sucesión de cálculos, una sucesión de llamadas a funciones, facilita mucho la lectura del código y por tanto el análisis. Lo vemos enseguida.

Para entender un poco más el funcionamiento de %>%, has de ver que estas tres instrucciones son equivalentes.

head(penguins, n = 4)         #- forma habitual de llamar/usar la función head()

penguins %>% head(. , n = 4)  #- usando el operador pipe (con el punto actuando como placeholder)

penguins %>% head(n = 4)      #- usando el operador pipe (SIN el punto)

El punto de la segunda expresión señala, le dice a the pipe donde debe situarse el argumento de la izquierda dentro de la función; en nuestro ejemplo le dice a %>% que penguins debe situarse en el primer slot de head(). El punto . le está diciendo a %>% donde debe situarse penguins; es decir, el punto actúa, lo estamos usando, como un “placeholder”.

La tercera expresión también funciona porque si no usamos el punto (.), entonces, por defecto, el operador pipe situará penguins en el primer slot de la función, en nuestro caso situará a penguins en el primer slot de head().

La forma más habitual es no poner el .; es decir, la tercera expresión. La razón es simplemente que se ahorra tiempo al escribir, aunque la segunda expresión es mucho más explicita, más descriptiva, de lo que hace el operador pipe.


Para casi terminar de entender la sintaxis del operador pipe. Intentad ver si entendéis la siguiente instrucción:

4 %>% head(penguins, .)

Si no sabéis lo que hace, siempre podéis ejecutar la instrucción en la consola de RStudio.

Recuerda: Cuando usamos el operador pipe, tenemos obligatoriamente que usar el punto si queremos que el argumento de la izquierda se sitúe en un slot diferente del primer slot


Para acabar nuestro repaso a %>% mirad por qué no funciona la siguiente instrucción:

4 %>% head(penguins)

El operador pipe quiere llevar el 4 al primer slot de head() ya que si no ponemos el punto, ese es su comportamiento por defecto. Sin embargo, al ejecutar la expresión, el interprete de R nos devuelve un mensaje de error. ¿Por qué? Tendrás que mirar la ayuda de la función con help(head).


Aún no sabemos muy bien cuál es su utilidad, pero ya conocemos la sintaxis de %>%. Lo que hace que este operador sea tan útil es que las pipes se pueden encadenar.


El operador pipe podemos leerlo como entonces y permite encadenar sucesivas llamadas a funciones. Por ejemplo:

penguins %>% filter(sex == "female") %>%
             group_by(species) %>%
             summarise(peso_medio = mean(body_mass_g))


La anterior linea de código R hace:

  1. coge los datos de pingüinos y selecciona (o filtra) las filas/pinguinos cuyo valor de la variable sex es female; es decir, seleccionamos los pingüinos hembras, entonces (o después)
  2. agrupa los datos/pingüinos por la variable species, entonces
  3. calcula la media de body_mass_g

En conjunto, encadenando las 3 funciones hemos seleccionado las filas que pertenecen a pingüinos hembras, hemos agrupado las pingüino hembras en función de su especie (hay 3 especies de pingüinos) y calculado el peso medio de cada uno de las 3 especies de pingüinos; es decir, hemos calculado el peso medio de las pingüinos hembra en cada uno de las tres especies de pingüinos.


Con esta nueva sintaxis (que permite el operador pipe) ya no necesitamos anidar funciones, sino que las instrucciones van una después de otra. Es mucho más fácil de leer y de escribir. Esta idea de que es mucho más fácil escribir à la tidyverse no se llega a apreciar con los ejemplos que hemos hecho en esta sección, pero se hará evidente cuando empecemos a encadenar operaciones con dplyr. Como ejemplo este tweet.


No te va a resultar sencillo porque no sabes que es letters, ni paste0(), ni toupper(), pero intenta entender por ti mismo que hace la siguiente linea de código. ya sabes que siempre puedes ejecutarla y ver que hace, y mucho mejor si la ejecutas por trozos para ir viendo poco a poco qué hace:

letters %>% paste0( "-----" ,  .  ,  "!!!" ) %>% toupper


Un poco más acerca de the pipe (%>%) [OPCIONAL]

De forma más técnica. Aquí podéis ver el funcionamiento del operador pipe:

library("magrittr")

#-------- Rule 1
f(xx)     es equivalente a     xx %>% f

#-------- Rule 2
g(xx, n = 5)
xx %>% g(n = 5)

#-------- Rule 3
g(f(xx), n = 5)
xx %>% f %>% g(n = 5)

Se lee como "Take xx then do f then do g with n = 5".

#-------- Rule 4
f(y, x)
x %>% f(y, .)

#-------- Rule 5
f(y, z = x)
x %>% f(y, z = .)

#(!!!!)------------- BONUS: The input to the pipeline can itself be a placeholder!!
num_unique <- . %>% unique %>% length

num_unique(iris$Species)

iris$Species %>% num_unique

-----

num_unique es equivalente a : f <- function(.) length(unique(.))


Un buen recurso para aprender el uso de %>% son estas transparencias. Una exposición más detallada de la sintaxis y posibilidades del operador %>%, así como la de otros operadores como %T>% y %<>% puedes encontrarla aquí.


The “tee pipe” (%T>%) permite hacer cosas como esta:

#- !!!!!!
rnorm(200) %>% matrix(ncol = 2) %T>%
plot %>% # plot usually does not return anything.
colSums

The “tee pipe”, como el pipe original, pasa el argumento de la izquierda a la función de la derecha, PERO devuelve el propio valor original, no devuelve el resultado de la evaluación de la función. Como se señala aquí, este comportamiento es útil cuando se usa la función por sus side-effects; es decir, para imprimir o graficar. Yo la utilidad que le veo es hacer chequeos dentro de una secuencia de pipes, como por ejemplo hacen en este tweet


The “exposition pipe” (%$%) también del pkg magrittr. En este post nos explican su utilidad. Pero solo leedlo cuando ya seáis usuarios intermedios de R. Sirve para hacer accesibles las columnas de un dataframe a funciones que no admiten dataframes como la función cor(). De esta forma podemos integrar en nuestro pipeline funciones que no están preparadas para el tidyverse.

#- !!!
library(magrittr)
iris %>% mean(Sepal.Length)   #- no funciona
iris %$% mean(Sepal.Length)   #- con the exposition pipe sí funciona

iris %>% cor(Sepal.Length, Sepal.Width)  #- no funciona
iris %$% cor(Sepal.Length, Sepal.Width)

Si no usásemos este nueva pipe, tendríamos que hacer lo siguiente:

cor(iris$Sepal.Length, iris$Sepal.Width)


Muchas funciones de R-base no están preparadas para trabajar con el operador pipe. Son funciones que se escribieron antes de que se creara %>%. Se puede tratar de reescribir código en “R-base” usando %>% pero no tiene mucho sentido y no es muy agradable; sin embargo si se trabaja con el tidyverse, utilizar the pipe hace la sintaxis muy fluida. Como ejemplo de esto el siguiente chunk:

library(tidyverse)

x1 <- c(-5:5, NA)  #- es un vector

#- escribiendo à la R-base
mean(x1[x1>0], na.rm = TRUE)   #-  calcula la media de los valores positivos de x1
sum(x1[!is.na(x1)])            #-  calcula la suma de los valores de x1 que no son NA

#- ahora haremos lo mismo, seguimos usando R-base, pero con el operador pipe (!!!!)
x1 %>% .[.>0] %>% mean(., na.rm = TRUE)
x1 %>% .[!is.na(.)]  %>% sum

#- podríamos trabajar con data.frames usando the exposition pipe
df <- as.data.frame(x1)   #- tidyverse usa data.frames
df %$% x1 %>% .[.>0] %>% mean(., na.rm = TRUE)
df %$% x1 %>% .[!is.na(.)]  %>% sum

#- con tydiverse
df <- as.data.frame(x1)   #- tidyverse usa data.frames
df %>% filter(x1 > 0) %>% summarise(mean_x1 = mean(x1, na.rm = TRUE))
df %>% filter(!is.na(x1)) %>% summarise(suma_x1 = sum(x1))


Bien, ya sabemos como funciona “the pipe”. Volvamos al tidyverse y a aprender a manipular datos en R.


Principales pkgs del tidyverse

Como puede verse en su página web, los principales packages del tidyverse son:

  • readr: para importar datos

  • tidyr: para convertir los datos a tidy data

  • dplyr: para manipular datos

  • ggplot2: para hacer gráficos

  • tibble: data frames actualizados

  • forcast: para manipular factores

  • stringr: para manipular strings

  • purrr: para functional programming

  • y algunos más

Nos centraremos en los cuatro primeros paquetes, principalmente en dplyr y ggplot2.


Los principales paquetes del tidyverse se han “agrupado” en un metapaquete llamado tidyverse, así que cuando ejecutas library(tidyverse) en realidad estás cargando varios paquetes del tidyverse



2. Tidy data (tidyr)


If I had one thing to tell biologists learning bioinformatics, it would be write code for humans, write data for computers. —-— Vince Buffalo (@vsbuffalo)

Y si vamos a manejar datos con R y a la manera del tidyverse, como Jenny Bryan señala en su excelente tutorial sobre tidy data:

An important aspect of “writing data for computers” is to make your data TIDY. —- Jenny Bryan


Antes de comenzar a manipular los datos à la tidyverse, es conviene saber que se entiende por tidy data. La razón es que los paquetes del tidyverse trabajan mejor si los datos están en formato tidy. Es fácil!!


¿Qué son los tidy data?

Ahora lo veremos, pero enfatizar que si los datos son tidy (si siguen ese formato) será más fácil trabajar con ellos con el tidyverse, ya sea para manipularlos o para hacer gráficos.

De forma sencilla, tidy data son simplemente datos organizados de una determinada manera. Además es justo de la manera a la estamos familiarizados. De forma más precisa se puede leer aquí, o de forma más elaborada [aquí] :

Tidy datasets provide a standardized way to link the structure of a dataset (its physical layout) with its semantics (its meaning). —– Hadley Wickham


La mayoría de datos en Ciencias Sociales se ajustan a la categoría de datos tabulares; es decir, están organizados en filas y columnas. En R este tipo de datos se almacenan en dataframes (o tibbles). En esencia, un dataframe será tidy si cada columna es una variable y cada fila es una unidad de análisis (persona, país, región etc…); es decir, cada celda contiene el valor de una variable para una unidad de análisis.

A dataset is a collection of values. Every value belongs to a variable and an observation. A variable contains all values that measure the same underlying attribute (like height, temperature, duration) across units. An observation contains all values measured on the same unit (like a person, or a day, or a race) across attributes


No parece muy alejado de lo que estamos acostumbrados. Pero …. desarrollemos la idea un poco más.


Un ejemplo de datos (no tidy)

Supongamos que la variable (o atributo) a medir es el salario y la unidad de análisis las personas. Hemos recogido datos para 3 personas. Veámoslos:

data_1 <- data.frame(
            year  = c("2014", "2015", "2016"),
            Pedro = c(100, 500, 200),
            Carla = c(400, 600, 250),
            María = c(200, 700, 900)  )
data_1
Tabla 1: Salario de 3 personas:
year Pedro Carla María
2014 100 400 200
2015 500 600 700
2016 200 250 900

Entendemos perfectamente estos datos, visualmente son cómodos, pero ¿son tidy data? NO* porque los individuos (o unidades de análisis) están en columnas.


Un ejemplo de datos (tidy* pero wide)

Exactamente los mismos datos podrían estructurarse así:

data_2 <- data.frame(names = c("Pedro", "Carla", "María"),
                      W_2014 = c(100, 400, 200),
                      W_2015 = c(500, 600, 700),
                      W_2016 = c(200, 250, 900)   )

data_2
Tabla 2: Salario de 3 personas (wide format)
names W_2014 W_2015 W_2016
Pedro 100 500 200
Carla 400 600 250
María 200 700 900

También es un formato fácil de entender por nosotros, pero ¿son tidy? SI*, pero …

  • Es el formato al que estamos más acostumbrados (individuos o registros en filas y “variables” en columnas). ¿Realmente el W de 2014 es una variable?

  • En jerga del tidyverse este formato de datos es “wide” (o ancho)


Podemos trabajar tranquilamente con el anterior formato, PERO, si queremos sacar todo el provecho al tidyverse es mejor tener los datos en long format.


Un ejemplo de datos (tidy-tidy y long)

data_3 <- data.frame(
            names =rep(c("Pedro", "Carla", "María"), times = 3),
            year = rep(c("2014", "2015", "2016"), each = 3),
            salario = c(100, 400, 200, 500, 600, 700, 200, 250,900) )
data_3
Tabla 3: Salario de 3 personas (long format)
names year salario
Pedro 2014 100
Carla 2014 400
María 2014 200
Pedro 2015 500
Carla 2015 600
María 2015 700
Pedro 2016 200
Carla 2016 250
María 2016 900

Este formato, formato long, es más difícil de leer para nosotros, pero es más eficiente para los ordenadores. Y los datos los procesan los ordenadores!!

Generalmente, cuando estemos trabajando con los datos con el tidyverse convendrá que los datos estén en formato long, pero habrá veces, por ejemplo para mostrar tablas, tendremos que pasarlos a formato ancho. ¿Cómo podemos pasar un df de formato long a wide y al contrario? Lo más habitual es usar dos funciones del paquete tidyr. Veámoslo.


pivot_longer() y pivot_wider()

funciones para pasar de wide a long (& viceversa)


Ya hemos dicho que los packages del tidyverse trabajan mejor con tidy data en formato “long”. ¿Qué hacemos si tenemos un dataframe en formato wide? Pues pasarlo a long. Afortunadamente tenemos un pkg que hace muy sencillo pasar los datos de wide a long (y viceversa): tidyr. Concretamente usaremos las funciones pivot_longer() y pivot_wider()1

Aquí tienes el post oficial donde se anunciaba la llegada a CRAN de tidyr 1.0.0 y aquí y aquí un post detallado sobre ellas. La conclusión del autor del último de ellos es:

The new tidyr functions have intuitive syntax, are easy to use, and are more flexibile than the prior functions. Several of the new arguments and features are extremely useful, and will save lots of time on common tasks.

En este otro post tienes también una explicación detallada, de pivot_*() pero además incluye una serie de gifs que ejemplifican el paso de wide a long.


De wide a long format con pivot_longer()

La función pivot_longer() convierte dataframes de wide a long format

Hagámoslo:

library(tidyr)
data_wide <- data_2   #- data_2 está en formato ancho (wide)

#- la función pivot_longer() transforma los datos de formato ancho(wide) a formato largo(long)
data_long <- data_wide %>% pivot_longer(cols = 2:4, names_to = "periodo")

Si quisiéramos arreglar los valores de los periodos:

#(!!) stringr::str_replace encuentra el texto "W_" en la columna "periodo" y lo sustituye por ""
data_long <- data_long %>% mutate(periodo = str_replace(periodo, "W_", "" ))


De long a wide format con pivot_wider()

Pasar pasar de long a wide, tidyr tiene la función pivot_longer()

Hagámoslo:

#- `pivot_longer()` convierte un df de long a wide
data_wide2 <- data_long %>% pivot_wider(names_from = periodo, values_from = value)


separate() y unite()

funciones para separar y unir columnas


El pkg tidyr contiene otras 2 funciones: separate() y unite() que facilitan el separar y unir columnas. Veamos un ejemplo:

df <- data.frame( names = c("Pedro_Navaja", "Bob_Dylan", "Cid_Campeador"),
                  year  = c(1978, 1941, 1048) )
df
names year
Pedro_Navaja 1978
Bob_Dylan 1941
Cid_Campeador 1048

Separamos la primera columna:

df_a <- df %>% separate(names, c("Nombre", "Apellido"), sep = "_")
df_a
Nombre Apellido year
Pedro Navaja 1978
Bob Dylan 1941
Cid Campeador 1048

Si queremos volver a unirlos, tendríamos que:

df_b <- df_a %>% unite(Nombre_y_Apellido, Nombre:Apellido, sep = "&")
df_b
Nombre_y_Apellido year
Pedro&Navaja 1978
Bob&Dylan 1941
Cid&Campeador 1048


mas funciones de tidyr

Además, recuerda que el paquete tidyr tiene muchas más funciones que nos facilitan conseguir que nuestros datos sean tidy.




3. DPLYR

En R hay varios enfoques para manipular datos en R, pero el más habitual, de hecho se ha convertido en el estándar, es utilizar el tidyverse, concretamente el paquete dplyr.

dplyr es un paquete que permite manipular datos de forma intuitiva. Tiene 6-7 funciones o verbos principales. Cada uno de ellos hace “una sola cosa”, así que para realizar transformaciones complejas hay que ir concatenando instrucciones sencillas. Esto se hace con el operador pipe (%>%)



dplyr basics

Tras mucho pensar como estructurábamos este apartado del tutorial, al final me decanté por utilizar los materiales del curso STAT 545. ¿que quien ha se encarga del curso? Pues Jenny Bryan (always rocks!!). Puedes encontrarlos aquí.

dplyr tiene muchas funciones, pero las principales son 6-7, luego las veremos. Con ellas se pueden resolver la mayoría de problemas asociados a la manipulación de datos.

Cada función (o verbo) hace una sola cosa, pero concatenándolas con %>% permiten resolver cuestiones complejas.

Todas las funciones tienen una estructura o comportamiento similar:

  • el primer argumento siempre es un df. Esto es importante
  • los siguientes argumentos describen que hacer con los datos
  • el resultado es siempre un nuevo df. Esto es importante

Por ejemplo, filter(df, X1 >= 10) devuelve un df con las filas del df original que cumplen la condición de que la variable X1 es mayor o igual a 10

Podemos escribir la anterior instrucción de 3 formas. La más utilizada es la última:

df_new <- filter(df, X1 >= 10)

df_new <- df %>% filter(. , X1 >= 10)

df_new <- df %>% filter(X1 >= 10)



4. Principales funciones de dplyr

Hay 6-7 principales.

  • filter() : permite seleccionar filas (que cumplen una o varias condiciones)
  • arrange(): reordena las filas (arrange()).
  • rename() : cambia los nombres de las columnas (variables)
  • select() : selecciona columnas (variables)
  • mutate() : crea nuevas variables
  • summarise() : resume (colapsa) unos cuantos valores a uno sólo. Por ejemplo, calcula la media, moda, etc… de un conjunto de valores

Hay una séptima:

  • group_by() : permite agrupar filas en función de una o varias condiciones

Y después de dplyr 1.0.0, en mayo de 2020, añado 2 más:

  • across() y where(). Estas funciones son un poco diferentes, solo se usan en combinación de otro función/verbo. Son 2 funciones que en la jerga del tidyverse no son verbos sino adverbios. Lo vemos


Veámoslas una a una. Veremos sólo algunos ejemplos. Ya iremos practicando



filter()

Esta función (o verbo) se utiliza para seleccionar filas de un dataframe (df). Se seleccionan las filas que cumplen una determinada condición o criterio lógico. Por ejemplo:

#- vamos a trabajar con los datos del [pkg gapminder](https://github.com/jennybc/gapminder)
gapminder <- gapminder::gapminder


Seleccionamos las filas que cumplen determinados criterios:

#- Observaciones de España (country == "Spain")
aa <- gapminder %>% filter(country == "Spain")

#- filas con valores de "lifeExp" < 29
aa <- gapminder %>% filter(lifeExp < 29)

#- filas con valores de "lifeExp" entre [29, 32]
aa <- gapminder %>% filter(lifeExp >=  29 , lifeExp <= 32)
aa <- gapminder %>% filter(lifeExp >=  29 &  lifeExp <= 32)
aa <- gapminder %>% filter(between(lifeExp, 29, 32))

#- observaciones de paises de África con lifeExp > 32
aa <- gapminder %>% filter(lifeExp > 72 &  continent == "Africa")

#- observaciones de países de África o Asia con lifeExp > 32
aa <- gapminder %>% filter(lifeExp > 72 &  continent %in% c("Africa", "Asia") )
aa <- gapminder %>% filter(lifeExp > 72 & (continent == "Africa" | continent == "Asia") )  


La función filter() tiene muchas más posibilidades. Ya las iremos viendo. PERO si quieres ver un resumen de las posibilidades del paquete dplyr mira su CHEAT SHEET. 2. La versión antigua de la Cheat sheet contiene también las funciones de tidyr.


slice() también es muy útil para seleccionar filas

  • slice(): filtra filas por su posición (física en el df)


Como dijimos, slice() sirve para seleccionar filas por posición:

#- selecciona las observaciones de la décima a la quinceava
aa <- gapminder %>% slice(c(10:15))

#- selecciona las observaciones de la 12 a 13 Y de la 44 a 46, Y las 4 últimas
aa <- gapminder %>% slice( c(12:14, 44:46, n()-4:n()) ) #- AQUI hay un error, tenéis que arreglarlo.

#- Pista: igual os ayuda crear una columna con el índice de rows y repetir el cálculo
aa <- gapminder %>% mutate(index = 1:n())
aa <- gapminder %>% slice( c(12:14, 44:46, n()-4:n()) )


variantes de slice()

Hay varias variantes de slice(). Concretamente slice_max() slice_min(), slice_smpl(), slice_head() slice_tail(). Veremos algún ejemplo con las 3 primeras.

Si queremos seleccionar las filas que tienen el valor máximo (o mínimo) de una determinada variable, podemos usar slice_max() y slice_min()

#- selecciona las 3 filas con mayor valor de lifeExp
aa <- gapminder %>% slice_max(lifeExp, n = 3)
#- selecciona las 4 filas con MENOR valor de pop
aa <- gapminder %>% slice_min(pop, n = 4)

Para ver la potencialidad de una función tienes que ver su ayuda interna (presionando F1 o con help()). Por ejemplo slice_min() tiene otro argumento (prop) que nos permite calcular, por ejemplo, el 10% de observaciones/países con menor esperanza de vida.

#- observaciones en el primer decil en cuanto a esperanza de vida, 10% con menor esperanza de vida
aa <- gapminder %>% slice_min(lifeExp, prop = 0.1)
#- 1% de observaciones con mayor población. Imagino que estarán China e India
aa <- gapminder %>% slice_max(pop, prop = 0.01)

A veces se necesita obtener una muestra aleatoria de los datos. La función slice_sample() está diseñada para ayudarnos en esta tarea:

#- selecciona (aleatoriamente) 100 filas de los datos
aa <- gapminder %>% slice_sample(n = 100)
#- selecciona (aleatoriamente) un 5% de los datos
aa <- gapminder %>% slice_sample(prop = 0.05)



arrange()

Esta función (o verbo) se utiliza para reordenar las filas de un dataframe (df).

#- ordena las filas de MENOR a mayor según los valores de la v. lifeExp
aa <- gapminder %>% arrange(lifeExp)

#- ordena las filas de MAYOR a menor según los valores de la v. lifeExp
aa <- gapminder %>% arrange(desc(lifeExp))

#- ordenada las filas de MENOR a mayor según los valores de la v. lifeExp.
#- Si hay empates se resuelve con la variable "pop"
aa <- gapminder %>% arrange(lifeExp, pop) 



rename()

Esta función permite cambiar los nombres de las columnas

#- cambia los nombres de lifeExp y gdpPercap a life_exp y gdp_percap
gapminder %>% rename(life_exp = lifeExp,  gdp_percap = gdpPercap)

#-(!!) la función names() de R-base es muy útil. Tb setNames() y set_names()
aa <- gapminder
names(aa) <- names(aa) %>% toupper
names(aa) <- names(aa) %>% tolower
names(aa) <- c("var_01", "var_02", "var_03", "var_04", "var_05" , "var_06")
names(aa) <- paste0("Var_", 1:6)
names(aa) <- paste0("Lag_", formatC(1:6, width = 2, flag = "0")) 


rename_with() , una variante de rename()

Si tienes que hacer transformaciones más complejas, que requieran el uso de funciones o pautas, de los nombres de las variables puedes usar rename_with()

aa <- gapminder
rename_with(aa, toupper)
rename_with(aa, toupper, starts_with("Life") | contains("countr"))
rename_with(aa, ~ str_replace(.x, "e", "Ö"))  #- (!!!!)


La función rename() es útil pero, enseguida veremos que la siguiente función, select(), también permite renombrar las columnas, e incluso reordenar la posición de estas.



select()

Esta función (o verbo) sirve para seleccionar columnas de un df.

seleccionar variables por nombre

Seleccionamos las variables “year” y “lifeExp”:

#- Se lee como: “Take el df gapminder, then select the variables year and lifeExp”
aa <- gapminder %>% select(year, lifeExp)
aa <- gapminder %>% select(c(year, lifeExp))


quitar variables

Para eliminar una variable hay varias formas:

aa <- gapminder %>% select(-year)   #- la forma mas habitual

#- estas dos formas son mucho menos habituales
aa <- gapminder %>% select(!year)
aa <- gapminder %>% mutate(year = NULL)   #- aún no hemos visto mutate()

Para eliminar varias variables:

#- quitamos las variables: year y lifeExp
aa <- gapminder %>% select(-c(year, lifeExp))


seleccionar por posición

Seleccionamos las variables del df gapminder siguientes: de la primera a la tercera y también la quinta (mejor seleccionarlas por nombre!!)

#- seleccionamos las variables {1, 2, 3 y 5}
aa <- gapminder %>% select(1:3, 5)


quitar variables por posición

Seleccionamos todas las variables del df gapminder excepto las siguientes: de la primera a la tercer y la quinta (mejor seleccionarlas por nombre)

#- quitamos las variables {1, 2, 3 y 5}
aa <- gapminder %>% select(- c(1:3, 5))


select() con la función auxiliar where()

En el data.frame gapminder las 2 primeras variables (country y continent) son factores y las 4 siguientes son variable numéricas.

print(gapminder, n = 3)
#> # A tibble: 1,704 x 6
#>   country     continent  year lifeExp      pop gdpPercap
#>   <fct>       <fct>     <int>   <dbl>    <int>     <dbl>
#> 1 Afghanistan Asia       1952    28.8  8425333      779.
#> 2 Afghanistan Asia       1957    30.3  9240934      821.
#> 3 Afghanistan Asia       1962    32.0 10267083      853.
#> # … with 1,701 more rows

Imagina que queremos seleccionar sólo las variables que son numéricas. Podemos hacerlo por nombre o por posición pero mejor con select() y la función auxiliar where()3.

aa <- gapminder %>% select(is.numeric)        #- funciona, pero ...
aa <- gapminder %>% select(where(is.numeric)) #- es "preferible" esta segunda expresión

select() y where() son dos funciones, sí, pero en la jerga del tidyverse, select() es un verbo y where() es un adverbio, cualifica/cambia lo que hace select().

Si quisiéramos seleccionar las variables que no son numéricas haríamos:

aa <- gapminder %>% select(!is.numeric)
aa <- gapminder %>% select(!where(is.numeric))  #- es preferible esta segunda expresión


La función select() tiene muchas más posibilidades. ya las iremos viendo. PERO si quieres ver un resumen de las posibilidades del pkg dplyr mira su CHEAT SHEET.


renombrando y reordenando columnas con select()

Lo que sí vamos a ver son 2 posibilidades de select() que son muy útiles. Con select() podemos: renombrar y reordenar las columnas:

#- dejamos en aa solamente a las columnas "year" y "pop"; ADEMÁS, ahora, "pop" irá antes que "year"
aa <- gapminder %>% select(pop, year)


#- dejamos en aa solamente a las columnas "year" y "pop" y les cambiamos el nombre
aa <- gapminder %>% select(poblacion = pop, año = year)


Imagina que quieres que la última columna pase a ser la primera (manías!!). Podemos hacerlo con select y everything(). everything es una función auxiliar:

#- "gdpPercap" que es la última columna pasa a ser la primera
aa <- gapminder %>% select(gdpPercap, everything())

#(!!) otras 3 formas de hacer lo mismo: que la última columna pase a ser la primera
aa <- gapminder %>% select(ncol(df), everything())
aa <- gapminder %>% select(length(df), everything())
aa <- gapminder %>% select(last_col(), everything())  #- usamos the selection helper last_col()

En la última instrucción hemos usado dos funciones auxiliares de select(), dos “helper functions”. La lista completa de funciones auxiliares para select() puedes verla aquí.


relocate()

Desde dplyr 1.0.0, tenemos otra función para reordenar las variables de un data.frame: relocate(). Veámosla en acción:

aa <- gapminder %>% dplyr::relocate(country, .after = lifeExp)
aa <- gapminder %>% dplyr::relocate(country, .before = lifeExp)

Las opciones .after y .before también están disponibles en mutate(), la siguiente función que presentaremos.




mutate()

Esta función (o verbo) sirve para crear nuevas variables (columnas). Lógicamente, es muy útil en análisis de datos.

Creamos la variable: GDP = pop*gdpperCap

#- Creamos la variable: GDP = pop*gdpperCap
aa <- gapminder %>% mutate(GDP = pop*gdpPercap)

Por defecto, la nueva variable creada se situará al final del data frame, a no ser que usemos los argumentos .after y .before

aa <- gapminder %>% mutate(GDP = pop*gdpPercap, .after = country)
aa <- gapminder %>% mutate(GDP = pop*gdpPercap, .before = country)

mutate() también tiene un argumento(.keep) para controlar que variables permanecen en el data frame. Por ejemplo, en el chunk de abajo dejaremos solo las variables usadas.

aa <- gapminder %>% mutate(GDP = pop*gdpPercap, .keep = "used")



summarise()

Esta función (o verbo) sirve para RESUMIR (o “colapsar filas”). Coge una variable o grupo de valores como input y devuelve un solo valor; por ejemplo, haya la media aritmética (o el mínimo, o el máximo …) de una columna/variable.


Obtengamos determinados estadísticos de una variable. Para esto no nos hace falta dplyr pero conviene ir habituándose a su sintaxis.

#- retornará un único valor: la media global de la v. "lifeExp"
aa <- gapminder %>% summarise(media = mean(lifeExp))

#- retornará un único valor: el número de filas
aa <- gapminder %>% summarise(NN = n())
aa <- gapminder %>% count()                #- más adelante veremos la utilidad de count()


#- retornará un único valor: la desviación típica de la v. "lifeExp"
aa <- gapminder %>% summarise(desviacion_tipica = sd(lifeExp))

#- retornará un único valor: el máximo de la variable "pop"
aa <- gapminder %>% summarise(max(pop))

#- retornará 2 valores: la media y sd de la v. "lifeExp"
aa <- gapminder %>% summarise(mean(lifeExp), sd(lifeExp))

#- retornará 2 valores: las medias de "lifeExp" y "gdpPercap"
aa <- gapminder %>% summarise(mean(lifeExp), mean(gdpPercap))  


across() y where()

Antes de pasar a ver group_by(), vamos a utilizar las 2 nuevas funciones across() y where().

Muchas veces en un trabajo se han de calcular estadísticos de todas las variables del df. Esto se hace con summarise(), PERO utilizando también una función de ayuda (helper function): across()4. A veces también hay que usar otra helper function: where().

La sintaxis de across() es:

across(.cols = everything(), .fns = NULL, ..., .names = NULL); es decir,

across("columnas seleccionadas", "funciones o cálculos a realizar", "si quieres controlar los nombres")).

Si al seleccionar las columnas utilizas algún criterio lógico, como por ejemplo is.numeric(), entonces la sintaxis de across() es un poco diferente; concretamente será:

across( where("columnas seleccionadas"), "funciones o cálculos a realizar", "si quieres controlar los nombres"))

Los ejemplos ayudan a entenderlo:

#- media de cada una de las 6 variables. Devuelve 2 warnings porque las 2 primeras son textuales. No se puede calcular la media de continent y country
gapminder %>% summarise(across(everything(), mean) )

#- calculamos la media de tercera a la sexta variable
gapminder %>% summarise(across(3:6, mean) ) 

Lo que os decía de seleccionar columnas con un criterio lógico y usar where()

gapminder %>% summarise(across(where(is.numeric), mean))

#- con los nombres de los argumentos (más largo pero conviene verlo de vez en cuando)
gapminder %>% summarise(across(.cols = where(is.numeric), .fns = mean)) 

Vamos a calcular cosas un poco más complejas. Imagina que no sólo quieres calcular la media sino que también quieres calcular la desviación típica. Seguiremos usando summarise() y across(). Lo único nuevo es que como vamos a aplicar dos funciones (mean() y sd()) las tenemos que poner dentro de list(). Tiene sentido, es una lista de funciones a aplicar a las columnas que seleccionemos con across()

#- calculamos la media y desviación típica de las columnas 3 a 6.
gapminder %>% summarise(across(3:6, list(media = mean, desv = sd)))

#- lo mismo, pero explicitando los nombres de los argumentos
gapminder %>% summarise(across(.cols = 3:6, .fns = list(media = mean, desv = sd) ))

#- lo mismo otra vez, pero eligiendo el nombre de las variables que se van aa crear con .names
gapminder %>% summarise(across(3:6, list(media = mean, desv = sd), .names = "{fn}_{col}"))

(!!!)Imagina que quisiéramos presentar en una tabla los anteriores resultados; tendríamos que usar tidyr::pivot_longer(). Lo voy a hacer por trozos. Nos va a costar un poco:

aa <- gapminder %>% summarise(across(3:6, list(media = mean, desv = sd), .names = "{fn}_{col}"))

aa1 <- aa %>% pivot_longer(1:8, names_to = "names", values_to = "values")
aa2 <- aa1 %>% separate(names, into = c("operacion", "variable"), sep =  "_") %>%
               select(variable, everything())
aa3 <- aa2 %>% pivot_wider(1:3, names_from = operacion, values_from = values)

Lo practicaremos, y veremos métodos más sencillos para hacer tablas con estadísticos descriptivos, pero eso será en el tutorial dedicado a tablas; pero no os olvidéis de across() que luego tenemos que volver a ella.


group_by()

Con esta función ya empezaremos a ver la potencia de dplyr. En análisis de datos muchas operaciones (media etc..) queremos calcularlas para distintos grupos (hombre, mujer …). group_by() permite hacerlo.

group_by()coge un df y lo convierte en un “df agrupado”. En ese nuevo “df agrupado”, las operaciones que hagamos con summarise() se harán por separado para cada uno de los grupos que hayamos definido. Ahora lo vemos.

Si, por ejemplo, agrupamos un df por países, al ejecutar summarise(), nos retornará una fila con el resultado para cada país. En realidad, podemos pensar que group_by() no hace “nada”, que en realidad solo cambia lo que hacen las otras funciones: ahora los cálculos se harán para cada uno de los grupos que define group_by()



Como dice Jenny: Let’s start with simple counting. ¿Cuantas observaciones(rows) tenemos por continente?

#- cogemos df y lo (des)agrupamos por grupos definidos por la variable "continent"; osea, habrá 5 grupos
#- después con summarise() calcularemos el nº de observaciones en cada continente o grupo; es decir, nos retornará un df con una fila por cada continente
aa <- gapminder %>% group_by(continent) %>% summarise(NN = n())
aa
continent NN
Africa 624
Americas 300
Asia 396
Europe 360
Oceania 24

Esto tan sencillo también se puede hacer con count()

aa <- gapminder %>% group_by(continent) %>% count()
aa <- gapminder %>% group_by(continent) %>% count(name = "NN")
aa


¿Y cuantos países hay en la base de datos? Para este tipo de cosas, se pueden usar funciones de R-base, pero dplyr tiene muchas funciones auxiliares.

#- cogemos df y lo agrupamos por "continent",
#- después calculamos 2 cosas: el número de observaciones o rows
#- y el número de países en cada continente (NN_countries)
aa <- gapminder %>% group_by(continent) %>%
          summarize(NN = n(),
                    NN_countries = n_distinct(country))
aa
continent NN NN_countries
Africa 624 52
Americas 300 25
Asia 396 33
Europe 360 30
Oceania 24 2


Calculemos la esperanza de vida media por continente

#- cogemos df y lo agrupamos por "continent", después calculamos la media de "lifeExp"
aa <- gapminder %>% group_by(continent) %>%
                    summarize(mean(lifeExp))
aa
continent mean(lifeExp)
Africa 48.86533
Americas 64.65874
Asia 60.06490
Europe 71.90369
Oceania 74.32621

Guau! Hay que irse a vivir a Oceanía!!


Calculemos la esperanza de vida media por continente en el primer periodo (1952)

#- cogemos df y filtramos para quedarnos con las observaciones de 1952
#- después lo agrupamos por "continent",
#- después calculamos la media de "lifeExp"
gapminder %>% filter(year == "1952") %>%
              group_by(continent) %>%
              summarize(mean(lifeExp)) 
continent mean(lifeExp)
Africa 39.13550
Americas 53.27984
Asia 46.31439
Europe 64.40850
Oceania 69.25500

Habría sido mejor en lugar de poner filter(year == "1952") haber puesto filter(year == min(year))

Guau! Habría que haber vivido en Oceanía (en 1952)!!


Se pueden calcular varios estadísticos a la vez

#- cogemos df y filtramos (cogemos) las observaciones de 1952 y 2007
#- agrupamos por "continent",
#- después calculamos la media de "lifeExp" y de "gdpPercap"
gapminder %>% filter(year %in% c(1952, 2007)) %>%
             group_by(continent, year) %>%
             summarize(mean(lifeExp), mean(gdpPercap)) 

Vamos a hacer cálculos cada vez más complejos:

#- cogemos df y lo agrupamos por "continent" y "year",
#- después calculamos la media de "lifeExp" y de "gdpPercap"
gapminder %>% filter(year %in% c(1952, 2007)) %>%
              group_by(continent, year) %>%
              #- después calculamos la media de "lifeExp"
              summarise(media = mean(lifeExp))
#- Voy a crear un nuevo df: "gapminder_gr" o "gapminder agrupado"
gapminder_gr <- gapminder %>% filter(year %in% c(1952, 2007)) %>%
                 group_by(continent, year)
#- y sobre "gapminder_gr" iremos haciendo cálculos

#- si queremos calcular la media de varias variables tenemos que usar across()
gapminder_gr %>% summarise(across(c(lifeExp, gdpPercap), mean))

#- si queremos calcular la media de todas las variables numéricas tenemos que usar across() y where()
gapminder_gr %>% summarise(across(where(is.numeric), mean))

#- si queremos calcular la media y la mediana, hay que usar list()
gapminder_gr %>% summarise(across(c(lifeExp, gdpPercap),
                            list (media = mean, mediana = median) ))

#- si ponemos los nombres de los argumentos quedaría como
gapminder_gr %>% summarise(across(.cols = c(lifeExp, gdpPercap),
                                  .fns = list (media = mean, mediana = median)))

#- además, podemos controlar el nombre de las variables creadas con el argumento .names
gapminder_gr %>% summarise(across(c(lifeExp, gdpPercap),
                        list (media = mean, mediana = median),
                        .names = "{fn}_{col}"))


Preguntas de verdad

Bueno, pues ya conocéis lo principal, lo básico y más importante de dplyr, solo queda ir cogiendo práctica y confianza, así que para ello toca hacer una serie de preguntas de verdad!!. Por ejemplo:

  1. ¿en que continente ha aumentado más la esperanza de vida en el periodo 1952-2007?
#- cogemos df y lo agrupamos por "continent", después calculamos la media de "lifeExp"
gapminder %>%
  filter(year %in% c(1952, 2007)) %>%
  group_by(continent, year) %>%
  summarize(media = mean(lifeExp)) %>% ungroup()
continent year media
Africa 1952 39.13550
Africa 2007 54.80604
Americas 1952 53.27984
Americas 2007 73.60812
Asia 1952 46.31439
Asia 2007 70.72848
Europe 1952 64.40850
Europe 2007 77.64860
Oceania 1952 69.25500
Oceania 2007 80.71950

Casi, pero no!! Sólo hemos conseguido ver la esperanza de vida por continente en 1952 y 2007.En realidad esto ya lo hicimos antes. Falta restar.

Quizás podríamos calcular el máximo, el mínimo y restarlos. No, porque supondríamos que “lifeExp” siempre aumenta. Vamos que el tiempo apremia:

#- primer intento: se puede hacer de una vez, pero vamos a partir el código en 2 trozos
aa <- gapminder %>% filter(year %in% c(1952, 2007)) %>%
  group_by(continent, year) %>%
  summarize(media = mean(lifeExp)) %>% ungroup()

aa1 <- aa %>% group_by(continent) %>%
  summarise(min_l = min(media), max_l = max(media)) %>%
  mutate(dif = max_l-min_l) %>%
  arrange(desc(dif))

aa1
continent min_l max_l dif
Asia 46.31439 70.72848 24.41409
Americas 53.27984 73.60812 20.32828
Africa 39.13550 54.80604 15.67054
Europe 64.40850 77.64860 13.24010
Oceania 69.25500 80.71950 11.46450

Asia son los ganadores. En promedio, en Asía se mejoró la esperanza de vida en 24.4 años entre 1952 y 2007.

Lo de restar el máximo y el mínimo ha funcionado, PERO podría no haberlo hecho si hubiese habido algún continente en el que en la esperanza de vida, en lugar de haber aumentado, hubiese bajado. Entonces, ¿cómo lo hacemos?

#- segundo intento: se puede hacer de una vez, pero vamos a partir el código en 2 trozos
aa <- gapminder %>% filter(year %in% c(1952, 2007)) %>%
         group_by(continent, year) %>%
         summarize(media = mean(lifeExp)) %>% ungroup()

#- usamos lag()
aa1 <- aa %>% group_by(continent) %>%
              arrange(year) %>%
              mutate(variac_l = media - lag(media))

#- mostramos los resultados
aa1 %>% filter(year == 2007) %>% arrange(desc(variac_l))
continent year media variac_l
Asia 2007 70.72848 24.41409
Americas 2007 73.60812 20.32828
Africa 2007 54.80604 15.67054
Europe 2007 77.64860 13.24010
Oceania 2007 80.71950 11.46450

Sí, Asia es la ganadora. En promedio, en Asía se mejoró la esperanza de vida en 24 años entre 1952 y 2007.

Otra forma de obtener el mismo resultado:

#- esta parte es común
aa <- gapminder %>%
  filter(year %in% c(1952, 2007)) %>%
  group_by(continent, year) %>%
  summarize(media = mean(lifeExp)) %>% ungroup()

#- pero ahora usamos pivot_wider()
aa %>% pivot_wider(names_from = year, values_from = media) %>%
     mutate(dif_l = `2007` - `1952`) %>%
     arrange(desc(dif_l))
continent 1952 2007 dif_l
Asia 46.31439 70.72848 24.41409
Americas 53.27984 73.60812 20.32828
Africa 39.13550 54.80604 15.67054
Europe 64.40850 77.64860 13.24010
Oceania 69.25500 80.71950 11.46450


El chunk de abajo, ¿qué hace? ¿qué se está calculando?:

aa <- gapminder %>%
  group_by(continent, year) %>%
  select(continent, year, lifeExp) %>%
  summarise(mean_life = mean(lifeExp)) %>%
  arrange(year) %>%
  mutate(incre_mean_life_0 = mean_life - first(mean_life)) %>%
  mutate(incre_mean_life_t = mean_life - lag(mean_life)) %>%
  arrange(continent)

#- por ejemplo veamos el resultado para Europe
aa %>% filter(continent == "Europe")
continent year mean_life incre_mean_life_0 incre_mean_life_t
Europe 1952 64.40850 0.000000 NA
Europe 1957 66.70307 2.294567 2.2945667
Europe 1962 68.53923 4.130733 1.8361667
Europe 1967 69.73760 5.329100 1.1983667
Europe 1972 70.77503 6.366533 1.0374333
Europe 1977 71.93777 7.529267 1.1627333
Europe 1982 72.80640 8.397900 0.8686333
Europe 1987 73.64217 9.233667 0.8357667
Europe 1992 74.44010 10.031600 0.7979333
Europe 1997 75.50517 11.096667 1.0650667
Europe 2002 76.70060 12.292100 1.1954333
Europe 2007 77.64860 13.240100 0.9480000

Sed conscientes de que la soluciones a una pregunta no sale a la primera, a veces hay que calentarse el cap:

Break the code into pieces, starting at the top, and inspect the intermediate results. That’s certainly how I was able to write such a thing. These commands do not leap fully formed out of anyone’s forehead – they are built up gradually, with lots of errors and refinements along the way. Is the statement above really hard for you to read? If yes, then by all means break it into pieces and make some intermediate objects. Your code should be easy to write and read when you’re done. —- Jenny Bryan


Otras cuestiones que podemos resolver con dplyr sobre la esperanza de vida:

  • ¿Cómo ha evolucionado la esperanza de vida en Spain lustro a lustro?
#- variación de lifeExp en Spain año a año (bueno lustro a lustro)
aa <- gapminder %>%
  group_by(country) %>%
  select(country, year, lifeExp) %>%
  mutate(lifeExp_gain_cada_lustro = lifeExp - lag(lifeExp)) %>%
  filter(country == "Spain" )
aa
country year lifeExp lifeExp_gain_cada_lustro
Spain 1952 64.940 NA
Spain 1957 66.660 1.720
Spain 1962 69.690 3.030
Spain 1967 71.440 1.750
Spain 1972 73.060 1.620
Spain 1977 74.390 1.330
Spain 1982 76.300 1.910
Spain 1987 76.900 0.600
Spain 1992 77.570 0.670
Spain 1997 78.770 1.200
Spain 2002 79.780 1.010
Spain 2007 80.941 1.161


  • ¿Y la variación acumulada? Fácil!! Sólo tendríamos que sumar o acumular la variable “lifeExp_gain_cada_lustro” que hemos generado anteriormente, así que sólo habría que añadir una linea a nuestro código:
#- ganancia acumulada
aa <- gapminder %>%
  group_by(country) %>%
  select(country, year, lifeExp) %>%
  mutate(lifeExp_gain_cada_lustro = lifeExp - lag(lifeExp)) %>%
  #--- 2 filas nuevas: ifelse()  y cumsum()
  mutate(lifeExp_gain_cada_lustro2 = ifelse(is.na(lifeExp_gain_cada_lustro), 0, lifeExp_gain_cada_lustro)) %>%
  mutate(lifeExp_gain_acumulado = cumsum(lifeExp_gain_cada_lustro2)) %>%
  filter(country == "Spain")
aa
country year lifeExp lifeExp_gain_cada_lustro lifeExp_gain_cada_lustro2 lifeExp_gain_acumulado
Spain 1952 64.940 NA 0.000 0.000
Spain 1957 66.660 1.720 1.720 1.720
Spain 1962 69.690 3.030 3.030 4.750
Spain 1967 71.440 1.750 1.750 6.500
Spain 1972 73.060 1.620 1.620 8.120
Spain 1977 74.390 1.330 1.330 9.450
Spain 1982 76.300 1.910 1.910 11.360
Spain 1987 76.900 0.600 0.600 11.960
Spain 1992 77.570 0.670 0.670 12.630
Spain 1997 78.770 1.200 1.200 13.830
Spain 2002 79.780 1.010 1.010 14.840
Spain 2007 80.941 1.161 1.161 16.001

Al final para hacerlo (como había pensado) me han hecho falta 2 lineas, porque la primera observación de “lifeExp_gain_cada_lustro” es un NA y eso hacía que la función cumsum() no funcionase.


  • Otra forma de hacer lo mismo sería (se me ha ocurrido después). Además es más fácil
#- ganancia acumulada (otra forma de hacer lo mismo)
aa <- gapminder %>%
  group_by(country) %>%
  select(country, year, lifeExp) %>%
  mutate(lifeExp_gain_acumulada = lifeExp - lifeExp[1])  %>%
  filter(country == "Spain")


  • Obtener, para cada periodo, los (3) países de Asia con MAYOR lifeExp. Usaremos una variante de slice(), concrétamente slice_max()
aa <- gapminder %>%
  filter(continent == "Asia") %>%
  select(year, country, lifeExp) %>%
  group_by(year) %>%
  slice_max(n = 3, lifeExp) %>%
  arrange(year) 

Para obtener los 4 países con MENOR “lifeExp” sólo tendríamos que sustituir la quinta linea por slice_min(n = 4, lifeExp)


  • Obtener, para cada periodo, los países de Asia con mayor y menor lifeExp.
#- Obtener, para cada periodo, los países de Asia con mayor y menor lifeExp.
aa <- gapminder %>%
  filter(continent == "Asia") %>%
  select(year, continent, country, lifeExp) %>%
  group_by(year) %>%
  filter(min_rank(desc(lifeExp)) < 2 | min_rank(lifeExp) < 2) %>%
  arrange(year) 

Las 2 últimas funciones que hemos usado: slice_min() y min_rank() son funciones de dplyr pero no son son funciones principales, en cierta forma son auxiliares.


Podéis ver las funciones auxiliares que tiene dplyr en la segunda página de CHEAT SHEET. también se pueden usar las funciones de R-base o de otros packages. Aquí tenéis algunas posibilidades sacadas de un tutorial de Hadley

Types of summary functions:
• min(x), median(x), max(x), quantile(x, p)
• n(), n_distinct(), sum(x), mean(x)
• sum(x > 10), mean(x > 10)
• sd(x), var(x), iqr(x), mad(x)

Types of window functions:
• Ranking and ordering
• Offsets: lead & lag
• Cumulative aggregates
• Rolling aggregates

Ejemplos:
• Was there a change?  x != lag(x)
• Percent change? (x - lag(x)) / x
• Fold-change? x / lag(x)
• Previously false, now true? !lag(x) & x

• If one of the specialised verbs doesn’t do what you need, you can use do()


A ver si entendeis este ejemplo

Una función auxiliar que es muy útil al utilizarla junto a mutate: case_when().

aa <- gapminder %>%
  group_by(continent, year)  %>%
  mutate(media_lifeExp = mean(lifeExp)) %>%
  mutate(media_gdpPercap = mean(gdpPercap)) %>%
  mutate(GOOD_or_BAD = case_when(
    lifeExp > mean(lifeExp) & gdpPercap > mean(gdpPercap)  ~ "good",
    lifeExp < mean(lifeExp) & gdpPercap < mean(gdpPercap)  ~ "bad" ,
    lifeExp < mean(lifeExp) | gdpPercap < mean(gdpPercap)  ~ "medium"
    )) %>%
  filter(country == "Spain")


Más funciones auxiliares

Hay algunas que no quiero olvidar:

dplyr::ntile(x, n) : categorizes a vector of values into "ntiles" such as quartiles if n = 4

dplyr::n_distinct(x):  counts unique values in a vector; similar a  length(unique(x))

dplyr::between(x, left, right) : is a shortcut for x >= left & x <= right,

tibble::add_row()  : añade rows a un df

tibble::rownames_to_column()


Más detalles sobre dplyr

Los verbos mutate() y summarise() ya sabemos que pueden hacer uso de funciones como mean(), sd() etc … Por ejemplo podemos transformar las variables numéricas a logaritmos:

gapminder %>% mutate(across(where(is.numeric), log)) %>% head(n = 3)

Se pueden usar tidy_helpers para seleccionar las columnas y aplicar más de una función:

gapminder %>% mutate(across(c(starts_with("life"), contains("po")), .fns =  mean)) %>% head

o calcular la media de las variable numéricas y controlar el nombre de las variables creadas:

gapminder %>% group_by(continent) %>%
  summarize(across(where(is.numeric), mean, .names = "mean_{col}")) %>% head(n = 3)

Si quieres seleccionar todas las variables usa everything()

gapminder %>% group_by(continent) %>%
  summarize(across(everything(), as.character, .names = "CHAR_{col}")) %>% head(n = 3)

Pero vamos a ver otras posibilidades:


más de una condición para seleccionar las columnas:

gapminder %>% mutate(across(where(is.double) & ends_with("cap"), as.integer)) %>% head(n = 3)
country continent year lifeExp pop gdpPercap
Afghanistan Asia 1952 28.801 8425333 779
Afghanistan Asia 1957 30.332 9240934 820
Afghanistan Asia 1962 31.997 10267083 853


uso de funciones propias (!!!!)

Tenemos varias posibilidades:

    1. definiendo primero la función:
dividir_100 <- function(x) {x / 100}  #- defino una función

gapminder %>% mutate(across(where(is.numeric), .fns = dividir_100)) %>% head
country continent year lifeExp pop gdpPercap
Afghanistan Asia 19.52 0.28801 84253.33 7.794453
Afghanistan Asia 19.57 0.30332 92409.34 8.208530
Afghanistan Asia 19.62 0.31997 102670.83 8.531007
Afghanistan Asia 19.67 0.34020 115379.66 8.361971
Afghanistan Asia 19.72 0.36088 130794.60 7.399811
Afghanistan Asia 19.77 0.38438 148803.72 7.861134
    1. usando formulas anónimas
gapminder %>% mutate(across(where(is.numeric), .fns = function(x) {x / 100})) %>% head
country continent year lifeExp pop gdpPercap
Afghanistan Asia 19.52 0.28801 84253.33 7.794453
Afghanistan Asia 19.57 0.30332 92409.34 8.208530
Afghanistan Asia 19.62 0.31997 102670.83 8.531007
Afghanistan Asia 19.67 0.34020 115379.66 8.361971
Afghanistan Asia 19.72 0.36088 130794.60 7.399811
Afghanistan Asia 19.77 0.38438 148803.72 7.861134
    1. usando fórmulas
#- con formulas
gapminder %>% mutate(across(where(is.numeric), .fns = ~ .x/100)) %>% head
country continent year lifeExp pop gdpPercap
Afghanistan Asia 19.52 0.28801 84253.33 7.794453
Afghanistan Asia 19.57 0.30332 92409.34 8.208530
Afghanistan Asia 19.62 0.31997 102670.83 8.531007
Afghanistan Asia 19.67 0.34020 115379.66 8.361971
Afghanistan Asia 19.72 0.36088 130794.60 7.399811
Afghanistan Asia 19.77 0.38438 148803.72 7.861134

gapminder %>% mutate(across(where(is.numeric), .fns = ~ {1/sqrt(.)})) %>% head
country continent year lifeExp pop gdpPercap
Afghanistan Asia 0.0226339 0.1863358 0.0003445 0.0358185
Afghanistan Asia 0.0226050 0.1815723 0.0003290 0.0349034
Afghanistan Asia 0.0225762 0.1767850 0.0003121 0.0342373
Afghanistan Asia 0.0225475 0.1714482 0.0002944 0.0345816
Afghanistan Asia 0.0225189 0.1664633 0.0002765 0.0367612
Afghanistan Asia 0.0224904 0.1612945 0.0002592 0.0356662

usar formulas facilita el uso de argumentos dentro de la función:

gapminder %>%
  group_by(continent, year) %>%
  summarise(across(c("lifeExp", "gdpPercap"),
                   list(mean = ~ mean(.x, na.rm = TRUE, trim = 0.1))))

crear indices de grupo

A veces cuando trabajas con df agrupados es útil saber a que grupo pertenece cada observación. Puedes hacerlo fácilmente con:

gapminder %>% group_by(year)  %>% mutate(id_grupo = cur_group_id()) 




5. Combinando (joining) df’s


OK, ya sabemos manejar/filtrar etc… un conjunto de datos, PERO muchas veces lo que hay que hacer es unir o combinar varias tablas o conjuntos de datos (Joinings en inglés).

Vamos a aprender como hacerlo usando dplyr. [Aquí] tenéis la vignette de dplyr para “two table verbs”, también podéis ver un vídeo muy ilustrativo [aquí]. También podéis usar el tutorial de Jenny. La CHEAT SHEET es muy-muy buena.

Los ejemplos e imágenes usados en esta sección se basan en los creados por Mara Averick (@dataandme) en este repo, que a su vez se basaron en la idea de Garrick Aden-Buie @grrrck


(dos) casos sencillos

Dos casos ideales (sencillos de unir): bind_cols() y bind_rows()

  1. Si los 2 dfs tienen exactamente las mismas filas o unidades de análisis ( y ademas en el mismo orden). En este caso, solo habría que juntar en una misma tabla las columnas de df1 y de df2. Esto lo podemos hacer con bind_cols() (o con cbind() de R-base).
df_1 <- iris[ , 1:2]  ; df_2 <- iris[ , 3:5]

df_1 <- iris %>% select(1:2)  ; df_2 <- iris %>% select(3:5)

df_3 <- `bind_cols`(df_1, df_2)

identical(iris, df_3)
  1. Si los 2 dfs tienen exactamente las mismas columnas ( y ademas en el mismo orden). En este caso, se trataría simplemente de juntar todas las observaciones o filas de los 2 df’s. Esto lo podemos hacer con bind_rows() (o con rbind() de R-base)
df_1 <- iris[1:75, ]  ; df_2 <- iris[76:150, ]

df_1 <- iris %>% slice(1:75)  ; df_2 <- iris %>% slice(76:150)

df_3 <- `bind_rows`(df_1, df_2)

identical(iris, df_3)


Olvidando ya los 2 casos ideales y sencillos


En dplyr hay 3 tipos de funciones(verbos) que se ocupan de diferentes operaciones para unir datasets:

  • Mutating joins, añade nuevas variables (o columnas) a un dataframe (df1). Estas nuevas columnas vienen de un segundo df2 (hay varias mutating joins, dependiendo del criterio para seleccionar las filas)

  • Filtering joins, filtra las filas (observaciones) de un dataframe (df1) basándose en sí las filas de df1 coinciden (match) o no con una observación del segundo df2

  • Set operations, combina las observaciones de los dos datasets (df1 y df2) as if they were set elements.


Todas estas funciones tienen una estructura similar: sus dos primeros argumentos son 2 df’s (en realidad tablas de datos): df1 y df2. El output de la función es siempre una nueva tabla (del mismo tipo que df1).



Mutating joins

Hay 4 tipos de mutating joins. Su sintaxis es idéntica, sólo se diferencian en que las filas que se seleccionan dependen del criterio para hacer el match:

  • inner_join(df1,df2): Retorna todas las columnas de df1 y también las de df2, PERO solo retorna las filas de df1 que tienen una equivalencia en df2. (la equivalencia se define en función del valor de una variable o variables comunes en df1 y df2)

  • left_join(df1,df2): Retorna todas las columnas de df1 y también las de df2; en cuanto a las filas, retorna TODAS las filas de df1. (Si hubiesen varios matches entre df1 e df2 se retornan todas las combinaciones!!!!)

  • rigth_join(df1,df2): Retorna todas las columnas de df1 y también las de df2; en cuanto a las filas, retorna TODAS las filas de df2. De df2!! (Si hubiesen varios matches entre df1 y df2 se retornan todas las combinaciones!!!!)

  • full_join(df1,df2): Retorna todas las columnas de df1 y también las de df2; en cuanto a las filas, retorna TODAS las filas de df1 y de df2. Osea, retorna TODAS las filas y TODAS las columnas de las 2 tablas. (Donde no hay matches retorna NA’s)


Ejemplos de mutating joins

Sean los siguientes 2 dataframes (tibbles):

x <- tibble(id = 1:3, x = paste0("x", 1:3))

y <- tibble(id = (1:4)[-3], y = paste0("y", (1:4)[-3]))


Inner Join
#- only includes observations that match in both x and y
df_inner <- inner_join(x, y)


Left Join
#- includes all observations in x, regardless of whether they match or not.
#- This is the most commonly used join because it ensures that you don’t lose observations from your primary table.
df_left_join <- left_join(x, y)


Right Joint
#- includes all observations in y.
#- It’s equivalent to left_join(y, x), but the columns will be ordered differently.
df_right_join <- right_join(x, y)


Full Joint
#- full_join() includes all observations from x and y
df_full_join <- full_join(x, y)


2 precisiones sobre las mutating joins

Las (left, right and full) joins se llaman colectivamente como “outer joins”. Cuando una fila no tiene match en una outer join, las nuevas variables que se crean se llenan con NA’s.

Las mutating joins se usan principalmente para añadir columnas, PERO en el proceso pueden generarse nuevas filas: si un match no es único, se añadirán todas las combinaciones posibles (el producto cartesiano) de las matching observations. Veamos un ejemplo con una left_join:

x <- tibble(id = 1:3, x = paste0("x", 1:3))

y <- tibble(id = c(1:4,2)[-3], y = paste0("y", c(1:5)[-3]))


left_join() en la que se crean nuevas filas
df_left_join <- left_join(x, y)


Importante ¿Cómo decir a las funciones la columnas (o columnas) que se usarán para hacer los matching?

Podemos(DEBEMOS) elegir las columnas (o variables) que nos servirán para unir los 2 df’s. Estas columnas que se usan para para hallar los matchings y que por tanto nos permiten fusionar los 2 df’s se llaman “keys”.

La opción de las funciones para seleccionar estas columnas “keys” es by =.

  • si ponemos left_join(df1, df2, by = "X1") se hará una left_join siendo la variable “X1” la que hará de key. Si las variables key no se llamasen igual en los 2 df’s siempre podemos renombrarlas o hacer lo siguiente: left_join(df1, df2, by = c("X1" = "D4")

  • También se pueden fusionar tablas usando dos keys; por ejemplo: left_join(df1, df2, by = c("X1", "X2")) . Si no se llamasen igual las variables en df1 y df2 haríamos left_join(df1, df2, by = c("X1" = "D4", "X2" = "D7"))




Filtering joins

Filtering joins son similares a los anteriores (Mutating joins); o sea, hacen machting con las filas de la misma manera, PERO afectan a las filas, NO a las columnas. Hay filtering joins de 2 tipos:

  • semi_join(df1,df2): retorna las observaciones de df1 que tienen un match en df2. En cuanto a las columnas sólo retorna las columnas de df1
  • anti_join(df1,df2): retorna las observaciones de df1 que NO tienen un match en df2; osea, quita las observaciones con match. En cuanto a las columnas sólo retorna las columnas de df1

La semi_join se diferencia de la inner_join en que la inner_join solo retorna una fila de df1 por cada matching, mientras que la semi_join NUNCA duplica filas de df1

Las filtering joins son útiles para diagnosticar mismatches. Si quieres saber sobre los matches, haz una semi_join() or anti_join(). semi_join() and anti_join() NUNCA duplican filas; solo pueden quitar filas.


semi_join
df_semi_join <-  semi_join(x, y, by = "id")


anti_join
df_anti_join <-  anti_join(x, y, by = "id")


comparemos la semi_join con la inner_join
df_inner <-  inner_join(x, y, by = "id")
df_semi_join <-  semi_join(x, y, by = "id")

La inner_join

    1. añade variables del data.frame y al data.frame x (en este caso y$y)
    1. sólo retiene las rows del data.frame x que tienen un match en y (en este caso las 2 primeras filas de x)
    1. PERO en este ejemplo ADEMÁS duplica rows. la razón es que hay matching duplicados, hay 2 unos en x y otros 2 unos en el data.frame y (con distintos valores de y$y) (así que salen 4 rows)


Importante ¿Cómo decir a las funciones la columna, o columnas, que se usarán para hacer los matching?

Al igual que con las mutating joins, en las filtering joins también podemos(DEBEMOS) elegir las columnas (o variables) que nos servirán para unir los 2 df’s. Estas columnas que se usan para para hallar los matchings y que por tanto que permiten fusionar los 2 df’s se llaman “keys”.

La opción de las funciones para seleccionar estas columnas “keys” es by =.

  • si ponemos semi_join(df1, df2, by = "X1") se hará una semi_join siendo la variable “X1” la que hará de key. Si las variables key no se llamasen igual en los 2 df’s siempre podemos renombrarlas o hacer semi_join(df1, df2, by = c("X1" = "D4")

  • si ponemos semi_join(df1, df2,by = c("X1", "X2") hará falta que una row de df1 tenga los valores tanto de X1 como de X2 iguales a los de esas mismas variables en df2. Si no se llamasen igual las variables en df1 y df2 haríamos by = c("X1" = "D4", "X2" = "D7")



Set operations

Este tipo de joins es más estricta: hace falta que los 2 df’s tengan las mismas variables (o columnas). Los 2 df’s pueden tener observaciones(filas) diferentes, PERO es necesario que tengan las mismas variables (o columnas).

Como los 2 df’s tienen las mismas columnas, entonces es como si se tratasen los dfs como conjuntos:

  • intersect(df1, df2): devuelve un df con las observaciones comunes en df1 y df2
  • union(df1, df2): devuelve la unión; o sea, las observaciones de df1 y de df2 (quitando las posibles filas duplicadas)
  • union_all(df1, df2): devuelve la unión (sin quitar los duplicados)
  • setdiff(df1, df2): devuelve las filas en df1 que no están en df2


  • setequal(df1,df2: retorna TRUE si df y df2 tienen exactamente las mismas filas (da igual el orden en el que estén las filas)


x <- tibble::tibble(v1 = c(1, 1, 2), v2 = c("a" , "b", "a"))
y <- tibble::tibble(v1 = c(1, 2),    v2 = c("a" , "b"))


intersección
intersect(x, y)


unión
union(x, y)


setdiff
setdiff(x, y)

Puedes probar tú mismo a cambiar el orden del los df’s en setdiff():

setdiff(y, x)


setqual

Sirve para determinar si 2 df’s son iguales (sin importar el orden en que estén las filas)

setequal(x, y)
#> [1] FALSE
setequal(union(x, y), union(y, x))
#> [1] TRUE




Esperando a GODOT/ggplot2


Bueno, pues hemos visto “TODO” sobre manipulación de datos. El próximo tutorial va de visualización: hacia ggplot2

Aquí un pequeño avance:

library("ggplot2")
my_plot <- ggplot(gapminder, aes(x = continent, y = lifeExp)) +
  geom_boxplot(outlier.colour = "hotpink") +
  geom_jitter(position = position_jitter(width = 0.1, height = 0), alpha = 1/4) +
  labs(title = "Experanza de vida (por continente)",
       subtitle = "Datos de gapminder. 1952-2007(observaciones cada 5 años)",
       caption = "Source: Gapminder. Jenny Bryan rocks in gapminder vignette!!",
       x = "Continente", y = "Esperanza de Vida (lifeExp)") 




gapminder2 <- gapminder %>% mutate(year = as.factor(year))

library("ggplot2")
my_plot <- ggplot(gapminder2, aes(x = year, y = lifeExp)) +
  geom_boxplot(outlier.colour = "hotpink") +
  geom_jitter(position = position_jitter(width = 0.1, height = 0), alpha = 1/4) +
  labs(title = "Experanza de vida (por año)",
       subtitle = "Datos de gapminder. 1952-2007(observaciones cada 5 años)",
       caption = "Source: Gapminder. Jenny Bryan rocks in gapminder vignette!!",
       x = "Periodo", y = "Esperanza de Vida (lifeExp)") 



library("ggplot2")
my_plot <- ggplot(gapminder, aes(x = gdpPercap, y = lifeExp, colour = continent)) +
  geom_jitter(position = position_jitter(width = 0.1, height = 0), alpha = 1/4) +
  labs(title = "Experanza de vida vs. GDP (per cápita)",
       subtitle = "Datos de gapminder. 1952-2007(observaciones cada 5 años)",
       caption = "Source: Gapminder. Jenny Bryan rocks in gapminder vignette!!",
       x = "GDP (per cápita)", y = "Esperanza de Vida (lifeExp)") 




Tidylog package

Una herramienta que os puede ser de utilidad para aprender el uso de dplyr es el paquete tidylog. Este paquete nos da feedback instantáneo sobre qué hacemos cuando usamos las principales funciones de dplyr y tidyr. Veámoslo en acción:

library("dplyr")
library("tidyr")
library("tidylog", warn.conflicts = FALSE)
filtered <- filter(mtcars, cyl == 4)
#> filter: removed 21 rows (66%), 11 rows remaining
mutated <- mutate(mtcars, new_var = wt ** 2)
#> mutate: new variable 'new_var' (double) with 29 unique values and 0% NA

Si os fijáis, cada vez que ejecutas una función de dplyr nos devuelve un mensaje explicándonos que se ha hecho. Por ejemplo la linea de código filtered <- filter(mtcars, cyl == 4) ha creado un nuevo data.frame donde se han eliminado 21 filas del df original: #> filter: removed 21 rows (66%), 11 rows remaining.

La segunda linea de código mutated <- mutate(mtcars, new_var = wt ** 2) ha creado una nueva variable con mutate(): #> mutate: new variable 'new_var' with 29 unique values and 0% NA.




Tidyverse vs. Base R

Todo lo que se puede hacer con dplyr, tidyr etc. ,también se puede hacer con Base-R pero de una manera mucho menos intuitiva.

El siguiente ejemplo esta sacado de este post. Son dos trozos de código que hacen exactamente lo mismo:

Con el tidyverse:

library(dplyr)
mtcars %>%
  group_by(cyl, am) %>%
  select(mpg, cyl, wt, am) %>%
  summarise(avgmpg = mean(mpg), avgwt = mean(wt)) %>%
  filter(avgmpg > 20)

Con la sintaxis de base R:

filter(
  summarise(
    select(
      group_by(mtcars, cyl, am),
      mpg, cyl, wt, am),
    avgmpg = mean(mpg), avgwt = mean(wt)),
  avgmpg > 20)

O puesto en horizontal

filter(summarise(select(group_by(mtcars, cyl, am),  mpg, cyl, wt, am),avgmpg = mean(mpg), avgwt = mean(wt)), avgmpg > 20)


Otros 2 ejemplos de comparación tidyverse versus Base-R:
df %>% filter(country == "Spain") %>%  select(year, lifeExp)

df[df$country == "Spain", c("year", "lifeExp")] 

Este último ejemplo lo introduzco porque quiero recordar estas trasparencias que explican que el paquete dbplyr traduce expresiones de dplyr a SQL. El código de las transparencias está aquí, y aquí puedes aprender como hacer queries SQL a un database utilizando la sintaxis de dplyr.

#- con tidyverse
df_cars %>%
  select(longname, cyl, hp) %>%
  mutate( shortname = word(longname, 1) ) %>%
  select( - longname)  ->
df_cars_limited
head( df_cars_limited )

#- con dplyr PERO sin %>%
head(
  select(
    mutate(
      select(df_cars, longname, cyl, hp) ,
      shortname = word(longname, 1)
      ),
    -longname
    )
)





Bibliografía



  1. Hasta la aparición de tidyr 1.0.0, las funciones que se usaban eran gather() y spread(). En esta conferencia, Hadley Wickham nos contó que uno de sus grandes errores durante el desarrollo del tidyverse fue la elección de los nombres de las funciones gather() y spread(). Finalmente podemos decir: bye bye gather() y spread(), wellcome tidyr 1.0.0 and pivot_longer() and pivot_wider().↩︎

  2. La última vez que miré la cheatsheet aún no estaba actualizada para recoger los cambios que aparecieron en dplyr 1.0.0, pero aún así os resultará de mucha utilidad↩︎

  3. Hasta mayo de 2020, esto es, hasta dplyr 1.0.0, esto se hacia con la función select_if()↩︎

  4. Hasta la aparición de dplyr 1.0.0 en mayo de 2020, se utilizaba la función sumarise_all()↩︎

LS0tCnRpdGxlOiAnRGF0YSBtdW5naW5nOiBtYW5lam8gZGUgZGF0b3MgY29uIFIsIHRoZSB0aWR5dmVyc2Ugd2F5JwphdXRob3I6ICJQZWRybyBKLiBQw6lyZXogKHBlZHJvLmoucGVyZXpAdXYuZXMpLiBVbml2ZXJzaXRhdCBkZSBWYWzDqG5jaWEgPGJyPiA8YnI+IFdlYiBkZWwgY3Vyc286IDxodHRwczovL3BlcmV6cDQ0LmdpdGh1Yi5pby9pbnRyby1kcy0yMC0yMS13ZWIvPiIKZGF0ZTogIk5vdmllbWJyZSBkZSAyMDE3IChhY3R1YWxpemFkbyBlbCBgciBmb3JtYXQoU3lzLnRpbWUoKSwgJyVkLSVtLSVZJylgKSIKb3V0cHV0OgogIGh0bWxfZG9jdW1lbnQ6CiAgICBjc3M6ICFleHByIGhlcmU6OmhlcmUoImFzc2V0cyIsICJzdHlsZXNfcGpwLmNzcyIpCiAgICB0aGVtZTogcGFwZXIKICAgIGhpZ2hsaWdodDogdGV4dG1hdGUKICAgIHRvYzogdHJ1ZQogICAgdG9jX2RlcHRoOiAzCiAgICB0b2NfZmxvYXQ6CiAgICAgIGNvbGxhcHNlZDogdHJ1ZQogICAgICBzbW9vdGhfc2Nyb2xsOiB0cnVlCiAgICBzZWxmX2NvbnRhaW5lZDogdHJ1ZQogICAgbnVtYmVyX3NlY3Rpb25zOiBmYWxzZQogICAgaW5jbHVkZXM6CiAgICAgIGFmdGVyX2JvZHk6ICFleHByIGhlcmU6OmhlcmUoImFzc2V0cyIsICJmb290ZXIuaHRtbCIpIAogICAgICBpbl9oZWFkZXI6IAogICAgICAgIC0gIWV4cHIgaGVyZTo6aGVyZSgiYXNzZXRzIiwgImdvb2dsZS1hbmFseXRpY3MuaHRtbCIpIAogICAgICAgIC0gIWV4cHIgaGVyZTo6aGVyZSgiYXNzZXRzIiwgImZhdmljb24tc29sLmh0bWwiKQogICAgZGZfcHJpbnQ6IGthYmxlCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCmVkaXRvcl9vcHRpb25zOgogIGNodW5rX291dHB1dF90eXBlOiBjb25zb2xlCi0tLQoKYGBge3IsIGluY2x1ZGUgPSBGQUxTRX0KbGlicmFyeSh0aWR5dmVyc2UpCmBgYAoKYGBge3IgY2h1bmstc2V0dXAsIGluY2x1ZGUgPSBGQUxTRX0Ka25pdHI6Om9wdHNfY2h1bmskc2V0KGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRSwgbWVzc2FnZSA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICAjcmVzdWx0cyA9ICJob2xkIiwKICAgICAgICAgICAgICAgICAgICAgIGNhY2hlID0gRkFMU0UsIGNhY2hlLnBhdGggPSAiL2NhY2hlcy8iLCBjb21tZW50ID0gIiM+IiwKICAgICAgICAgICAgICAgICAgICAgICNmaWcud2lkdGggPSA3LCAjZmlnLmhlaWdodD0gNywKICAgICAgICAgICAgICAgICAgICAgICNvdXQud2lkdGggPSA3LCBvdXQuaGVpZ2h0ID0gNywKICAgICAgICAgICAgICAgICAgICAgIGNvbGxhcHNlID0gVFJVRSwgIGZpZy5zaG93ID0gImhvbGQiLAogICAgICAgICAgICAgICAgICAgICAgZmlnLmFzcCA9IDcvOSwgb3V0LndpZHRoID0gIjYwJSIsIGZpZy5hbGlnbiA9ICJjZW50ZXIiKQoKIy0gcGFyYSBtZWpvcmFyIGxvcyBncsOhZmljb3MsIGJ1ZW5vIGVuIHJlYWxpZGFkIHBhcmEgcXVlIHNlIHZlYW4gaWd1YWwgZW4gZGlzdGludG9zIFNPCiMtIGh0dHBzOi8vd3d3Lmp1bXBpbmdyaXZlcnMuY29tL2Jsb2cvci1rbml0ci1tYXJrZG93bi1wbmctcGRmLWdyYXBoaWNzLwprbml0cjo6b3B0c19jaHVuayRzZXQoZGV2ID0gInBuZyIsIGRldi5hcmdzID0gbGlzdCh0eXBlID0gImNhaXJvLXBuZyIpKQpgYGAKCmBgYHtyIG9wdGlvbnMtc2V0dXAsIGluY2x1ZGUgPSBGQUxTRX0Kb3B0aW9ucyhzY2lwZW4gPSA5OTkpICMtIHBhcmEgcXVpdGFyIGxhIG5vdGFjacOzbiBjaWVudMOtZmljYQpvcHRpb25zKCJ5YW1sLmV2YWwuZXhwciIgPSBUUlVFKSAjLSBodHRwczovL2dpdGh1Yi5jb20vdmlraW5nL3IteWFtbC9pc3N1ZXMvNDcgIChsbyBwdXNlIHggZWwgcGIgY29uIGVsIHdhcm5pbmcpIEVuIHJlYWxpZGFkIGNyZW8gcXVlIG1lam9yIHNlcsOtYSBwb25lcmxvIGVuIFJQcm9maWxlCmBgYAoKCmBgYHtyIGtsaXBweSwgZWNobyA9IEZBTFNFfQprbGlwcHk6OmtsaXBweShwb3NpdGlvbiA9IGMoInRvcCIsICJyaWdodCIpKSAjLSByZW1vdGVzOjppbnN0YWxsX2dpdGh1Yigicmxlc3VyL2tsaXBweSIpCmBgYAoKCmBgYHtyLCBpbmNsdWRlID0gRkFMU0V9Cm9wdGlvbnMoaHRtbHRvb2xzLmRpci52ZXJzaW9uID0gRkFMU0UpCiNrbml0cjo6b3B0c19jaHVuayRzZXQoZmlnLnJldGluYSA9IDMsIHdhcm5pbmcgPSBGQUxTRSwgbWVzc2FnZSA9IEZBTFNFLCBlY2hvPUZBTFNFLCBvdXQud2lkdGggPSAiODUlIikKbGlicmFyeShtZXRhdGhpcykKbWV0YSgpICU+JSBtZXRhX25hbWUoImdpdGh1Yi1yZXBvIiA9ICJwZXJlenA0NC9pbnRyby1kcy0yMC0yMS13ZWIiKSAlPiUgCiAgbWV0YV9zb2NpYWwoCiAgICB0aXRsZSA9ICJEYXRhIG11bmdpbmc6IG1hbmVqbyBkZSBkYXRvcyBjb24gUiwgdGhlIHRpZHl2ZXJzZSB3YXkuIiwKICAgIGRlc2NyaXB0aW9uID0gcGFzdGUoCiAgICAgICJTZSBwcmVzZW50YW4gbG9zIHByaW5jaXBhbGVzIHBhcXVldGVzIGRlbCB0aWR5dmVyc2UgcGFyYSBhcnJlZ2xhciB5IG1hbmVqYXIgZGF0b3MsIG9zIHBhcXVldGVzIHRpZHlyIHkgZHBseXIuIiwKICAgICAgIlNlIHByZXNlbnRhIGVsIG9wZXJhZG9yIHBpcGUsIGxhcyBmdW5jaW9uZXMgcGl2b3RfbG9uZ2VyKCkgeSBwaXZvdF93aWRlcigpIHBhcmEgaGFjZXIgbG9zIGRhdG9zIHRpZHksIHkgbGFzIHByaW5jaXBhbGVzIGZ1bmNpb25lcyBkZSBkcGx5ciwgaWNsdWlkYXMgbGFzIG51ZXZhcyBmdW5jaW9uZXMgYWNyb29zKCkgeSB3aGVyZSgpLiIpLAogICAgdXJsID0gImh0dHBzOi8vcGVyZXpwNDQuZ2l0aHViLmlvL2ludHJvLWRzLTIwLTIxLXdlYi90dXRvcmlhbGVzL3R0XzA1X2RhdGEtbXVuZ2luZy5odG1sIiwKICAgIG9nX3R5cGUgPSAid2Vic2l0ZSIsCiAgICBvZ19hdXRob3IgPSAiUGVkcm8gSi4gUMOpcmV6IgogICkKYGBgCgoKCmBgYHtyIGNhcmdhcl9wa2dzLCBlY2hvID0gRn0KI2xpYnJhcnkoInBlcnNvbmFsLnBqcCIpCmxpYnJhcnkoInRpZHl2ZXJzZSIpCmxpYnJhcnkoImhhdmVuIikKI2xpYnJhcnkoInhsc3giKQpsaWJyYXJ5KCJmb3JlaWduIikKbGlicmFyeSgicmVhZHhsIikKbGlicmFyeSgiZ2FwbWluZGVyIikKYGBgCgoKLS0tLS0tLS0tLS0tLS0tLS0KCjxicj4KCiMgMS4gSW50cm9kdWNjacOzbgoKPGJyPgoKRW4gZWwgdHV0b3JpYWwgYW50ZXJpb3IgYXByZW5kaW1vcyBhIGNhcmdhciBkYXRvcyBlbiBSLiBTaW4gZW1iYXJnbywgZXMgZGlmw61jaWwgcXVlIGVuIHVuYSBhcGxpY2FjacOzbiByZWFsIHRlbmdhbW9zIGxvcyBkYXRvcyB0YWwgeSBjb21vIGxvcyBuZWNlc2l0YW1vcyBwYXJhIGhhY2VyIG51ZXN0cm8gYW7DoWxpc2lzLiBIYWJpdHVhbG1lbnRlIHRlbmRyZW1vcyBxdWUgdHJhYmFqYXIgbG9zIGRhdG9zIHBhcmEgYXJyZWdsYXJsb3MuIEVzdGUgcHJvY2VzbywgcXVlIGVuIGNhc3RlbGxhbm8gcG9kcsOtYSBsbGFtYXJzZSAibGltcGllemEiIG8gKipwcm9jZXNhZG8gZGUgZGF0b3MqKiwgc2UgY29ub2NlIGVuIGluZ2zDqXMgY29tbyAqKmRhdGEgbXVuZ2luZyBvciBkYXRhIHdyYW5nbGluZyoqLgoKRW4gZWwgY3Vyc28gdmFtb3MgYSB0cmFiYWphci9tYW5lamFyIGxvcyBkYXRvcyB1c2FuZG8gdW4gY29uanVudG8gZGUgcGFxdWV0ZXMgYXNvY2lhZG9zIGFzb2NpYWRvcyBjb24gZWwgZW5mb3F1ZSBjb25vY2lkbyBjb21vICoqdGlkeXZlcnNlKiouIENvbW8gcHVlZGVzIHZlciBlbiBsYSBpbWFnZW4sIHlhIGhlbW9zIGltcG9ydGFkbyBsb3MgZGF0b3MgeSwgYW50ZXMgZGUgZW1wZXphciBhIGhhY2VyIGVsIHZlcmRhZGVybyBhbsOhbGlzaXMsIHRlbmVtb3MgcXVlIHBhc2FyIHBvciAyIGV0YXBhcyBtw6FzOiAKCiAgLSBoYWNlciBudWVzdHJvcyBkYXRvcyB0aWR5CiAgLSBhcnJlZ2xhcmxvcyBwYXJhIHF1ZSBzZWFuIMO6dGlsZXMgcGFyYSBudWVzdHJvcyBwcm9ww7NzaXRvcwoKPGJyPgoKCmBgYHtyICwgZWNobz1GQUxTRSwgZmlnLmNhcD0iKipEYXRhIHdyYW5nbGluZyoqIGZyb20gaHR0cDovL3I0ZHMuaGFkLmNvLm56L3dyYW5nbGUtaW50cm8uaHRtbCIsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiODAlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18wMV9kYXRhLXdyYW5nbGUucG5nIikpCmBgYAoKPGJyPgoKClNlIHN1ZWxlIGRlY2lyIHF1ZSBlbCBwcm9jZXNhZG8vbGltcGllemEgZGUgbG9zIGRhdG9zIHN1ZWxlIG9jdXBhciB1biA4MCUgZGVsIHRpZW1wbyBkZSB1biBhbsOhbGlzaXMgZGUgZGF0b3MuIFF1aXrDoXMgc2VhIHVuYSBjaWZyYSB1biBwb2NvIGV4YWdlcmFkYSwgcGVybywgZW4gY3VhbHF1aWVyIGNhc28sIGVzIHVuYSB0YXJlYSBxdWUgb2N1cGEgdGllbXBvIHkgcXVlIHB1ZWRlIGxsZWdhciBhIHNlciB0ZWRpb3NhIHkgZnJ1c3RyYW50ZSBzaSBubyBzZSBkaXNwb25lIGRlIGxhcyAqKmhlcnJhbWllbnRhcyBhZGVjdWFkYXMqKi4gSW5jbHVzbyBkYXRvcyBxdWUgcGFyZWNlbiBxdWUgeWEgZXN0w6FuIHRyYWJhamFkb3MgZXMgYmFzdGFudGUgZsOhY2lsIHF1ZSB0ZW5nYW1vcyBxdWUgdHJhYmFqYXJsb3MgcGFyYSBhZGFwdGFybG9zIGEgbnVlc3RyYXMgbmVjZXNpZGFkZXMuCgoKYGBgcgpVbmEgc2VjdWVuY2lhICJyZWFsIiBkZSB0d2VldHM6CgowOTMwOiBIb3cgbHVja3kgSSBhbSB0byB3b3JrIG9uIGNsZWFuIGRhdGFzZXRzIGN1cmF0ZWQgYnkgdHJ1ZSBwcm9mZXNzaW9uYWxzLgoxMzMwOiBIdWguIFNvbWUgaW5jb25zaXN0ZW5jaWVzIGhlcmUuIE5vIGJpZ3MuIElsbCBqdXN0IHdyaXRlIHVwIHNvbWUgcXVpY2sgYW5kIGRpcnR5IHJlZ2V4IHRvIGNsZWFuIHRoaXMgdXAuCgoxNzIwOiBJIFdJTEwgQlVSTiBUSElTIEhFUkVUSUNBTCBEQVRBIENFTlRFUiBBTkQgU0NBVFRFUiBJVFMgQVNIRVMgKHRyYWR1Y2Npw7NuOiBtZSBjYWdvIGVuIHRvZG8gbG8gcXVlIHNlIG1lbmVhKQpgYGAKCkVuIGNsYXNlIHV0aWxpemFtb3MgZGF0b3MgcmVhbGVzLCBwZXJvIGxhIHZlcmRhZCBlcyBxdWUgc3VlbGVuIHlhIGVzdGFyIGNhc2kgbGltcGlvcyBkZWwgdG9kby4gRXN0YW1vcyBhcHJlbmRpZW5kby4KCj4gQ2xhc3Nyb29tIGRhdGEgYXJlIGxpa2UgdGVkZHkgYmVhcnM7IHJlYWwgZGF0YSBhcmUgbGlrZSBhIGdyaXp6bHkgd2l0aCBzYWxtb24gYmxvb2QgZHJpcHBpbmcgb3V0IGl0cyBtb3V0aC4gLS0tLSAgW1xASmVubnlCcnlhbl0KCgpDb21vIGRpY2UgQWxiZXJ0IFkuIEtpbSBlbiBbZXN0YXMgdHJhbnNwYXJlbmNpYXNdKGh0dHA6Ly9ycHVicy5jb20vcnVkZWJveWJlcnQvZUNPVFNfMjAxOCkgbG9zIGRhdG9zIHV0aWxpemFkb3MgcGFyYSBhcHJlbmRlciBhIG1hbmVqYXIgZGF0b3MgdGllbmVuIHF1ZSBzZXIgcmVhbGlzdGFzIHBlcm8gc2luIGxsZWdhciBhIHNlciBpbnRpbWlkYW50ZXMuCgpgYGB7ciBlY2hvID0gRkFMU0UsIG91dC53aWR0aCA9ICIxMjAlIiwgZXZhbCA9IFRSVUV9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoIi4vaW1hZ2VuZXMvdHRfMDVfaW1nXzAyX2dyaXpsbHktdnMtdGVkZHkuanBnIikpCmBgYAoKCjxicj4KCgpFbiBlc3RlIHR1dG9yaWFsIGFwcmVuZGVyZW1vcyBhIGxpbXBpYXIgeSB0cmFuc2Zvcm1hciBkYXRvcyBlbiBSLiBQcmlvcml6YXJlbW9zIGxhIG51ZXZhIGZvcm1hIGRlIGhhY2VyIGxhcyBjb3NhcyBlbiBSIChvIHdvcmtmbG93KSBjb25vY2lkbyBjb21vICoqdGlkeXZlcnNlKiouIEVuIGxvcyDDumx0aW1vcyBhw7FvcyBzZSBoYSBjb252ZXJ0aWRvLCBwb3IgdmFyaWFzIHJhem9uZXMsIGVuIGVsIGVuZm9xdWUgZXN0w6FuZGFyOyBhIHBlc2FyIGRlIGVsbG8sIGNhZGEgY2llcnRvIHRpZW1wbyB2dWVsdmUgYSByZWFicmlyc2UgZWwgZGViYXRlIHNvYnJlIGPDs21vIGVuc2XDsWFyL2FwcmVuZGVyIFIgeSBzaSBlcyBhcHJvcGlhZG8gcHJpb3JpemFyIGVsIHRpZHl2ZXJzZSBzb2JyZSBSLWJhc2UuIFtBcXXDrV0oaHR0cHM6Ly90d2l0dGVyLmNvbS9rYWlqYV9iZWFuL3N0YXR1cy8xMjE3MjkzMzk2NzA2MDU0MTQ1KSB0aWVuZXMgdW4gaGlsbyBkZSB0d2l0dGVyIGRvbmRlIHNlIGRlYmF0ZSBzb2JyZSBlc3RlIHRlbWEuCgo8YnI+CgoKW0FxdcOtXShodHRwOi8vd3d3Lm9udGhlbGFtYmRhLmNvbS8yMDE0LzAyLzEwL2hvdy1kcGx5ci1yZXBsYWNlZC1teS1tb3N0LWNvbW1vbi1yLWlkaW9tcy8pIHRlbsOpaXMgdW4gcG9zdCBzb2JyZSBsYXMgZGlmZXJlbmNpYXMgZW50cmUgbGFzIGZ1bmNpb25lcyBkZSBSLWJhc2UgeSBsYXMgZGVsIHRpZHl2ZXJzZSBwYXJhIGVsIHByb2Nlc2FkbyBkZSBkYXRvcywgeSBbYXF1w61dKGh0dHA6Ly96ZXZyb3NzLmNvbS9ibG9nLzIwMTUvMDEvMTMvYS1uZXctZGF0YS1wcm9jZXNzaW5nLXdvcmtmbG93LWZvci1yLWRwbHlyLW1hZ3JpdHRyLXRpZHlyLWdncGxvdDIvKSBvdHJvIHBvc3QgZGUgdW4gbnVldm8gY29udmVuY2lkbyBkZSBsYXMgYm9uZGFkZXMgZGUgZXN0YSBudWV2YSBmb3JtYSBkZSBtYW5pcHVsYXIgZGF0b3MgZW4gUi4gQ29tbyBlamVtcGxvOgoKCj4gVXAgdW50aWwgbGFzdCB5ZWFyIG15IFIgd29ya2Zsb3cgd2FzIG5vdCBkcmFtYXRpY2FsbHkgZGlmZmVyZW50IGZyb20gd2hlbiBJIHN0YXJ0ZWQgdXNpbmcgUiBtb3JlIHRoYW4gMTAgeWVhcnMgYWdvLiBUaGFua3MgdG8gc2V2ZXJhbCBSIHBhY2thZ2UgYXV0aG9ycywgbW9zdCBub3RhYmx5IEhhZGxleSBXaWNraGFtLCBteSB3b3JrZmxvdyBoYXMgY2hhbmdlZCBmb3IgdGhlIGJldHRlciB1c2luZyBkcGx5ciwgbWFncml0dHIsIHRpZHlyIGFuZCBnZ3Bsb3QyLiBHaXZlbiBob3cgbXVjaCBJJ3ZlIGVuam95ZWQgdGhlIHNwZWVkIGFuZCBjbGFyaXR5IG9mIHRoZSBuZXcgd29ya2Zsb3cKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyAgVGlkeXZlcnNlCgojIyMjIMK/UXXDqSBlcyBlc3RvIGRlbCB0aWR5dmVyc2U/Cgo8YnI+CgpDb24gbGEgcGFsYWJyYSB0aWR5dmVyc2Ugc2UgaGFjZSByZWZlcmVuY2lhIGEgdW5hICJudWV2YSIgZm9ybWEgZGUgYWZyb250YXIgZWwgYW7DoWxpc2lzIGRlIGRhdG9zIGVuIFIgZW4gbGEgcXVlIHNlICBoYWNlIHVzbyBkZSAqKnVuIGdydXBvIGRlIHBhcXVldGVzIHF1ZSB0cmFiYWphbiBlbiBhcm1vbsOtYSoqIHBvcnF1ZSBjb21wYXJ0ZW4gY2llcnRvcyBwcmluY2lwaW9zLCBjb21vIHBvciBlamVtcGxvLCBsYSBmb3JtYSBkZSBlc3RydWN0dXJhciBsb3MgZGF0b3MuIAoKTGEgbWF5b3LDrWEgZGUgZXN0b3MgcGFxdWV0ZXMgaGFuIHNpZG8gZGVzYXJyb2xsYWRvcyBwb3IgKG8gYWwgbWVub3MgY29uIGxhIGNvbGFib3JhY2nDs24gZGUpIFtIYWRsZXkgV2lja2hhbV0oaHR0cDovL2hhZGxleS5uei8pLiBFc3RhIGVzIGxhIFtww6FnaW5hIHdlYiBkZWwgdGlkeXZlcnNlXShodHRwczovL3d3dy50aWR5dmVyc2Uub3JnLykKCgpObyBlcyBuZWNlc2FyaW8sIHBlcm8gc2kgcXVpZXJlcyBjb25vY2VyIHVuIHBvY28gbWVqb3IgcXXDqSBlcyBlbCB0aWR5dmVyc2UsIHB1ZWRlcyBoYWNlcmxvIGxleWVuZG8gW1RoZSB0aWR5IHRvb2xzIG1hbmlmZXN0b10oaHR0cHM6Ly9jcmFuLnItcHJvamVjdC5vcmcvd2ViL3BhY2thZ2VzL3RpZHl2ZXJzZS92aWduZXR0ZXMvbWFuaWZlc3RvLmh0bWwpLiBFc3TDoSBjaXRhIGVzIHVuIGJ1ZW4gcmVmZXJlbnRlIGRlIGxhIGZpbG9zb2bDrWEgbyBlbmZvcXVlIGRlbCB0aWR5dmVyc2UKCj4gUHJvZ3JhbXMgbXVzdCBiZSB3cml0dGVuIGZvciBwZW9wbGUgdG8gcmVhZCwgYW5kIG9ubHkgaW5jaWRlbnRhbGx5IGZvciBtYWNoaW5lcyB0byBleGVjdXRlICAtLSBIYWwgQWJlbHNvbgoKCjxicj4KClBhcmEgY29udGludWFyIGVudGVuZGllbmRvIHF1w6kgZXMgZXN0byBkZWwgdGlkeXZlcnNlLCBjaXRhcsOpIDIgZGUgc3VzIHByaW5jaXBpb3M6CgogIC0gTG9zIHNjcmlwdHMgZGViZW4gc2VyICoqImbDoWNpbG1lbnRlIiBsZWdpYmxlcyBwb3IgbGFzIHBlcnNvbmFzKiogIAogIAogIC0gKipSZXNvbHZlciBwcm9ibGVtYXMgY29tcGxlam9zKiogZW5jYWRlbmFuZG8gZnVuY2lvbmVzIHNpbXBsZXMgY29uIGVsICoqb3BlcmFkb3IgYCU+JWAqKgoKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCgojIyBUaGUgcGlwZSAoYCAlPiUgYCkKCkVzdGUgb3BlcmFkb3Igb2N1cGEgdW4gbHVnYXIgZnVuZGFtZW50YWwgZW4gZWwgdGlkeXZlcnNlLiBQZXJtaXRlIHJlc29sdmVyIHVuIHByb2JsZW1hIGNvbXBsZWpvIG5vIGRlIHVuYSBzb2xhIHZleiwgc2lubyBlbmNhZGVuYW5kbyBsbGFtYWRhcyBhIGZ1bmNpb25lcyBxdWUgc2UgdmFuIGVuY2FkZW5hbmRvIGNvbiBlbCBvcGVyYWRvciBgJT4lYC4gRXN0ZSBvcGVyYWRvciBmYWNpbGl0YSBtdWNobyBsYSBsZWN0dXJhIGUgaW50ZXJwcmV0YWNpw7NuIGRlbCBjw7NkaWdvLCB5YSBxdWUgc2UgdmFuIGVuY2FkZW5hbmRvIG9wZXJhY2lvbmVzIHNlbmNpbGxhcyBwYXJhLCBwb2NvIGEgcG9jbywgY29uc2VndWlyIHRyYW5zZm9ybWFjaW9uZXMgZGUgZGF0b3MgY29tcGxlamFzLiBFbCAqKm9wZXJhZG9yIHBpcGUqKiBzZSBsbyBkZWJlbW9zIGEgU3RlZmFuIEJhY2hlIGVuIHN1IHBrZyBbbWFncml0dHJdKGh0dHBzOi8vZ2l0aHViLmNvbS90aWR5dmVyc2UvbWFncml0dHIpLgoKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiMjAlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18wM19vcGVyYWRvci1waXBlLnBuZyIpKQpgYGAKCjxicj4KCkVuIHBhbGFicmFzLCBsbyBxdWUgaGFjZSBlc3RlIG9wZXJhZG9yIGVzICoqcGFzYXIgZWwgZWxlbWVudG8gcXVlIGVzdMOhIGEgc3UgaXpxdWllcmRhIGNvbW8gdW4gYXJndW1lbnRvIGRlIGxhIGZ1bmNpw7NuIHF1ZSB0aWVuZSBhIGxhIGRlcmVjaGEqKi4gQXPDrSBhbCBwcmluY2lwaW8gcGFyZWNlIGNvbXBsaWNhZG8uIAoKCioqQ29uIGV4cHJlc2lvbmVzKiogZWwgb3BlcmFkb3IgcGlwZSBoYWNlOiAKYGBge3IsIGVjaG8gPSBGQUxTRSwgZXZhbCA9IFRSVUV9CmFhIDwtIGRhdGEuZnJhbWUoeCA9IGMoImYob2JqZWN0LCBhcmd1bWVudG9zIGRlIGxhIGZ1bmNpw7NuKSIsICIiKSwgIHkgPSBjKCJFUyBFUVVJVkFMRU5URSBhIiwgIi0tIiksIHogPSBjKCJvYmplY3QgICU+JSAgZihhcmd1bWVudG9zIGRlIGxhIGZ1bmNpw7NuKSIsICIiKSApCmtuaXRyOjprYWJsZShhYSxjb2wubmFtZXMgPSBjKCIgIiwgIi0tIiwgIiAiKSwgYWxpZ24gPSAiYyIpCmBgYAoKCgo8YnI+CgoqKlNlIGVudGllbmRlIG1lam9yIGNvbiBlamVtcGxvcyBzZW5jaWxsb3MqKi4gTGFzIHNpZ3VpZW50ZXMgZG9zIGluc3RydWNjaW9uZXMgZGUgUiAqKmhhY2VuIGV4YWN0YW1lbnRlIGxvIG1pc21vKio6IHBlcm1pdGVuIHZlciBsYXMgNCBwcmltZXJhcyBmaWxhcyBkZWwgYHBlbmd1aW5zYCBkYXRhc2V0LgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmxpYnJhcnkocGFsbWVycGVuZ3VpbnMpCgpoZWFkKHBlbmd1aW5zLCBuID0gNCkgICAgICAgICAjLSBmb3JtYSBoYWJpdHVhbCBkZSBsbGFtYXIvdXNhciBsYSBmdW5jacOzbiBoZWFkKCkKCnBlbmd1aW5zICU+JSBoZWFkKC4gLCBuID0gNCkgICMtIHVzYW5kbyBlbCBvcGVyYWRvciBwaXBlCmBgYAoKCkxhIHByaW1lcmEgZXhwcmVzacOzbiBlcyBsYSBtYW5lcmEgaGFiaXR1YWwgZGUgdXNhci9sbGFtYXIgYSBsYSBmdW5jacOzbiBgaGVhZCgpYC4gTGEgc2VndW5kYSBleHByZXNpw7NuIGVzIGxhIHNpbnRheGlzLCBsYSBmb3JtYSBxdWUgaGF5IHF1ZSB1c2FyLCBzaSB0cmFiYWphbW9zIGNvbiBlbCBvcGVyYWRvciBwaXBlLgoKCkFzw60sIGEgcHJpbWVyYSB2aXN0YSwgcGFyZWNlIHF1ZSBlbCBvcGVyYWRvciBgJT4lYCBubyBzdXBvbmUgbmluZ3VuYSB2ZW50YWphLCBzw7NsbyBlcyB1bmEgZm9ybWEgZGlzdGludGEgZGUgZWplY3V0YXIgbyBsbGFtYXIgYSB1bmEgZnVuY2nDs24sIHkgYSBwcmltZXJhIHZpc3RhIHBhcmVjZSBjb21wbGljYXIgbGFzIGNvc2FzLiBTw60sIGVzbyBlcyBjaWVydG8sIHNpIHNvbG8gdXNhcyB1bmEgZnVuY2nDs24gbm8gdGVuZHLDrWEgbXVjaG8gc2VudGlkbyB1c2FyIGAlPiVgLCBwZXJvIGN1YW5kbyB0aWVuZXMgcXVlIGhhY2VyIHVuYSBzdWNlc2nDs24gZGUgY8OhbGN1bG9zLCB1bmEgc3VjZXNpw7NuIGRlIGxsYW1hZGFzIGEgZnVuY2lvbmVzLCBmYWNpbGl0YSBtdWNobyBsYSBsZWN0dXJhIGRlbCBjw7NkaWdvIHkgcG9yIHRhbnRvIGVsIGFuw6FsaXNpcy4gTG8gdmVtb3MgZW5zZWd1aWRhLgoKUGFyYSBlbnRlbmRlciB1biBwb2NvIG3DoXMgZWwgZnVuY2lvbmFtaWVudG8gZGUgYCU+JWAsIGhhcyBkZSB2ZXIgcXVlIGVzdGFzIHRyZXMgaW5zdHJ1Y2Npb25lcyBzb24gZXF1aXZhbGVudGVzLgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmhlYWQocGVuZ3VpbnMsIG4gPSA0KSAgICAgICAgICMtIGZvcm1hIGhhYml0dWFsIGRlIGxsYW1hci91c2FyIGxhIGZ1bmNpw7NuIGhlYWQoKQoKcGVuZ3VpbnMgJT4lIGhlYWQoLiAsIG4gPSA0KSAgIy0gdXNhbmRvIGVsIG9wZXJhZG9yIHBpcGUgKGNvbiBlbCBwdW50byBhY3R1YW5kbyBjb21vIHBsYWNlaG9sZGVyKQoKcGVuZ3VpbnMgJT4lIGhlYWQobiA9IDQpICAgICAgIy0gdXNhbmRvIGVsIG9wZXJhZG9yIHBpcGUgKFNJTiBlbCBwdW50bykKYGBgCgpFbCBwdW50byBkZSBsYSBzZWd1bmRhIGV4cHJlc2nDs24gc2XDsWFsYSwgbGUgZGljZSBhIHRoZSBwaXBlIGRvbmRlIGRlYmUgc2l0dWFyc2UgZWwgYXJndW1lbnRvIGRlIGxhIGl6cXVpZXJkYSBkZW50cm8gZGUgbGEgZnVuY2nDs247IGVuIG51ZXN0cm8gZWplbXBsbyBsZSBkaWNlIGEgYCU+JWAgcXVlIGBwZW5ndWluc2AgZGViZSBzaXR1YXJzZSBlbiBlbCBwcmltZXIgc2xvdCBkZSBgaGVhZCgpYC4gRWwgcHVudG8gYC5gIGxlIGVzdMOhIGRpY2llbmRvIGEgYCU+JWAgZG9uZGUgZGViZSBzaXR1YXJzZSBgcGVuZ3VpbnNgOyBlcyBkZWNpciwgZWwgcHVudG8gYWN0w7phLCBsbyBlc3RhbW9zIHVzYW5kbywgY29tbyB1biAicGxhY2Vob2xkZXIiLgoKTGEgdGVyY2VyYSBleHByZXNpw7NuIHRhbWJpw6luIGZ1bmNpb25hIHBvcnF1ZSBzaSBubyB1c2Ftb3MgZWwgcHVudG8gKGAuYCksIGVudG9uY2VzLCBwb3IgZGVmZWN0bywgZWwgb3BlcmFkb3IgcGlwZSBzaXR1YXLDoSBgcGVuZ3VpbnNgIGVuIGVsIHByaW1lciBzbG90IGRlIGxhIGZ1bmNpw7NuLCBlbiBudWVzdHJvIGNhc28gc2l0dWFyw6EgYSBgcGVuZ3VpbnNgIGVuIGVsIHByaW1lciBzbG90IGRlIGBoZWFkKClgLgoKTGEgZm9ybWEgbcOhcyBoYWJpdHVhbCBlcyBubyBwb25lciBlbCBgLmA7IGVzIGRlY2lyLCBsYSB0ZXJjZXJhIGV4cHJlc2nDs24uIExhIHJhesOzbiBlcyBzaW1wbGVtZW50ZSBxdWUgc2UgYWhvcnJhIHRpZW1wbyBhbCBlc2NyaWJpciwgYXVucXVlIGxhIHNlZ3VuZGEgZXhwcmVzacOzbiBlcyBtdWNobyBtw6FzIGV4cGxpY2l0YSwgbcOhcyBkZXNjcmlwdGl2YSwgZGUgbG8gcXVlIGhhY2UgZWwgb3BlcmFkb3IgcGlwZS4KCjxicj4KClBhcmEgY2FzaSB0ZXJtaW5hciBkZSBlbnRlbmRlciBsYSBzaW50YXhpcyBkZWwgb3BlcmFkb3IgcGlwZS4gSW50ZW50YWQgdmVyIHNpIGVudGVuZMOpaXMgbGEgc2lndWllbnRlIGluc3RydWNjacOzbjoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQo0ICU+JSBoZWFkKHBlbmd1aW5zLCAuKQpgYGAKClNpIG5vIHNhYsOpaXMgbG8gcXVlIGhhY2UsIHNpZW1wcmUgcG9kw6lpcyBlamVjdXRhciBsYSBpbnN0cnVjY2nDs24gZW4gbGEgY29uc29sYSBkZSBSU3R1ZGlvLgoKUmVjdWVyZGE6IEN1YW5kbyB1c2Ftb3MgZWwgb3BlcmFkb3IgcGlwZSwgdGVuZW1vcyBvYmxpZ2F0b3JpYW1lbnRlIHF1ZSB1c2FyIGVsIHB1bnRvIHNpIHF1ZXJlbW9zIHF1ZSBlbCBhcmd1bWVudG8gZGUgbGEgaXpxdWllcmRhIHNlIHNpdMO6ZSBlbiB1biBzbG90IGRpZmVyZW50ZSBkZWwgcHJpbWVyIHNsb3QKCjxicj4KClBhcmEgYWNhYmFyIG51ZXN0cm8gcmVwYXNvIGEgYCU+JWAgbWlyYWQgcG9yIHF1w6kgbm8gZnVuY2lvbmEgbGEgc2lndWllbnRlIGluc3RydWNjacOzbjoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQo0ICU+JSBoZWFkKHBlbmd1aW5zKQpgYGAKCkVsIG9wZXJhZG9yIHBpcGUgcXVpZXJlIGxsZXZhciBlbCBgNGAgYWwgcHJpbWVyIHNsb3QgZGUgYGhlYWQoKWAgeWEgcXVlIHNpIG5vIHBvbmVtb3MgZWwgcHVudG8sIGVzZSBlcyBzdSBjb21wb3J0YW1pZW50byBwb3IgZGVmZWN0by4gU2luIGVtYmFyZ28sIGFsIGVqZWN1dGFyIGxhIGV4cHJlc2nDs24sIGVsIGludGVycHJldGUgZGUgUiBub3MgZGV2dWVsdmUgdW4gbWVuc2FqZSBkZSBlcnJvci4gwr9Qb3IgcXXDqT8gVGVuZHLDoXMgcXVlIG1pcmFyIGxhIGF5dWRhIGRlIGxhIGZ1bmNpw7NuIGNvbiBgaGVscChoZWFkKWAuCgo8YnI+CgpBw7puIG5vIHNhYmVtb3MgbXV5IGJpZW4gY3XDoWwgZXMgc3UgdXRpbGlkYWQsIHBlcm8geWEgY29ub2NlbW9zIGxhIHNpbnRheGlzIGRlIGAlPiVgLiBMbyBxdWUgaGFjZSBxdWUgZXN0ZSBvcGVyYWRvciBzZWEgdGFuIMO6dGlsIGVzIHF1ZSAqKmxhcyBwaXBlcyBzZSBwdWVkZW4gZW5jYWRlbmFyKiouIAoKPGJyPgoKRWwgb3BlcmFkb3IgcGlwZSBwb2RlbW9zIGxlZXJsbyBjb21vICoqIiplbnRvbmNlcyoiKiogeSBwZXJtaXRlIGVuY2FkZW5hciBzdWNlc2l2YXMgbGxhbWFkYXMgYSBmdW5jaW9uZXMuIFBvciBlamVtcGxvOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CnBlbmd1aW5zICU+JSBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgICAgICAgICAgICBncm91cF9ieShzcGVjaWVzKSAlPiUgCiAgICAgICAgICAgICBzdW1tYXJpc2UocGVzb19tZWRpbyA9IG1lYW4oYm9keV9tYXNzX2cpKQpgYGAKPGJyPgoKTGEgYW50ZXJpb3IgbGluZWEgZGUgY8OzZGlnbyBSIGhhY2U6CgogIDEpIGNvZ2UgbG9zIGRhdG9zIGRlIHBpbmfDvGlub3MgeSBzZWxlY2Npb25hIChvIGZpbHRyYSkgbGFzIGZpbGFzL3Bpbmd1aW5vcyBjdXlvIHZhbG9yIGRlIGxhIHZhcmlhYmxlIGBzZXhgIGVzIGZlbWFsZTsgZXMgZGVjaXIsIHNlbGVjY2lvbmFtb3MgbG9zIHBpbmfDvGlub3MgaGVtYnJhcywgKmVudG9uY2VzKiAobyBkZXNwdcOpcykgIAogIDIpIGFncnVwYSBsb3MgZGF0b3MvcGluZ8O8aW5vcyBwb3IgbGEgdmFyaWFibGUgYHNwZWNpZXNgLCAqZW50b25jZXMqICAKICAzKSBjYWxjdWxhIGxhIG1lZGlhIGRlIGBib2R5X21hc3NfZ2AKCkVuIGNvbmp1bnRvLCBlbmNhZGVuYW5kbyBsYXMgMyBmdW5jaW9uZXMgaGVtb3Mgc2VsZWNjaW9uYWRvIGxhcyBmaWxhcyBxdWUgcGVydGVuZWNlbiBhIHBpbmfDvGlub3MgaGVtYnJhcywgaGVtb3MgYWdydXBhZG8gbGFzIHBpbmfDvGlubyBoZW1icmFzIGVuIGZ1bmNpw7NuIGRlIHN1IGVzcGVjaWUgKGhheSAzIGVzcGVjaWVzIGRlIHBpbmfDvGlub3MpIHkgY2FsY3VsYWRvIGVsIHBlc28gbWVkaW8gZGUgY2FkYSB1bm8gZGUgbGFzIDMgZXNwZWNpZXMgZGUgcGluZ8O8aW5vczsgZXMgZGVjaXIsIGhlbW9zIGNhbGN1bGFkbyBlbCBwZXNvIG1lZGlvIGRlIGxhcyBwaW5nw7xpbm9zIGhlbWJyYSBlbiBjYWRhIHVubyBkZSBsYXMgdHJlcyBlc3BlY2llcyBkZSBwaW5nw7xpbm9zLgoKPGJyPgoKQ29uIGVzdGEgbnVldmEgc2ludGF4aXMgKHF1ZSBwZXJtaXRlIGVsIG9wZXJhZG9yIHBpcGUpIHlhIG5vIG5lY2VzaXRhbW9zIGFuaWRhciBmdW5jaW9uZXMsIHNpbm8gcXVlIGxhcyBpbnN0cnVjY2lvbmVzIHZhbiB1bmEgZGVzcHXDqXMgZGUgb3RyYS4gRXMgKiptdWNobyBtw6FzIGbDoWNpbCBkZSBsZWVyIHkgZGUgZXNjcmliaXIqKi4gRXN0YSBpZGVhIGRlIHF1ZSBlcyBtdWNobyBtw6FzIGbDoWNpbCBlc2NyaWJpciDDoCBsYSB0aWR5dmVyc2Ugbm8gc2UgbGxlZ2EgYSBhcHJlY2lhciBjb24gbG9zIGVqZW1wbG9zIHF1ZSBoZW1vcyBoZWNobyBlbiBlc3RhIHNlY2Npw7NuLCBwZXJvIHNlIGhhcsOhIGV2aWRlbnRlIGN1YW5kbyBlbXBlY2Vtb3MgYSBlbmNhZGVuYXIgb3BlcmFjaW9uZXMgY29uIGRwbHlyLiBDb21vIGVqZW1wbG8gZXN0ZSB0d2VldC4KCgoKYGBge3IsIGVjaG8gPSBGQUxTRX0KdHdlZXRybWQ6OnR3ZWV0X2VtYmVkKCJodHRwczovL3R3aXR0ZXIuY29tL2FuZHJld2hlaXNzL3N0YXR1cy8xMTczNzQzNDQ3MTcxMzU0NjI0IiwgdGhlbWUgPSAibGlnaHQiLCBhbGlnbiA9ICJjZW50ZXIiLCBtYXh3aWR0aCA9IDY1MCkKYGBgCgo8YnI+CgpObyB0ZSB2YSBhIHJlc3VsdGFyIHNlbmNpbGxvIHBvcnF1ZSBubyBzYWJlcyBxdWUgZXMgYGxldHRlcnNgLCBuaSBgcGFzdGUwKClgLCBuaSBgdG91cHBlcigpYCwgcGVybyBpbnRlbnRhIGVudGVuZGVyIHBvciB0aSBtaXNtbyBxdWUgaGFjZSBsYSBzaWd1aWVudGUgbGluZWEgZGUgY8OzZGlnby4geWEgc2FiZXMgcXVlIHNpZW1wcmUgcHVlZGVzIGVqZWN1dGFybGEgeSB2ZXIgcXVlIGhhY2UsIHkgbXVjaG8gbWVqb3Igc2kgbGEgZWplY3V0YXMgcG9yIHRyb3pvcyBwYXJhIGlyIHZpZW5kbyBwb2NvIGEgcG9jbyBxdcOpIGhhY2U6CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KbGV0dGVycyAlPiUgcGFzdGUwKCAiLS0tLS0iICwgIC4gICwgICIhISEiICkgJT4lIHRvdXBwZXIKYGBgCgo8YnI+CgojIyMjIFVuIHBvY28gbcOhcyBhY2VyY2EgZGUgdGhlIHBpcGUgKGAgJT4lIGApIFtPUENJT05BTF0KCgpEZSBmb3JtYSBtw6FzIHTDqWNuaWNhLiBBcXXDrSBwb2TDqWlzIHZlciBlbCBmdW5jaW9uYW1pZW50byBkZWwgb3BlcmFkb3IgcGlwZToKCmBgYHIKbGlicmFyeSgibWFncml0dHIiKQoKIy0tLS0tLS0tIFJ1bGUgMQpmKHh4KSAgICAgZXMgZXF1aXZhbGVudGUgYSAgICAgeHggJT4lIGYKCiMtLS0tLS0tLSBSdWxlIDIKZyh4eCwgbiA9IDUpCnh4ICU+JSBnKG4gPSA1KQoKIy0tLS0tLS0tIFJ1bGUgMwpnKGYoeHgpLCBuID0gNSkKeHggJT4lIGYgJT4lIGcobiA9IDUpCgpTZSBsZWUgY29tbyAiVGFrZSB4eCB0aGVuIGRvIGYgdGhlbiBkbyBnIHdpdGggbiA9IDUiLgoKIy0tLS0tLS0tIFJ1bGUgNApmKHksIHgpCnggJT4lIGYoeSwgLikKCiMtLS0tLS0tLSBSdWxlIDUKZih5LCB6ID0geCkKeCAlPiUgZih5LCB6ID0gLikKCiMoISEhISktLS0tLS0tLS0tLS0tIEJPTlVTOiBUaGUgaW5wdXQgdG8gdGhlIHBpcGVsaW5lIGNhbiBpdHNlbGYgYmUgYSBwbGFjZWhvbGRlciEhCm51bV91bmlxdWUgPC0gLiAlPiUgdW5pcXVlICU+JSBsZW5ndGggICAgICAgCgpudW1fdW5pcXVlKGlyaXMkU3BlY2llcykKCmlyaXMkU3BlY2llcyAlPiUgbnVtX3VuaXF1ZQoKLS0tLS0KCm51bV91bmlxdWUgZXMgZXF1aXZhbGVudGUgYSA6IGYgPC0gZnVuY3Rpb24oLikgbGVuZ3RoKHVuaXF1ZSguKSkgCgpgYGAKCjxicj4KClVuIGJ1ZW4gcmVjdXJzbyBwYXJhIGFwcmVuZGVyIGVsIHVzbyBkZSBgJT4lYCBzb24gW2VzdGFzIHRyYW5zcGFyZW5jaWFzXShodHRwczovL3RlYWNodGhhdC5uZXRsaWZ5LmFwcC9waXBlLyMxKS4gVW5hIGV4cG9zaWNpw7NuIG3DoXMgZGV0YWxsYWRhIGRlIGxhIHNpbnRheGlzIHkgcG9zaWJpbGlkYWRlcyBkZWwgb3BlcmFkb3IgYCU+JWAsIGFzw60gY29tbyBsYSBkZSBvdHJvcyBvcGVyYWRvcmVzIGNvbW8gYCVUPiVgIHkgYCU8PiVgIHB1ZWRlcyBlbmNvbnRyYXJsYSBbYXF1w61dKGh0dHBzOi8vd3d3LnJkb2N1bWVudGF0aW9uLm9yZy9wYWNrYWdlcy9tYWdyaXR0ci92ZXJzaW9ucy8xLjUpLgoKPGJyPgoKVGhlICoqInRlZSBwaXBlIioqIChgJVQ+JWApIHBlcm1pdGUgaGFjZXIgY29zYXMgY29tbyBlc3RhOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KIy0gISEhISEhCnJub3JtKDIwMCkgJT4lIG1hdHJpeChuY29sID0gMikgJVQ+JQpwbG90ICU+JSAjIHBsb3QgdXN1YWxseSBkb2VzIG5vdCByZXR1cm4gYW55dGhpbmcuIApjb2xTdW1zCmBgYAoKClRoZSAqKiJ0ZWUgcGlwZSIqKiwgY29tbyBlbCBwaXBlIG9yaWdpbmFsLCBwYXNhIGVsIGFyZ3VtZW50byBkZSBsYSBpenF1aWVyZGEgYSBsYSBmdW5jacOzbiBkZSBsYSBkZXJlY2hhLCBQRVJPIGRldnVlbHZlIGVsIHByb3BpbyB2YWxvciBvcmlnaW5hbCwgbm8gZGV2dWVsdmUgZWwgcmVzdWx0YWRvIGRlIGxhIGV2YWx1YWNpw7NuIGRlIGxhIGZ1bmNpw7NuLiBDb21vIHNlIHNlw7FhbGEgW2FxdcOtXShodHRwczovL21hZ3JpdHRyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL3RlZS5odG1sKSwgZXN0ZSBjb21wb3J0YW1pZW50byBlcyDDunRpbCBjdWFuZG8gc2UgdXNhIGxhIGZ1bmNpw7NuIHBvciBzdXMgc2lkZS1lZmZlY3RzOyBlcyBkZWNpciwgcGFyYSBpbXByaW1pciBvIGdyYWZpY2FyLiBZbyBsYSB1dGlsaWRhZCBxdWUgbGUgdmVvIGVzIGhhY2VyIGNoZXF1ZW9zIGRlbnRybyBkZSB1bmEgc2VjdWVuY2lhIGRlIHBpcGVzLCBjb21vIHBvciBlamVtcGxvIGhhY2VuIGVuIFtlc3RlIHR3ZWV0XShodHRwczovL3R3aXR0ZXIuY29tL2xlcG92YXMvc3RhdHVzLzEyODM0MTE3NzQxMzQ1MzgyNTEpCgo8YnI+ClRoZSAqKiJleHBvc2l0aW9uIHBpcGUiKiogKGAlJCVgKSB0YW1iacOpbiBkZWwgcGtnIGBtYWdyaXR0cmAuIEVuIFtlc3RlIHBvc3RdKGh0dHBzOi8vdGhld29vZHBlY2tyLndvcmRwcmVzcy5jb20vMjAyMC8wMi8xMC91cHBpbmcteW91ci1waXBlLWdhbWUvKSBub3MgZXhwbGljYW4gc3UgdXRpbGlkYWQuIFBlcm8gc29sbyBsZWVkbG8gY3VhbmRvIHlhIHNlw6FpcyB1c3VhcmlvcyBpbnRlcm1lZGlvcyBkZSBSLiBTaXJ2ZSBwYXJhIGhhY2VyIGFjY2VzaWJsZXMgbGFzIGNvbHVtbmFzIGRlIHVuIGRhdGFmcmFtZSBhIGZ1bmNpb25lcyBxdWUgbm8gYWRtaXRlbiBkYXRhZnJhbWVzIGNvbW8gbGEgZnVuY2nDs24gYGNvcigpYC4gRGUgZXN0YSBmb3JtYSBwb2RlbW9zIGludGVncmFyIGVuIG51ZXN0cm8gcGlwZWxpbmUgZnVuY2lvbmVzIHF1ZSBubyBlc3TDoW4gcHJlcGFyYWRhcyBwYXJhIGVsIHRpZHl2ZXJzZS4KCmBgYHtyLCBldmFsID0gRkFMU0V9CiMtICEhIQpsaWJyYXJ5KG1hZ3JpdHRyKQppcmlzICU+JSBtZWFuKFNlcGFsLkxlbmd0aCkgICAjLSBubyBmdW5jaW9uYQppcmlzICUkJSBtZWFuKFNlcGFsLkxlbmd0aCkgICAjLSBjb24gdGhlIGV4cG9zaXRpb24gcGlwZSBzw60gZnVuY2lvbmEKCmlyaXMgJT4lIGNvcihTZXBhbC5MZW5ndGgsIFNlcGFsLldpZHRoKSAgIy0gbm8gZnVuY2lvbmEKaXJpcyAlJCUgY29yKFNlcGFsLkxlbmd0aCwgU2VwYWwuV2lkdGgpCmBgYAoKU2kgbm8gdXPDoXNlbW9zIGVzdGUgbnVldmEgcGlwZSwgdGVuZHLDrWFtb3MgcXVlIGhhY2VyIGxvIHNpZ3VpZW50ZToKCmBgYHtyLCBldmFsID0gRkFMU0V9CmNvcihpcmlzJFNlcGFsLkxlbmd0aCwgaXJpcyRTZXBhbC5XaWR0aCkKYGBgCgo8YnI+CgpNdWNoYXMgZnVuY2lvbmVzIGRlIFItYmFzZSBubyBlc3TDoW4gcHJlcGFyYWRhcyBwYXJhIHRyYWJhamFyIGNvbiBlbCBvcGVyYWRvciBwaXBlLiBTb24gZnVuY2lvbmVzIHF1ZSBzZSBlc2NyaWJpZXJvbiBhbnRlcyBkZSBxdWUgc2UgY3JlYXJhIGAlPiVgLiBTZSBwdWVkZSB0cmF0YXIgZGUgcmVlc2NyaWJpciBjw7NkaWdvIGVuICJSLWJhc2UiIHVzYW5kbyBgICU+JSBgIHBlcm8gbm8gdGllbmUgbXVjaG8gc2VudGlkbyB5IG5vIGVzIG11eSBhZ3JhZGFibGU7IHNpbiBlbWJhcmdvIHNpIHNlIHRyYWJhamEgY29uIGVsIHRpZHl2ZXJzZSwgdXRpbGl6YXIgdGhlIHBpcGUgaGFjZSBsYSBzaW50YXhpcyBtdXkgZmx1aWRhLiBDb21vIGVqZW1wbG8gZGUgZXN0byBlbCBzaWd1aWVudGUgY2h1bms6CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KbGlicmFyeSh0aWR5dmVyc2UpICAgIAoKeDEgPC0gYygtNTo1LCBOQSkgICMtIGVzIHVuIHZlY3RvcgoKIy0gZXNjcmliaWVuZG8gw6AgbGEgUi1iYXNlCm1lYW4oeDFbeDE+MF0sIG5hLnJtID0gVFJVRSkgICAjLSAgY2FsY3VsYSBsYSBtZWRpYSBkZSBsb3MgdmFsb3JlcyBwb3NpdGl2b3MgZGUgeDEKc3VtKHgxWyFpcy5uYSh4MSldKSAgICAgICAgICAgICMtICBjYWxjdWxhIGxhIHN1bWEgZGUgbG9zIHZhbG9yZXMgZGUgeDEgcXVlIG5vIHNvbiBOQQoKIy0gYWhvcmEgaGFyZW1vcyBsbyBtaXNtbywgc2VndWltb3MgdXNhbmRvIFItYmFzZSwgcGVybyBjb24gZWwgb3BlcmFkb3IgcGlwZSAoISEhISkKeDEgJT4lIC5bLj4wXSAlPiUgbWVhbiguLCBuYS5ybSA9IFRSVUUpCngxICU+JSAuWyFpcy5uYSguKV0gICU+JSBzdW0KCiMtIHBvZHLDrWFtb3MgdHJhYmFqYXIgY29uIGRhdGEuZnJhbWVzIHVzYW5kbyB0aGUgZXhwb3NpdGlvbiBwaXBlCmRmIDwtIGFzLmRhdGEuZnJhbWUoeDEpICAgIy0gdGlkeXZlcnNlIHVzYSBkYXRhLmZyYW1lcwpkZiAlJCUgeDEgJT4lIC5bLj4wXSAlPiUgbWVhbiguLCBuYS5ybSA9IFRSVUUpCmRmICUkJSB4MSAlPiUgLlshaXMubmEoLildICAlPiUgc3VtCgojLSBjb24gdHlkaXZlcnNlCmRmIDwtIGFzLmRhdGEuZnJhbWUoeDEpICAgIy0gdGlkeXZlcnNlIHVzYSBkYXRhLmZyYW1lcwpkZiAlPiUgZmlsdGVyKHgxID4gMCkgJT4lIHN1bW1hcmlzZShtZWFuX3gxID0gbWVhbih4MSwgbmEucm0gPSBUUlVFKSkKZGYgJT4lIGZpbHRlcighaXMubmEoeDEpKSAlPiUgc3VtbWFyaXNlKHN1bWFfeDEgPSBzdW0oeDEpKQpgYGAKCgo8YnI+CgpCaWVuLCB5YSBzYWJlbW9zIGNvbW8gZnVuY2lvbmEgInRoZSBwaXBlIi4gVm9sdmFtb3MgYWwgdGlkeXZlcnNlIHkgYSBhcHJlbmRlciBhIG1hbmlwdWxhciBkYXRvcyBlbiBSLgoKCjxicj4KCiMjIFByaW5jaXBhbGVzIHBrZ3MgZGVsIHRpZHl2ZXJzZQoKCkNvbW8gcHVlZGUgdmVyc2UgZW4gc3UgW3DDoWdpbmEgd2ViXShodHRwczovL3d3dy50aWR5dmVyc2Uub3JnLyksIGxvcyBwcmluY2lwYWxlcyBwYWNrYWdlcyBkZWwgdGlkeXZlcnNlIHNvbjogCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiODAlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18wM2JfcGtncy10aWR5dmVyc2UucG5nIikpCmBgYAoKCgogIC0gYHJlYWRyYDogcGFyYSBpbXBvcnRhciBkYXRvcyAgCiAgLSBgdGlkeXJgOiBwYXJhIGNvbnZlcnRpciBsb3MgZGF0b3MgYSB0aWR5IGRhdGEgIAogIC0gYGRwbHlyYDogcGFyYSBtYW5pcHVsYXIgZGF0b3MgIAogIC0gYGdncGxvdDJgOiBwYXJhIGhhY2VyIGdyw6FmaWNvcyAgCiAgCiAgLSBgdGliYmxlYDogZGF0YSBmcmFtZXMgYWN0dWFsaXphZG9zCiAgICAKICAtIGBmb3JjYXN0YDogcGFyYSBtYW5pcHVsYXIgZmFjdG9yZXMgIAogIC0gYHN0cmluZ3JgOiBwYXJhIG1hbmlwdWxhciBzdHJpbmdzICAKICAKICAtIGBwdXJycmA6IHBhcmEgZnVuY3Rpb25hbCBwcm9ncmFtbWluZwoKICAgICAgCiAgLSB5IGFsZ3Vub3MgbcOhcyAgCgpOb3MgY2VudHJhcmVtb3MgZW4gbG9zICoqY3VhdHJvIHByaW1lcm9zIHBhcXVldGVzKiosIHByaW5jaXBhbG1lbnRlIGVuIGBkcGx5cmAgeSBgZ2dwbG90MmAuCgo8YnI+CgoKTG9zIHByaW5jaXBhbGVzIHBhcXVldGVzIGRlbCB0aWR5dmVyc2Ugc2UgaGFuICJhZ3J1cGFkbyIgZW4gdW4gbWV0YXBhcXVldGUgbGxhbWFkbyBgdGlkeXZlcnNlYCwgYXPDrSBxdWUgY3VhbmRvIGVqZWN1dGFzIGBsaWJyYXJ5KHRpZHl2ZXJzZSlgIGVuIHJlYWxpZGFkIGVzdMOhcyBjYXJnYW5kbyB2YXJpb3MgcGFxdWV0ZXMgZGVsIHRpZHl2ZXJzZQoKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0KCgojIDIuIFRpZHkgZGF0YSAodGlkeXIpCgo8YnI+Cgo+IElmIEkgaGFkIG9uZSB0aGluZyB0byB0ZWxsIGJpb2xvZ2lzdHMgbGVhcm5pbmcgYmlvaW5mb3JtYXRpY3MsIGl0IHdvdWxkIGJlIHdyaXRlIGNvZGUgZm9yIGh1bWFucywgKip3cml0ZSBkYXRhIGZvciBjb21wdXRlcnMqKi4gIC0tLS3igJQgVmluY2UgQnVmZmFsbyAoXEB2c2J1ZmZhbG8pCgoKWSBzaSB2YW1vcyBhIG1hbmVqYXIgZGF0b3MgY29uIFIgeSBhIGxhIG1hbmVyYSBkZWwgdGlkeXZlcnNlLCBjb21vIEplbm55IEJyeWFuIHNlw7FhbGEgZW4gc3UgZXhjZWxlbnRlIFt0dXRvcmlhbCBzb2JyZSB0aWR5IGRhdGFdKGh0dHBzOi8vZ2l0aHViLmNvbS9qZW5ueWJjL2xvdHItdGlkeS9ibG9iL21hc3Rlci8wMS1pbnRyby5tZCk6IAoKPiBBbiBpbXBvcnRhbnQgYXNwZWN0IG9mICJ3cml0aW5nIGRhdGEgZm9yIGNvbXB1dGVycyIgaXMgdG8gKiptYWtlIHlvdXIgZGF0YSBUSURZKiouIC0tLS0gSmVubnkgQnJ5YW4KCgo8YnI+CgpBbnRlcyBkZSBjb21lbnphciBhIG1hbmlwdWxhciBsb3MgZGF0b3Mgw6AgbGEgdGlkeXZlcnNlLCBlcyBjb252aWVuZSBzYWJlciBxdWUgc2UgZW50aWVuZGUgcG9yICoqdGlkeSBkYXRhKiouIExhIHJhesOzbiBlcyBxdWUgbG9zIHBhcXVldGVzIGRlbCB0aWR5dmVyc2UgdHJhYmFqYW4gbWVqb3Igc2kgbG9zIGRhdG9zIGVzdMOhbiBlbiBmb3JtYXRvIHRpZHkuIEVzIGbDoWNpbCEhCgoKPGJyPgoKIyMjIyDCv1F1w6kgc29uIGxvcyAqKnRpZHkgZGF0YSoqPwoKCkFob3JhIGxvIHZlcmVtb3MsIHBlcm8gZW5mYXRpemFyIHF1ZSBzaSBsb3MgZGF0b3Mgc29uIHRpZHkgKHNpIHNpZ3VlbiBlc2UgZm9ybWF0bykgKipzZXLDoSBtw6FzIGbDoWNpbCB0cmFiYWphciBjb24gZWxsb3MgY29uIGVsIHRpZHl2ZXJzZSoqLCB5YSBzZWEgcGFyYSBtYW5pcHVsYXJsb3MgbyBwYXJhIGhhY2VyIGdyw6FmaWNvcy4KCkRlIGZvcm1hIHNlbmNpbGxhLCB0aWR5IGRhdGEgc29uIHNpbXBsZW1lbnRlIGRhdG9zIG9yZ2FuaXphZG9zIGRlIHVuYSBkZXRlcm1pbmFkYSBtYW5lcmEuIEFkZW3DoXMgZXMganVzdG8gZGUgbGEgbWFuZXJhIGEgbGEgZXN0YW1vcyBmYW1pbGlhcml6YWRvcy4gRGUgZm9ybWEgbcOhcyBwcmVjaXNhIHNlIHB1ZWRlIGxlZXIgW2FxdcOtXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9UaWR5X2RhdGEpLCBvIGRlIGZvcm1hIG3DoXMgZWxhYm9yYWRhIFtbYXF1w61dKGZ0cDovL2NyYW4uci1wcm9qZWN0Lm9yZy9wdWIvUi93ZWIvcGFja2FnZXMvdGlkeXIvdmlnbmV0dGVzL3RpZHktZGF0YS5odG1sKV0gIDoKCj4gVGlkeSBkYXRhc2V0cyBwcm92aWRlIGEgc3RhbmRhcmRpemVkIHdheSB0byBsaW5rIHRoZSBzdHJ1Y3R1cmUgb2YgYSBkYXRhc2V0IChpdHMgcGh5c2ljYWwgbGF5b3V0KSB3aXRoIGl0cyBzZW1hbnRpY3MgKGl0cyBtZWFuaW5nKS4gICAtLS0tLSBIYWRsZXkgV2lja2hhbQoKPGJyPgoKTGEgbWF5b3LDrWEgZGUgZGF0b3MgZW4gQ2llbmNpYXMgU29jaWFsZXMgc2UgYWp1c3RhbiBhIGxhIGNhdGVnb3LDrWEgZGUgZGF0b3MgdGFidWxhcmVzOyBlcyAgZGVjaXIsICoqZXN0w6FuIG9yZ2FuaXphZG9zIGVuIGZpbGFzIHkgY29sdW1uYXMqKi4gRW4gUiBlc3RlIHRpcG8gZGUgZGF0b3Mgc2UgYWxtYWNlbmFuIGVuIGRhdGFmcmFtZXMgKG8gdGliYmxlcykuIEVuIGVzZW5jaWEsIHVuIGRhdGFmcmFtZSBzZXLDoSB0aWR5IHNpIGNhZGEgY29sdW1uYSBlcyB1bmEgdmFyaWFibGUgeSBjYWRhIGZpbGEgZXMgdW5hIHVuaWRhZCBkZSBhbsOhbGlzaXMgKHBlcnNvbmEsIHBhw61zLCByZWdpw7NuIGV0Yy4uLik7IGVzIGRlY2lyLCBjYWRhIGNlbGRhIGNvbnRpZW5lIGVsIHZhbG9yIGRlIHVuYSB2YXJpYWJsZSBwYXJhIHVuYSB1bmlkYWQgZGUgYW7DoWxpc2lzLgoKPiBBIGRhdGFzZXQgaXMgYSBjb2xsZWN0aW9uIG9mIHZhbHVlcy4gRXZlcnkgdmFsdWUgYmVsb25ncyB0byBhIHZhcmlhYmxlIGFuZCBhbiBvYnNlcnZhdGlvbi4gQSB2YXJpYWJsZSBjb250YWlucyBhbGwgdmFsdWVzIHRoYXQgbWVhc3VyZSB0aGUgc2FtZSB1bmRlcmx5aW5nIGF0dHJpYnV0ZSAobGlrZSBoZWlnaHQsIHRlbXBlcmF0dXJlLCBkdXJhdGlvbikgYWNyb3NzIHVuaXRzLiBBbiBvYnNlcnZhdGlvbiBjb250YWlucyBhbGwgdmFsdWVzIG1lYXN1cmVkIG9uIHRoZSBzYW1lIHVuaXQgKGxpa2UgYSBwZXJzb24sIG9yIGEgZGF5LCBvciBhIHJhY2UpIGFjcm9zcyBhdHRyaWJ1dGVzCgoKYGBge3IgLCBlY2hvPUZBTFNFLCBmaWcuY2FwPSJUaWR5IGRhdGEgZnJvbSBodHRwOi8vcjRkcy5oYWQuY28ubnovdGlkeS1kYXRhLmh0bWwiLCBldmFsID0gVFJVRSwgZmlnLmFzcCA9IDQvMiwgb3V0LndpZHRoID0gIjgwJSIsIGZpZy5hbGlnbiA9ICJjZW50ZXIifQprbml0cjo6aW5jbHVkZV9ncmFwaGljcyhoZXJlOjpoZXJlKCJpbWFnZW5lcyIsICJ0dF8wNV9pbWdfMDRfdGlkeS1kYXRhLnBuZyIpKQpgYGAKCjxicj4KCk5vIHBhcmVjZSBtdXkgYWxlamFkbyBkZSBsbyBxdWUgZXN0YW1vcyBhY29zdHVtYnJhZG9zLiBQZXJvIC4uLi4gZGVzYXJyb2xsZW1vcyBsYSBpZGVhIHVuIHBvY28gbcOhcy4KCjxicj4KCiMjIyMgVW4gZWplbXBsbyBkZSBkYXRvcyAobm8gdGlkeSkKClN1cG9uZ2Ftb3MgcXVlIGxhIHZhcmlhYmxlIChvIGF0cmlidXRvKSBhIG1lZGlyIGVzIGVsIHNhbGFyaW8geSBsYSB1bmlkYWQgZGUgYW7DoWxpc2lzIGxhcyBwZXJzb25hcy4gSGVtb3MgcmVjb2dpZG8gZGF0b3MgcGFyYSAzIHBlcnNvbmFzLiBWZcOhbW9zbG9zOgoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUUsIHJlc3VsdHMgPSAnaGlkZSd9CmRhdGFfMSA8LSBkYXRhLmZyYW1lKAogICAgICAgICAgICB5ZWFyICA9IGMoIjIwMTQiLCAiMjAxNSIsICIyMDE2IiksICAKICAgICAgICAgICAgUGVkcm8gPSBjKDEwMCwgNTAwLCAyMDApLCAKICAgICAgICAgICAgQ2FybGEgPSBjKDQwMCwgNjAwLCAyNTApLCAKICAgICAgICAgICAgTWFyw61hID0gYygyMDAsIDcwMCwgOTAwKSAgKQpkYXRhXzEKYGBgCgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgZXZhbCA9IFRSVUUsIHJlc3VsdHMgPSAnbWFya3VwJ30Ka25pdHI6OmthYmxlKGRhdGFfMSwgYWxpZ24gPSAnYycsIGNhcHRpb24gPSAiVGFibGEgMTogU2FsYXJpbyBkZSAzIHBlcnNvbmFzOiIpCmBgYAoKCgpFbnRlbmRlbW9zIHBlcmZlY3RhbWVudGUgZXN0b3MgZGF0b3MsIHZpc3VhbG1lbnRlIHNvbiBjw7Ntb2RvcywgcGVybyDCv3NvbiB0aWR5IGRhdGE/IE5PKiBwb3JxdWUgbG9zIGluZGl2aWR1b3MgKG8gdW5pZGFkZXMgZGUgYW7DoWxpc2lzKSBlc3TDoW4gZW4gY29sdW1uYXMuCgo8YnI+CgojIyMjIFVuIGVqZW1wbG8gZGUgZGF0b3MgKHRpZHkqIHBlcm8gd2lkZSkKCkV4YWN0YW1lbnRlIGxvcyBtaXNtb3MgZGF0b3MgcG9kcsOtYW4gZXN0cnVjdHVyYXJzZSBhc8OtOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRSwgcmVzdWx0cyA9ICdoaWRlJ30KZGF0YV8yIDwtIGRhdGEuZnJhbWUobmFtZXMgPSBjKCJQZWRybyIsICJDYXJsYSIsICJNYXLDrWEiKSwgCiAgICAgICAgICAgICAgICAgICAgICBXXzIwMTQgPSBjKDEwMCwgNDAwLCAyMDApLCAKICAgICAgICAgICAgICAgICAgICAgIFdfMjAxNSA9IGMoNTAwLCA2MDAsIDcwMCksCiAgICAgICAgICAgICAgICAgICAgICBXXzIwMTYgPSBjKDIwMCwgMjUwLCA5MDApICAgKQoKZGF0YV8yCmBgYAoKCgpgYGB7ciwgZWNobyA9IEZBTFNFLCBldmFsID0gVFJVRSwgcmVzdWx0cyA9ICdtYXJrdXAnfQprbml0cjo6a2FibGUoZGF0YV8yLCBhbGlnbiA9ICdjJywgY2FwdGlvbiA9ICJUYWJsYSAyOiBTYWxhcmlvIGRlIDMgcGVyc29uYXMgKHdpZGUgZm9ybWF0KSIpCmBgYAoKCgpUYW1iacOpbiBlcyB1biBmb3JtYXRvIGbDoWNpbCBkZSBlbnRlbmRlciBwb3Igbm9zb3Ryb3MsIHBlcm8gwr9zb24gdGlkeT8gU0kqLCBwZXJvIC4uLgoKICAtIEVzIGVsIGZvcm1hdG8gYWwgcXVlIGVzdGFtb3MgbcOhcyBhY29zdHVtYnJhZG9zIChpbmRpdmlkdW9zIG8gcmVnaXN0cm9zIGVuIGZpbGFzIHkgInZhcmlhYmxlcyIgZW4gY29sdW1uYXMpLiDCv1JlYWxtZW50ZSBlbCBXIGRlIDIwMTQgZXMgdW5hIHZhcmlhYmxlPwoKICAtIEVuIGplcmdhIGRlbCB0aWR5dmVyc2UgZXN0ZSBmb3JtYXRvIGRlIGRhdG9zIGVzICoqIndpZGUiKiogKG8gYW5jaG8pCgo8YnI+CgoKUG9kZW1vcyB0cmFiYWphciB0cmFucXVpbGFtZW50ZSBjb24gZWwgYW50ZXJpb3IgZm9ybWF0bywgUEVSTywgc2kgcXVlcmVtb3Mgc2FjYXIgdG9kbyBlbCBwcm92ZWNobyBhbCB0aWR5dmVyc2UgZXMgbWVqb3IgdGVuZXIgbG9zIGRhdG9zIGVuICoqbG9uZyBmb3JtYXQqKi4KCjxicj4KCiMjIyMgVW4gZWplbXBsbyBkZSBkYXRvcyAodGlkeS10aWR5IHkgbG9uZykKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUUsIHJlc3VsdHMgPSAnaGlkZSd9CmRhdGFfMyA8LSBkYXRhLmZyYW1lKAogICAgICAgICAgICBuYW1lcyA9cmVwKGMoIlBlZHJvIiwgIkNhcmxhIiwgIk1hcsOtYSIpLCB0aW1lcyA9IDMpLCAgCiAgICAgICAgICAgIHllYXIgPSByZXAoYygiMjAxNCIsICIyMDE1IiwgIjIwMTYiKSwgZWFjaCA9IDMpLAogICAgICAgICAgICBzYWxhcmlvID0gYygxMDAsIDQwMCwgMjAwLCA1MDAsIDYwMCwgNzAwLCAyMDAsIDI1MCw5MDApICkKZGF0YV8zCmBgYAoKCmBgYHtyLCBlY2hvID0gRkFMU0UsIGV2YWwgPSBUUlVFLCByZXN1bHRzID0gJ21hcmt1cCd9CmtuaXRyOjprYWJsZShkYXRhXzMsIGFsaWduID0gJ2MnLCBjYXB0aW9uID0gIlRhYmxhIDM6IFNhbGFyaW8gZGUgMyBwZXJzb25hcyAobG9uZyBmb3JtYXQpIikKYGBgCgpFc3RlIGZvcm1hdG8sIGZvcm1hdG8gbG9uZywgZXMgbcOhcyBkaWbDrWNpbCBkZSBsZWVyIHBhcmEgbm9zb3Ryb3MsIHBlcm8gZXMgbcOhcyBlZmljaWVudGUgcGFyYSBsb3Mgb3JkZW5hZG9yZXMuIFkgbG9zIGRhdG9zIGxvcyBwcm9jZXNhbiBsb3Mgb3JkZW5hZG9yZXMhIQoKR2VuZXJhbG1lbnRlLCBjdWFuZG8gZXN0ZW1vcyB0cmFiYWphbmRvIGNvbiBsb3MgZGF0b3MgY29uIGVsIHRpZHl2ZXJzZSBjb252ZW5kcsOhIHF1ZSBsb3MgZGF0b3MgZXN0w6luIGVuIGZvcm1hdG8gbG9uZywgcGVybyBoYWJyw6EgdmVjZXMsIHBvciBlamVtcGxvIHBhcmEgbW9zdHJhciB0YWJsYXMsIHRlbmRyZW1vcyBxdWUgcGFzYXJsb3MgYSBmb3JtYXRvIGFuY2hvLiDCv0PDs21vIHBvZGVtb3MgcGFzYXIgdW4gZGYgZGUgZm9ybWF0byBsb25nIGEgd2lkZSB5IGFsIGNvbnRyYXJpbz8gTG8gbcOhcyBoYWJpdHVhbCBlcyB1c2FyIGRvcyBmdW5jaW9uZXMgZGVsIHBhcXVldGUgYHRpZHlyYC4gVmXDoW1vc2xvLgoKCgo8YnI+CgoKCiMjIyBwaXZvdF9sb25nZXIoKSB5IHBpdm90X3dpZGVyKCkKCiMjIyMgZnVuY2lvbmVzIHBhcmEgcGFzYXIgZGUgd2lkZSBhIGxvbmcgKCYgdmljZXZlcnNhKQoKPGJyPgoKWWEgaGVtb3MgZGljaG8gcXVlIGxvcyBwYWNrYWdlcyBkZWwgdGlkeXZlcnNlIHRyYWJhamFuIG1lam9yIGNvbiB0aWR5IGRhdGEgZW4gKipmb3JtYXRvICJsb25nIioqLiDCv1F1w6kgaGFjZW1vcyBzaSB0ZW5lbW9zIHVuIGRhdGFmcmFtZSBlbiBmb3JtYXRvIHdpZGU/IFB1ZXMgcGFzYXJsbyBhIGxvbmcuIEFmb3J0dW5hZGFtZW50ZSB0ZW5lbW9zIHVuIHBrZyBxdWUgaGFjZSBtdXkgc2VuY2lsbG8gcGFzYXIgbG9zIGRhdG9zIGRlIHdpZGUgYSBsb25nICh5IHZpY2V2ZXJzYSk6IGB0aWR5cmAuIENvbmNyZXRhbWVudGUgdXNhcmVtb3MgbGFzIGZ1bmNpb25lcyBgcGl2b3RfbG9uZ2VyKClgIHkgYHBpdm90X3dpZGVyKClgXltIYXN0YSBsYSBhcGFyaWNpw7NuIGRlIHRpZHlyIDEuMC4wLCBsYXMgZnVuY2lvbmVzIHF1ZSBzZSB1c2FiYW4gZXJhbiBgZ2F0aGVyKClgIHkgYHNwcmVhZCgpYC4gRW4gW2VzdGEgY29uZmVyZW5jaWFdKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9dll3WE1uQzAzSTQmbGlzdD1MTExtNms3UGdEalZjRlhKRGJXZ19PYVEpLCAgSGFkbGV5IFdpY2toYW0gbm9zIGNvbnTDsyBxdWUgdW5vIGRlIHN1cyBncmFuZGVzIGVycm9yZXMgZHVyYW50ZSBlbCBkZXNhcnJvbGxvIGRlbCB0aWR5dmVyc2UgZnVlIGxhIGVsZWNjacOzbiBkZSBsb3Mgbm9tYnJlcyBkZSBsYXMgZnVuY2lvbmVzIGBnYXRoZXIoKWAgeSBgc3ByZWFkKClgLiBGaW5hbG1lbnRlIHBvZGVtb3MgZGVjaXI6IGJ5ZSBieWUgYGdhdGhlcigpYCB5IGBzcHJlYWQoKWAsIHdlbGxjb21lIGB0aWR5ciAxLjAuMGAgYW5kIGBwaXZvdF9sb25nZXIoKWAgYW5kIGBwaXZvdF93aWRlcigpYC5dCgoKCltBcXXDrV0oaHR0cHM6Ly93d3cudGlkeXZlcnNlLm9yZy9ibG9nLzIwMjAvMDUvdGlkeXItMS4xLjAvKSB0aWVuZXMgZWwgcG9zdCBvZmljaWFsIGRvbmRlIHNlIGFudW5jaWFiYSBsYSBsbGVnYWRhIGEgQ1JBTiBkZSBgdGlkeXIgMS4wLjBgIHkgW2FxdcOtXShodHRwczovL3RpZHlyLnRpZHl2ZXJzZS5vcmcvYXJ0aWNsZXMvcGl2b3QuaHRtbCkgeSBbYXF1w61dKGh0dHBzOi8vYmxvZy5tZXRob2RzY29uc3VsdGFudHMuY29tL3Bvc3RzL2RhdGEtcGl2b3Rpbmctd2l0aC10aWR5ci8pIHVuIHBvc3QgZGV0YWxsYWRvIHNvYnJlIGVsbGFzLiBMYSBjb25jbHVzacOzbiBkZWwgYXV0b3IgZGVsIMO6bHRpbW8gZGUgZWxsb3MgZXM6CgoKPiBUaGUgbmV3IHRpZHlyIGZ1bmN0aW9ucyBoYXZlIGludHVpdGl2ZSBzeW50YXgsIGFyZSBlYXN5IHRvIHVzZSwgYW5kIGFyZSBtb3JlIGZsZXhpYmlsZSB0aGFuIHRoZSBwcmlvciBmdW5jdGlvbnMuIFNldmVyYWwgb2YgdGhlIG5ldyBhcmd1bWVudHMgYW5kIGZlYXR1cmVzIGFyZSBleHRyZW1lbHkgdXNlZnVsLCBhbmQgd2lsbCBzYXZlIGxvdHMgb2YgdGltZSBvbiBjb21tb24gdGFza3MuCgoKCkVuIGVzdGUgW290cm8gcG9zdF0oaHR0cHM6Ly9mcm9tdGhlYm90dG9tb2Z0aGVoZWFwLm5ldC8yMDE5LzEwLzI1L3Bpdm90aW5nLXRpZGlseS8pIHRpZW5lcyB0YW1iacOpbiB1bmEgZXhwbGljYWNpw7NuIGRldGFsbGFkYSwgZGUgYHBpdm90XyooKWAgcGVybyBhZGVtw6FzIGluY2x1eWUgdW5hIHNlcmllIGRlIGdpZnMgcXVlIGVqZW1wbGlmaWNhbiBlbCBwYXNvIGRlIHdpZGUgYSBsb25nLgoKPGJyPgoKIyMjIyBEZSB3aWRlIGEgbG9uZyBmb3JtYXQgY29uIGBwaXZvdF9sb25nZXIoKWAKCgpMYSBmdW5jacOzbiBgcGl2b3RfbG9uZ2VyKClgIGNvbnZpZXJ0ZSBkYXRhZnJhbWVzIGRlIHdpZGUgYSBsb25nIGZvcm1hdAoKCmBgYHtyICwgZWNobz1GQUxTRSwgZXZhbCA9IFRSVUUsIGZpZy5hc3AgPSA0LzIsIG91dC53aWR0aCA9ICI5NSUiLCBmaWcuYWxpZ24gPSAiY2VudGVyIn0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzA1YV93aWRlLWxvbmcucG5nIikpCmBgYAoKSGFnw6Ftb3NsbzoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CmxpYnJhcnkodGlkeXIpCmRhdGFfd2lkZSA8LSBkYXRhXzIgICAjLSBkYXRhXzIgZXN0w6EgZW4gZm9ybWF0byBhbmNobyAod2lkZSkKCiMtIGxhIGZ1bmNpw7NuIHBpdm90X2xvbmdlcigpIHRyYW5zZm9ybWEgbG9zIGRhdG9zIGRlIGZvcm1hdG8gYW5jaG8od2lkZSkgYSBmb3JtYXRvIGxhcmdvKGxvbmcpCmRhdGFfbG9uZyA8LSBkYXRhX3dpZGUgJT4lIHBpdm90X2xvbmdlcihjb2xzID0gMjo0LCBuYW1lc190byA9ICJwZXJpb2RvIikKYGBgCgpTaSBxdWlzacOpcmFtb3MgYXJyZWdsYXIgbG9zIHZhbG9yZXMgZGUgbG9zIHBlcmlvZG9zOgoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CiMoISEpIHN0cmluZ3I6OnN0cl9yZXBsYWNlIGVuY3VlbnRyYSBlbCB0ZXh0byAiV18iIGVuIGxhIGNvbHVtbmEgInBlcmlvZG8iIHkgbG8gc3VzdGl0dXllIHBvciAiIgpkYXRhX2xvbmcgPC0gZGF0YV9sb25nICU+JSBtdXRhdGUocGVyaW9kbyA9IHN0cl9yZXBsYWNlKHBlcmlvZG8sICJXXyIsICIiICkpCmBgYAoKPGJyPgoKIyMjIyBEZSBsb25nIGEgd2lkZSBmb3JtYXQgY29uIGBwaXZvdF93aWRlcigpYAoKUGFzYXIgcGFzYXIgZGUgbG9uZyBhIHdpZGUsIHRpZHlyIHRpZW5lIGxhIGZ1bmNpw7NuIGBwaXZvdF9sb25nZXIoKWAKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiOTUlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18wNWJfd2lkZS1sb25nLnBuZyIpKQpgYGAKCkhhZ8OhbW9zbG86CgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gYHBpdm90X2xvbmdlcigpYCBjb252aWVydGUgdW4gZGYgZGUgbG9uZyBhIHdpZGUKZGF0YV93aWRlMiA8LSBkYXRhX2xvbmcgJT4lIHBpdm90X3dpZGVyKG5hbWVzX2Zyb20gPSBwZXJpb2RvLCB2YWx1ZXNfZnJvbSA9IHZhbHVlKQpgYGAKCgoKPGJyPgoKIyMjIHNlcGFyYXRlKCkgeSB1bml0ZSgpCgojIyMjIGZ1bmNpb25lcyBwYXJhIHNlcGFyYXIgeSB1bmlyIGNvbHVtbmFzCgo8YnI+CgpFbCBwa2cgdGlkeXIgY29udGllbmUgb3RyYXMgMiBmdW5jaW9uZXM6IGBzZXBhcmF0ZSgpYCB5IGB1bml0ZSgpYCBxdWUgZmFjaWxpdGFuIGVsIHNlcGFyYXIgeSB1bmlyIGNvbHVtbmFzLiBWZWFtb3MgdW4gZWplbXBsbzoKCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFLCByZXN1bHRzID0gJ21hcmt1cCd9CmRmIDwtIGRhdGEuZnJhbWUoIG5hbWVzID0gYygiUGVkcm9fTmF2YWphIiwgIkJvYl9EeWxhbiIsICJDaWRfQ2FtcGVhZG9yIiksIAogICAgICAgICAgICAgICAgICB5ZWFyICA9IGMoMTk3OCwgMTk0MSwgMTA0OCkgKQpkZgpgYGAKCgpTZXBhcmFtb3MgbGEgcHJpbWVyYSBjb2x1bW5hOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRSwgcmVzdWx0cyA9ICdtYXJrdXAnfQpkZl9hIDwtIGRmICU+JSBzZXBhcmF0ZShuYW1lcywgYygiTm9tYnJlIiwgIkFwZWxsaWRvIiksIHNlcCA9ICJfIikKZGZfYQpgYGAKCgpTaSBxdWVyZW1vcyB2b2x2ZXIgYSB1bmlybG9zLCB0ZW5kcsOtYW1vcyBxdWU6CgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRSwgcmVzdWx0cyA9ICdtYXJrdXAnfQpkZl9iIDwtIGRmX2EgJT4lIHVuaXRlKE5vbWJyZV95X0FwZWxsaWRvLCBOb21icmU6QXBlbGxpZG8sIHNlcCA9ICImIikKZGZfYgpgYGAKCjxicj4KCiMjIyMgbWFzIGZ1bmNpb25lcyBkZSB0aWR5cgoKCkFkZW3DoXMsIHJlY3VlcmRhIHF1ZSBlbCBwYXF1ZXRlIGB0aWR5cmAgdGllbmUgbXVjaGFzIFttw6FzIGZ1bmNpb25lc10oaHR0cHM6Ly90aWR5ci50aWR5dmVyc2Uub3JnL3JlZmVyZW5jZS9pbmRleC5odG1sKSBxdWUgbm9zIGZhY2lsaXRhbiBjb25zZWd1aXIgcXVlIG51ZXN0cm9zIGRhdG9zIHNlYW4gKip0aWR5KiouCgoKPGJyPgoKPGJyPgoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMgMy4gIERQTFlSCgpFbiBSIGhheSB2YXJpb3MgZW5mb3F1ZXMgcGFyYSBtYW5pcHVsYXIgZGF0b3MgZW4gUiwgcGVybyBlbCBtw6FzIGhhYml0dWFsLCBkZSBoZWNobyBzZSBoYSBjb252ZXJ0aWRvIGVuIGVsIGVzdMOhbmRhciwgZXMgdXRpbGl6YXIgZWwgdGlkeXZlcnNlLCBjb25jcmV0YW1lbnRlIGVsIHBhcXVldGUgW2BkcGx5cmBdKGh0dHBzOi8vZHBseXIudGlkeXZlcnNlLm9yZy8pLgoKYGRwbHlyYCBlcyB1biBwYXF1ZXRlIHF1ZSBwZXJtaXRlIG1hbmlwdWxhciBkYXRvcyBkZSBmb3JtYSBpbnR1aXRpdmEuIFRpZW5lIDYtNyBmdW5jaW9uZXMgbyB2ZXJib3MgcHJpbmNpcGFsZXMuIENhZGEgdW5vIGRlIGVsbG9zIGhhY2UgInVuYSBzb2xhIGNvc2EiLCBhc8OtIHF1ZSBwYXJhIHJlYWxpemFyIHRyYW5zZm9ybWFjaW9uZXMgY29tcGxlamFzIGhheSBxdWUgaXIgY29uY2F0ZW5hbmRvIGluc3RydWNjaW9uZXMgc2VuY2lsbGFzLiBFc3RvIHNlIGhhY2UgY29uIGVsICoqb3BlcmFkb3IgcGlwZSoqIChgICU+JSBgKQoKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIGRwbHlyIGJhc2ljcwoKVHJhcyBtdWNobyBwZW5zYXIgY29tbyBlc3RydWN0dXLDoWJhbW9zIGVzdGUgYXBhcnRhZG8gZGVsIHR1dG9yaWFsLCBhbCBmaW5hbCBtZSBkZWNhbnTDqSBwb3IgdXRpbGl6YXIgbG9zIG1hdGVyaWFsZXMgZGVsIGN1cnNvIFNUQVQgNTQ1LiDCv3F1ZSBxdWllbiBoYSBzZSBlbmNhcmdhIGRlbCBjdXJzbz8gUHVlcyBKZW5ueSBCcnlhbiAoYWx3YXlzIHJvY2tzISEpLiBQdWVkZXMgZW5jb250cmFybG9zIFthcXXDrV0oaHR0cDovL3N0YXQ1NDUuY29tL2Jsb2NrMDA5X2RwbHlyLWludHJvLmh0bWwpLgoKYGRwbHlyYCB0aWVuZSBtdWNoYXMgZnVuY2lvbmVzLCBwZXJvIGxhcyBwcmluY2lwYWxlcyBzb24gNi03LCBsdWVnbyBsYXMgdmVyZW1vcy4gQ29uIGVsbGFzIHNlIHB1ZWRlbiByZXNvbHZlciBsYSBtYXlvcsOtYSBkZSBwcm9ibGVtYXMgYXNvY2lhZG9zIGEgbGEgbWFuaXB1bGFjacOzbiBkZSBkYXRvcy4KCkNhZGEgZnVuY2nDs24gKG8gdmVyYm8pIGhhY2UgdW5hIHNvbGEgY29zYSwgcGVybyBjb25jYXRlbsOhbmRvbGFzIGNvbiBgJT4lIGAgcGVybWl0ZW4gcmVzb2x2ZXIgY3Vlc3Rpb25lcyBjb21wbGVqYXMuCgpUb2RhcyBsYXMgZnVuY2lvbmVzIHRpZW5lbiB1bmEgZXN0cnVjdHVyYSBvIGNvbXBvcnRhbWllbnRvIHNpbWlsYXI6CgogIC0gZWwgcHJpbWVyIGFyZ3VtZW50byBzaWVtcHJlIGVzIHVuIGRmLiBFc3RvIGVzIGltcG9ydGFudGUgICAgCiAgLSBsb3Mgc2lndWllbnRlcyBhcmd1bWVudG9zIGRlc2NyaWJlbiBxdWUgaGFjZXIgY29uIGxvcyBkYXRvcyAgIAogIC0gZWwgcmVzdWx0YWRvIGVzIHNpZW1wcmUgdW4gbnVldm8gZGYuIEVzdG8gZXMgaW1wb3J0YW50ZSAgIAogIApQb3IgZWplbXBsbywgYGZpbHRlcihkZiwgWDEgPj0gMTApYCBkZXZ1ZWx2ZSB1biBkZiBjb24gbGFzIGZpbGFzIGRlbCBkZiBvcmlnaW5hbCBxdWUgY3VtcGxlbiBsYSBjb25kaWNpw7NuIGRlIHF1ZSBsYSB2YXJpYWJsZSBYMSBlcyBtYXlvciBvIGlndWFsIGEgMTAKClBvZGVtb3MgZXNjcmliaXIgbGEgYW50ZXJpb3IgaW5zdHJ1Y2Npw7NuIGRlIDMgZm9ybWFzLiBMYSBtw6FzIHV0aWxpemFkYSBlcyBsYSDDumx0aW1hOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmRmX25ldyA8LSBmaWx0ZXIoZGYsIFgxID49IDEwKQoKZGZfbmV3IDwtIGRmICU+JSBmaWx0ZXIoLiAsIFgxID49IDEwKQoKZGZfbmV3IDwtIGRmICU+JSBmaWx0ZXIoWDEgPj0gMTApCmBgYAoKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKIyA0LiBQcmluY2lwYWxlcyBmdW5jaW9uZXMgZGUgZHBseXIKCkhheSA2LTcgcHJpbmNpcGFsZXMuIAoKCiAgLSBgZmlsdGVyKClgIDogcGVybWl0ZSBzZWxlY2Npb25hciBmaWxhcyAocXVlIGN1bXBsZW4gdW5hIG8gdmFyaWFzIGNvbmRpY2lvbmVzKQogIC0gYGFycmFuZ2UoKWA6IHJlb3JkZW5hIGxhcyBmaWxhcyAoYGFycmFuZ2UoKWApLgogIC0gYHJlbmFtZSgpYCA6IGNhbWJpYSBsb3Mgbm9tYnJlcyBkZSBsYXMgY29sdW1uYXMgKHZhcmlhYmxlcykKICAtIGBzZWxlY3QoKWAgOiBzZWxlY2Npb25hIGNvbHVtbmFzICh2YXJpYWJsZXMpCiAgLSBgbXV0YXRlKClgIDogY3JlYSBudWV2YXMgdmFyaWFibGVzCiAgLSBgc3VtbWFyaXNlKClgIDogcmVzdW1lIChjb2xhcHNhKSB1bm9zIGN1YW50b3MgdmFsb3JlcyBhIHVubyBzw7Nsby4gUG9yIGVqZW1wbG8sIGNhbGN1bGEgbGEgbWVkaWEsIG1vZGEsIGV0Yy4uLiBkZSB1biBjb25qdW50byBkZSB2YWxvcmVzCiAgCkhheSB1bmEgc8OpcHRpbWE6CgogIC0gYGdyb3VwX2J5KClgIDogcGVybWl0ZSBhZ3J1cGFyIGZpbGFzIGVuIGZ1bmNpw7NuIGRlIHVuYSBvIHZhcmlhcyBjb25kaWNpb25lcwogIApZIGRlc3B1w6lzIGRlIGBkcGx5ciAxLjAuMGAsIGVuIG1heW8gZGUgMjAyMCwgYcOxYWRvIDIgbcOhczoKCiAgLSBgYWNyb3NzKClgICAgeSBgd2hlcmUoKWAuIEVzdGFzIGZ1bmNpb25lcyBzb24gdW4gcG9jbyBkaWZlcmVudGVzLCBzb2xvIHNlIHVzYW4gZW4gY29tYmluYWNpw7NuIGRlIG90cm8gZnVuY2nDs24vdmVyYm8uIFNvbiAyIGZ1bmNpb25lcyBxdWUgZW4gbGEgamVyZ2EgZGVsIHRpZHl2ZXJzZSBubyBzb24gdmVyYm9zIHNpbm8gYWR2ZXJiaW9zLiBMbyB2ZW1vcwoKPGJyPgoKVmXDoW1vc2xhcyB1bmEgYSB1bmEuIFZlcmVtb3Mgc8OzbG8gYWxndW5vcyBlamVtcGxvcy4gWWEgaXJlbW9zIHByYWN0aWNhbmRvCgo8YnI+CgotLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIGBmaWx0ZXIoKWAKCkVzdGEgZnVuY2nDs24gKG8gdmVyYm8pIHNlIHV0aWxpemEgcGFyYSAqKnNlbGVjY2lvbmFyIGZpbGFzKiogZGUgdW4gZGF0YWZyYW1lIChkZikuIFNlIHNlbGVjY2lvbmFuIGxhcyBmaWxhcyBxdWUgY3VtcGxlbiB1bmEgZGV0ZXJtaW5hZGEgY29uZGljacOzbiBvIGNyaXRlcmlvIGzDs2dpY28uIFBvciBlamVtcGxvOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gdmFtb3MgYSB0cmFiYWphciBjb24gbG9zIGRhdG9zIGRlbCBbcGtnIGdhcG1pbmRlcl0oaHR0cHM6Ly9naXRodWIuY29tL2plbm55YmMvZ2FwbWluZGVyKQpnYXBtaW5kZXIgPC0gZ2FwbWluZGVyOjpnYXBtaW5kZXIKYGBgCgo8YnI+CgpTZWxlY2Npb25hbW9zIGxhcyBmaWxhcyBxdWUgY3VtcGxlbiBkZXRlcm1pbmFkb3MgY3JpdGVyaW9zOgoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBPYnNlcnZhY2lvbmVzIGRlIEVzcGHDsWEgKGNvdW50cnkgPT0gIlNwYWluIikKYWEgPC0gZ2FwbWluZGVyICU+JSBmaWx0ZXIoY291bnRyeSA9PSAiU3BhaW4iKSAKCiMtIGZpbGFzIGNvbiB2YWxvcmVzIGRlICJsaWZlRXhwIiA8IDI5CmFhIDwtIGdhcG1pbmRlciAlPiUgZmlsdGVyKGxpZmVFeHAgPCAyOSkgICAgICAgCgojLSBmaWxhcyBjb24gdmFsb3JlcyBkZSAibGlmZUV4cCIgZW50cmUgWzI5LCAzMl0KYWEgPC0gZ2FwbWluZGVyICU+JSBmaWx0ZXIobGlmZUV4cCA+PSAgMjkgLCBsaWZlRXhwIDw9IDMyKSAgIAphYSA8LSBnYXBtaW5kZXIgJT4lIGZpbHRlcihsaWZlRXhwID49ICAyOSAmICBsaWZlRXhwIDw9IDMyKSAgCmFhIDwtIGdhcG1pbmRlciAlPiUgZmlsdGVyKGJldHdlZW4obGlmZUV4cCwgMjksIDMyKSkgICAgICAgCgojLSBvYnNlcnZhY2lvbmVzIGRlIHBhaXNlcyBkZSDDgWZyaWNhIGNvbiBsaWZlRXhwID4gMzIKYWEgPC0gZ2FwbWluZGVyICU+JSBmaWx0ZXIobGlmZUV4cCA+IDcyICYgIGNvbnRpbmVudCA9PSAiQWZyaWNhIikgCgojLSBvYnNlcnZhY2lvbmVzIGRlIHBhw61zZXMgZGUgw4FmcmljYSBvIEFzaWEgY29uIGxpZmVFeHAgPiAzMgphYSA8LSBnYXBtaW5kZXIgJT4lIGZpbHRlcihsaWZlRXhwID4gNzIgJiAgY29udGluZW50ICVpbiUgYygiQWZyaWNhIiwgIkFzaWEiKSApICAKYWEgPC0gZ2FwbWluZGVyICU+JSBmaWx0ZXIobGlmZUV4cCA+IDcyICYgKGNvbnRpbmVudCA9PSAiQWZyaWNhIiB8IGNvbnRpbmVudCA9PSAiQXNpYSIpICkgIApgYGAKCjxicj4KCkxhIGZ1bmNpw7NuIGBmaWx0ZXIoKWAgdGllbmUgbXVjaGFzIG3DoXMgcG9zaWJpbGlkYWRlcy4gWWEgbGFzIGlyZW1vcyB2aWVuZG8uIFBFUk8gc2kgcXVpZXJlcyB2ZXIgdW4gcmVzdW1lbiBkZSBsYXMgcG9zaWJpbGlkYWRlcyBkZWwgcGFxdWV0ZSBkcGx5ciBtaXJhIHN1IFtDSEVBVCBTSEVFVF0oaHR0cHM6Ly93d3cucnN0dWRpby5jb20vcmVzb3VyY2VzL2NoZWF0c2hlZXRzLykuIF5bTGEgw7psdGltYSB2ZXogcXVlIG1pcsOpIGxhIGNoZWF0c2hlZXQgYcO6biBubyBlc3RhYmEgYWN0dWFsaXphZGEgcGFyYSByZWNvZ2VyIGxvcyBjYW1iaW9zIHF1ZSBhcGFyZWNpZXJvbiBlbiBkcGx5ciAxLjAuMCwgcGVybyBhw7puIGFzw60gb3MgcmVzdWx0YXLDoSBkZSBtdWNoYSB1dGlsaWRhZF0uIExhIFt2ZXJzacOzbiBhbnRpZ3VhXShodHRwczovL3d3dy5yc3R1ZGlvLmNvbS93cC1jb250ZW50L3VwbG9hZHMvMjAxNS8wMi9kYXRhLXdyYW5nbGluZy1jaGVhdHNoZWV0LnBkZikgZGUgbGEgQ2hlYXQgc2hlZXQgY29udGllbmUgdGFtYmnDqW4gbGFzIGZ1bmNpb25lcyBkZSBgdGlkeXJgLgoKPGJyPgoKIyMjIyBgc2xpY2UoKWAgdGFtYmnDqW4gZXMgbXV5IMO6dGlsIHBhcmEgc2VsZWNjaW9uYXIgZmlsYXMKCiAgLSBgc2xpY2UoKWA6IGZpbHRyYSBmaWxhcyBwb3Igc3UgcG9zaWNpw7NuIChmw61zaWNhIGVuIGVsIGRmKQoKPGJyPgoKQ29tbyBkaWppbW9zLCBgc2xpY2UoKWAgc2lydmUgcGFyYSBzZWxlY2Npb25hciBmaWxhcyBwb3IgcG9zaWNpw7NuOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CiMtIHNlbGVjY2lvbmEgbGFzIG9ic2VydmFjaW9uZXMgZGUgbGEgZMOpY2ltYSBhIGxhIHF1aW5jZWF2YQphYSA8LSBnYXBtaW5kZXIgJT4lIHNsaWNlKGMoMTA6MTUpKSAKCiMtIHNlbGVjY2lvbmEgbGFzIG9ic2VydmFjaW9uZXMgZGUgbGEgMTIgYSAxMyBZIGRlIGxhIDQ0IGEgNDYsIFkgbGFzIDQgw7psdGltYXMKYWEgPC0gZ2FwbWluZGVyICU+JSBzbGljZSggYygxMjoxNCwgNDQ6NDYsIG4oKS00Om4oKSkgKSAjLSBBUVVJIGhheSB1biBlcnJvciwgdGVuw6lpcyBxdWUgYXJyZWdsYXJsby4gCgojLSBQaXN0YTogaWd1YWwgb3MgYXl1ZGEgY3JlYXIgdW5hIGNvbHVtbmEgY29uIGVsIMOtbmRpY2UgZGUgcm93cyB5IHJlcGV0aXIgZWwgY8OhbGN1bG8KYWEgPC0gZ2FwbWluZGVyICU+JSBtdXRhdGUoaW5kZXggPSAxOm4oKSkKYWEgPC0gZ2FwbWluZGVyICU+JSBzbGljZSggYygxMjoxNCwgNDQ6NDYsIG4oKS00Om4oKSkgKQpgYGBgCiAgCjxicj4KCiMjIyMgdmFyaWFudGVzIGRlIGBzbGljZSgpYAoKSGF5IHZhcmlhcyB2YXJpYW50ZXMgZGUgYHNsaWNlKClgLiBDb25jcmV0YW1lbnRlIGBzbGljZV9tYXgoKWAgIGBzbGljZV9taW4oKWAsIGBzbGljZV9zbXBsKClgLCBgc2xpY2VfaGVhZCgpYCBgc2xpY2VfdGFpbCgpYC4gVmVyZW1vcyBhbGfDum4gZWplbXBsbyBjb24gbGFzIDMgcHJpbWVyYXMuCgoKU2kgcXVlcmVtb3Mgc2VsZWNjaW9uYXIgbGFzIGZpbGFzIHF1ZSB0aWVuZW4gZWwgdmFsb3IgbcOheGltbyAobyBtw61uaW1vKSBkZSB1bmEgZGV0ZXJtaW5hZGEgdmFyaWFibGUsIHBvZGVtb3MgdXNhciBgc2xpY2VfbWF4KClgIHkgYHNsaWNlX21pbigpYAoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CiMtIHNlbGVjY2lvbmEgbGFzIDMgZmlsYXMgY29uIG1heW9yIHZhbG9yIGRlIGxpZmVFeHAKYWEgPC0gZ2FwbWluZGVyICU+JSBzbGljZV9tYXgobGlmZUV4cCwgbiA9IDMpCiMtIHNlbGVjY2lvbmEgbGFzIDQgZmlsYXMgY29uIE1FTk9SIHZhbG9yIGRlIHBvcAphYSA8LSBnYXBtaW5kZXIgJT4lIHNsaWNlX21pbihwb3AsIG4gPSA0KQpgYGBgCgpQYXJhIHZlciBsYSBwb3RlbmNpYWxpZGFkIGRlIHVuYSBmdW5jacOzbiB0aWVuZXMgcXVlIHZlciBzdSBheXVkYSBpbnRlcm5hIChwcmVzaW9uYW5kbyBGMSBvIGNvbiBgaGVscCgpYCkuIFBvciBlamVtcGxvIGBzbGljZV9taW4oKWAgdGllbmUgb3RybyBhcmd1bWVudG8gKGBwcm9wYCkgcXVlIG5vcyBwZXJtaXRlIGNhbGN1bGFyLCBwb3IgZWplbXBsbywgZWwgMTAlIGRlIG9ic2VydmFjaW9uZXMvcGHDrXNlcyBjb24gbWVub3IgZXNwZXJhbnphIGRlIHZpZGEuCgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CiMtIG9ic2VydmFjaW9uZXMgZW4gZWwgcHJpbWVyIGRlY2lsIGVuIGN1YW50byBhIGVzcGVyYW56YSBkZSB2aWRhLCAxMCUgY29uIG1lbm9yIGVzcGVyYW56YSBkZSB2aWRhCmFhIDwtIGdhcG1pbmRlciAlPiUgc2xpY2VfbWluKGxpZmVFeHAsIHByb3AgPSAwLjEpCiMtIDElIGRlIG9ic2VydmFjaW9uZXMgY29uIG1heW9yIHBvYmxhY2nDs24uIEltYWdpbm8gcXVlIGVzdGFyw6FuIENoaW5hIGUgSW5kaWEKYWEgPC0gZ2FwbWluZGVyICU+JSBzbGljZV9tYXgocG9wLCBwcm9wID0gMC4wMSkKYGBgYAoKQSB2ZWNlcyBzZSBuZWNlc2l0YSBvYnRlbmVyIHVuYSBtdWVzdHJhIGFsZWF0b3JpYSBkZSBsb3MgZGF0b3MuIExhIGZ1bmNpw7NuIGBzbGljZV9zYW1wbGUoKWAgZXN0w6EgZGlzZcOxYWRhIHBhcmEgYXl1ZGFybm9zIGVuIGVzdGEgdGFyZWE6CgoKYGBge3J9CiMtIHNlbGVjY2lvbmEgKGFsZWF0b3JpYW1lbnRlKSAxMDAgZmlsYXMgZGUgbG9zIGRhdG9zCmFhIDwtIGdhcG1pbmRlciAlPiUgc2xpY2Vfc2FtcGxlKG4gPSAxMDApCiMtIHNlbGVjY2lvbmEgKGFsZWF0b3JpYW1lbnRlKSB1biA1JSBkZSBsb3MgZGF0b3MKYWEgPC0gZ2FwbWluZGVyICU+JSBzbGljZV9zYW1wbGUocHJvcCA9IDAuMDUpCmBgYAoKPGJyPgoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyBgYXJyYW5nZSgpYAoKRXN0YSBmdW5jacOzbiAobyB2ZXJibykgc2UgdXRpbGl6YSBwYXJhICoqcmVvcmRlbmFyIGxhcyBmaWxhcyoqIGRlIHVuIGRhdGFmcmFtZSAoZGYpLgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CiMtIG9yZGVuYSBsYXMgZmlsYXMgZGUgTUVOT1IgYSBtYXlvciBzZWfDum4gbG9zIHZhbG9yZXMgZGUgbGEgdi4gbGlmZUV4cCAKYWEgPC0gZ2FwbWluZGVyICU+JSBhcnJhbmdlKGxpZmVFeHApCgojLSBvcmRlbmEgbGFzIGZpbGFzIGRlIE1BWU9SIGEgbWVub3Igc2Vnw7puIGxvcyB2YWxvcmVzIGRlIGxhIHYuIGxpZmVFeHAKYWEgPC0gZ2FwbWluZGVyICU+JSBhcnJhbmdlKGRlc2MobGlmZUV4cCkpICAKCiMtIG9yZGVuYWRhIGxhcyBmaWxhcyBkZSBNRU5PUiBhIG1heW9yIHNlZ8O6biBsb3MgdmFsb3JlcyBkZSBsYSB2LiBsaWZlRXhwLiAKIy0gU2kgaGF5IGVtcGF0ZXMgc2UgcmVzdWVsdmUgY29uIGxhIHZhcmlhYmxlICJwb3AiCmFhIDwtIGdhcG1pbmRlciAlPiUgYXJyYW5nZShsaWZlRXhwLCBwb3ApIApgYGAKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIGByZW5hbWUoKWAKCkVzdGEgZnVuY2nDs24gcGVybWl0ZSBjYW1iaWFyIGxvcyBub21icmVzIGRlIGxhcyBjb2x1bW5hcyAKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBjYW1iaWEgbG9zIG5vbWJyZXMgZGUgbGlmZUV4cCB5IGdkcFBlcmNhcCBhIGxpZmVfZXhwIHkgZ2RwX3BlcmNhcCAKZ2FwbWluZGVyICU+JSByZW5hbWUobGlmZV9leHAgPSBsaWZlRXhwLCAgZ2RwX3BlcmNhcCA9IGdkcFBlcmNhcCkKCiMtKCEhKSBsYSBmdW5jacOzbiBuYW1lcygpIGRlIFItYmFzZSBlcyBtdXkgw7p0aWwuIFRiIHNldE5hbWVzKCkgeSBzZXRfbmFtZXMoKQphYSA8LSBnYXBtaW5kZXIKbmFtZXMoYWEpIDwtIG5hbWVzKGFhKSAlPiUgdG91cHBlcgpuYW1lcyhhYSkgPC0gbmFtZXMoYWEpICU+JSB0b2xvd2VyCm5hbWVzKGFhKSA8LSBjKCJ2YXJfMDEiLCAidmFyXzAyIiwgInZhcl8wMyIsICJ2YXJfMDQiLCAidmFyXzA1IiAsICJ2YXJfMDYiKQpuYW1lcyhhYSkgPC0gcGFzdGUwKCJWYXJfIiwgMTo2KQpuYW1lcyhhYSkgPC0gcGFzdGUwKCJMYWdfIiwgZm9ybWF0QygxOjYsIHdpZHRoID0gMiwgZmxhZyA9ICIwIikpIApgYGAKCjxicj4KCiMjIyMgcmVuYW1lX3dpdGgoKSAsIHVuYSB2YXJpYW50ZSBkZSByZW5hbWUoKQoKU2kgdGllbmVzIHF1ZSBoYWNlciB0cmFuc2Zvcm1hY2lvbmVzIG3DoXMgY29tcGxlamFzLCBxdWUgcmVxdWllcmFuIGVsIHVzbyBkZSBmdW5jaW9uZXMgbyBwYXV0YXMsIGRlIGxvcyBub21icmVzIGRlIGxhcyB2YXJpYWJsZXMgcHVlZGVzIHVzYXIgYHJlbmFtZV93aXRoKClgCgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KYWEgPC0gZ2FwbWluZGVyCnJlbmFtZV93aXRoKGFhLCB0b3VwcGVyKQpyZW5hbWVfd2l0aChhYSwgdG91cHBlciwgc3RhcnRzX3dpdGgoIkxpZmUiKSB8IGNvbnRhaW5zKCJjb3VudHIiKSkKcmVuYW1lX3dpdGgoYWEsIH4gc3RyX3JlcGxhY2UoLngsICJlIiwgIsOWIikpICAjLSAoISEhISkKYGBgCgo8YnI+CgpMYSBmdW5jacOzbiBgcmVuYW1lKClgIGVzIMO6dGlsIHBlcm8sIGVuc2VndWlkYSB2ZXJlbW9zIHF1ZSBsYSBzaWd1aWVudGUgZnVuY2nDs24sIGBzZWxlY3QoKWAsIHRhbWJpw6luIHBlcm1pdGUgcmVub21icmFyIGxhcyBjb2x1bW5hcywgZSBpbmNsdXNvIHJlb3JkZW5hciBsYSBwb3NpY2nDs24gZGUgZXN0YXMuCgoKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIGBzZWxlY3QoKWAKCkVzdGEgZnVuY2nDs24gKG8gdmVyYm8pIHNpcnZlIHBhcmEgKipzZWxlY2Npb25hciBjb2x1bW5hcyoqIGRlIHVuIGRmLgoKIyMjIyBzZWxlY2Npb25hciB2YXJpYWJsZXMgcG9yIG5vbWJyZQoKU2VsZWNjaW9uYW1vcyBsYXMgdmFyaWFibGVzICJ5ZWFyIiB5ICJsaWZlRXhwIjoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBTZSBsZWUgY29tbzog4oCcVGFrZSBlbCBkZiBnYXBtaW5kZXIsIHRoZW4gc2VsZWN0IHRoZSB2YXJpYWJsZXMgeWVhciBhbmQgbGlmZUV4cOKAnQphYSA8LSBnYXBtaW5kZXIgJT4lIHNlbGVjdCh5ZWFyLCBsaWZlRXhwKSAKYWEgPC0gZ2FwbWluZGVyICU+JSBzZWxlY3QoYyh5ZWFyLCBsaWZlRXhwKSkKYGBgCgo8YnI+CgojIyMjIHF1aXRhciB2YXJpYWJsZXMKClBhcmEgZWxpbWluYXIgdW5hIHZhcmlhYmxlIGhheSB2YXJpYXMgZm9ybWFzOgoKYGBge3J9CmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KC15ZWFyKSAgICMtIGxhIGZvcm1hIG1hcyBoYWJpdHVhbAoKIy0gZXN0YXMgZG9zIGZvcm1hcyBzb24gbXVjaG8gbWVub3MgaGFiaXR1YWxlcwphYSA8LSBnYXBtaW5kZXIgJT4lIHNlbGVjdCgheWVhcikgICAKYWEgPC0gZ2FwbWluZGVyICU+JSBtdXRhdGUoeWVhciA9IE5VTEwpICAgIy0gYcO6biBubyBoZW1vcyB2aXN0byBtdXRhdGUoKQpgYGAKClBhcmEgZWxpbWluYXIgdmFyaWFzIHZhcmlhYmxlczoKCmBgYHtyfQojLSBxdWl0YW1vcyBsYXMgdmFyaWFibGVzOiB5ZWFyIHkgbGlmZUV4cAphYSA8LSBnYXBtaW5kZXIgJT4lIHNlbGVjdCgtYyh5ZWFyLCBsaWZlRXhwKSkKYGBgCgo8YnI+CgojIyMjIHNlbGVjY2lvbmFyIHBvciBwb3NpY2nDs24KClNlbGVjY2lvbmFtb3MgbGFzIHZhcmlhYmxlcyBkZWwgZGYgZ2FwbWluZGVyIHNpZ3VpZW50ZXM6IGRlIGxhIHByaW1lcmEgYSBsYSB0ZXJjZXJhIHkgdGFtYmnDqW4gbGEgcXVpbnRhIChtZWpvciBzZWxlY2Npb25hcmxhcyBwb3Igbm9tYnJlISEpCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KIy0gc2VsZWNjaW9uYW1vcyBsYXMgdmFyaWFibGVzIHsxLCAyLCAzIHkgNX0KYWEgPC0gZ2FwbWluZGVyICU+JSBzZWxlY3QoMTozLCA1KQpgYGAKCjxicj4KCiMjIyMgcXVpdGFyIHZhcmlhYmxlcyBwb3IgcG9zaWNpw7NuCgpTZWxlY2Npb25hbW9zIHRvZGFzIGxhcyB2YXJpYWJsZXMgZGVsIGRmIGdhcG1pbmRlciAqKmV4Y2VwdG8qKiBsYXMgc2lndWllbnRlczogZGUgbGEgcHJpbWVyYSBhIGxhIHRlcmNlciB5IGxhIHF1aW50YSAobWVqb3Igc2VsZWNjaW9uYXJsYXMgcG9yIG5vbWJyZSkKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBxdWl0YW1vcyBsYXMgdmFyaWFibGVzIHsxLCAyLCAzIHkgNX0KYWEgPC0gZ2FwbWluZGVyICU+JSBzZWxlY3QoLSBjKDE6MywgNSkpCmBgYAoKPGJyPgoKIyMjIyBgc2VsZWN0KClgIGNvbiBsYSBmdW5jacOzbiBhdXhpbGlhciBgd2hlcmUoKWAKCkVuIGVsIGRhdGEuZnJhbWUgYGdhcG1pbmRlcmAgbGFzIDIgcHJpbWVyYXMgdmFyaWFibGVzIChjb3VudHJ5IHkgY29udGluZW50KSBzb24gZmFjdG9yZXMgeSBsYXMgNCBzaWd1aWVudGVzIHNvbiB2YXJpYWJsZSBudW3DqXJpY2FzLiAKCmBgYHtyfQpwcmludChnYXBtaW5kZXIsIG4gPSAzKQpgYGAKCgpJbWFnaW5hIHF1ZSBxdWVyZW1vcyBzZWxlY2Npb25hciBzw7NsbyBsYXMgdmFyaWFibGVzIHF1ZSBzb24gbnVtw6lyaWNhcy4gUG9kZW1vcyBoYWNlcmxvIHBvciBub21icmUgbyBwb3IgcG9zaWNpw7NuIHBlcm8gbWVqb3IgY29uIGBzZWxlY3QoKWAgeSBsYSBmdW5jacOzbiBhdXhpbGlhciBgd2hlcmUoKWBeW0hhc3RhIG1heW8gZGUgMjAyMCwgZXN0byBlcywgaGFzdGEgZHBseXIgMS4wLjAsIGVzdG8gc2UgaGFjaWEgY29uIGxhIGZ1bmNpw7NuIGBzZWxlY3RfaWYoKWBdLiAKCgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KGlzLm51bWVyaWMpICAgICAgICAjLSBmdW5jaW9uYSwgcGVybyAuLi4KYWEgPC0gZ2FwbWluZGVyICU+JSBzZWxlY3Qod2hlcmUoaXMubnVtZXJpYykpICMtIGVzICJwcmVmZXJpYmxlIiBlc3RhIHNlZ3VuZGEgZXhwcmVzacOzbgpgYGAKCgpgc2VsZWN0KClgIHkgYHdoZXJlKClgIHNvbiBkb3MgZnVuY2lvbmVzLCBzw60sIHBlcm8gZW4gbGEgamVyZ2EgZGVsIHRpZHl2ZXJzZSwgYHNlbGVjdCgpYCBlcyB1biB2ZXJibyB5IGB3aGVyZSgpYCBlcyB1biBhZHZlcmJpbywgY3VhbGlmaWNhL2NhbWJpYSBsbyBxdWUgaGFjZSBgc2VsZWN0KClgLgoKClNpIHF1aXNpw6lyYW1vcyBzZWxlY2Npb25hciBsYXMgdmFyaWFibGVzIHF1ZSBubyBzb24gbnVtw6lyaWNhcyBoYXLDrWFtb3M6CgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KCFpcy5udW1lcmljKQphYSA8LSBnYXBtaW5kZXIgJT4lIHNlbGVjdCghd2hlcmUoaXMubnVtZXJpYykpICAjLSBlcyBwcmVmZXJpYmxlIGVzdGEgc2VndW5kYSBleHByZXNpw7NuCmBgYAoKPGJyPgoKTGEgZnVuY2nDs24gYHNlbGVjdCgpYCB0aWVuZSBtdWNoYXMgbcOhcyBwb3NpYmlsaWRhZGVzLiB5YSBsYXMgaXJlbW9zIHZpZW5kby4gUEVSTyBzaSBxdWllcmVzIHZlciB1biByZXN1bWVuIGRlIGxhcyBwb3NpYmlsaWRhZGVzIGRlbCBwa2cgZHBseXIgbWlyYSBzdSBbQ0hFQVQgU0hFRVRdKGh0dHBzOi8vd3d3LnJzdHVkaW8uY29tL3Jlc291cmNlcy9jaGVhdHNoZWV0cy8pLgoKPGJyPgoKIyMjIyByZW5vbWJyYW5kbyB5IHJlb3JkZW5hbmRvIGNvbHVtbmFzIGNvbiBgc2VsZWN0KClgCgpMbyBxdWUgc8OtIHZhbW9zIGEgdmVyIHNvbiAyIHBvc2liaWxpZGFkZXMgZGUgYHNlbGVjdCgpYCBxdWUgc29uIG11eSDDunRpbGVzLiBDb24gYHNlbGVjdCgpYCBwb2RlbW9zOiAqKnJlbm9tYnJhcioqIHkgKipyZW9yZGVuYXIqKiBsYXMgY29sdW1uYXM6CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KIy0gZGVqYW1vcyBlbiBhYSBzb2xhbWVudGUgYSBsYXMgY29sdW1uYXMgInllYXIiIHkgInBvcCI7IEFERU3DgVMsIGFob3JhLCAicG9wIiBpcsOhIGFudGVzIHF1ZSAieWVhciIKYWEgPC0gZ2FwbWluZGVyICU+JSBzZWxlY3QocG9wLCB5ZWFyKQpgYGAKCjxicj4KCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBkZWphbW9zIGVuIGFhIHNvbGFtZW50ZSBhIGxhcyBjb2x1bW5hcyAieWVhciIgeSAicG9wIiB5IGxlcyBjYW1iaWFtb3MgZWwgbm9tYnJlCmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KHBvYmxhY2lvbiA9IHBvcCwgYcOxbyA9IHllYXIpCmBgYAoKPGJyPgoKSW1hZ2luYSBxdWUgcXVpZXJlcyBxdWUgbGEgw7psdGltYSBjb2x1bW5hIHBhc2UgYSBzZXIgbGEgcHJpbWVyYSAobWFuw61hcyEhKS4gUG9kZW1vcyBoYWNlcmxvIGNvbiBzZWxlY3QgeSBgZXZlcnl0aGluZygpYC4gZXZlcnl0aGluZyBlcyB1bmEgKipmdW5jacOzbiBhdXhpbGlhcioqOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CiMtICJnZHBQZXJjYXAiIHF1ZSBlcyBsYSDDumx0aW1hIGNvbHVtbmEgcGFzYSBhIHNlciBsYSBwcmltZXJhCmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KGdkcFBlcmNhcCwgZXZlcnl0aGluZygpKQoKIyghISkgb3RyYXMgMyBmb3JtYXMgZGUgaGFjZXIgbG8gbWlzbW86IHF1ZSBsYSDDumx0aW1hIGNvbHVtbmEgcGFzZSBhIHNlciBsYSBwcmltZXJhCmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KG5jb2woZGYpLCBldmVyeXRoaW5nKCkpCmFhIDwtIGdhcG1pbmRlciAlPiUgc2VsZWN0KGxlbmd0aChkZiksIGV2ZXJ5dGhpbmcoKSkKYWEgPC0gZ2FwbWluZGVyICU+JSBzZWxlY3QobGFzdF9jb2woKSwgZXZlcnl0aGluZygpKSAgIy0gdXNhbW9zIHRoZSBzZWxlY3Rpb24gaGVscGVyIGxhc3RfY29sKCkKYGBgCgpFbiBsYSDDumx0aW1hIGluc3RydWNjacOzbiBoZW1vcyB1c2FkbyBkb3MgZnVuY2lvbmVzIGF1eGlsaWFyZXMgZGUgc2VsZWN0KCksIGRvcyAiaGVscGVyIGZ1bmN0aW9ucyIuIExhIGxpc3RhIGNvbXBsZXRhIGRlIGZ1bmNpb25lcyBhdXhpbGlhcmVzIHBhcmEgc2VsZWN0KCkgcHVlZGVzIHZlcmxhIFthcXXDrV0oaHR0cHM6Ly90aWR5c2VsZWN0LnItbGliLm9yZy9yZWZlcmVuY2Uvc2VsZWN0X2hlbHBlcnMuaHRtbCkuCgo8YnI+CgoKIyMjIyBgcmVsb2NhdGUoKSBgCgpEZXNkZSBkcGx5ciAxLjAuMCwgdGVuZW1vcyBvdHJhIGZ1bmNpw7NuIHBhcmEgcmVvcmRlbmFyIGxhcyB2YXJpYWJsZXMgZGUgdW4gZGF0YS5mcmFtZTogYHJlbG9jYXRlKClgLiBWZcOhbW9zbGEgZW4gYWNjacOzbjoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQphYSA8LSBnYXBtaW5kZXIgJT4lIGRwbHlyOjpyZWxvY2F0ZShjb3VudHJ5LCAuYWZ0ZXIgPSBsaWZlRXhwKQphYSA8LSBnYXBtaW5kZXIgJT4lIGRwbHlyOjpyZWxvY2F0ZShjb3VudHJ5LCAuYmVmb3JlID0gbGlmZUV4cCkKYGBgCgpMYXMgb3BjaW9uZXMgYC5hZnRlcmAgeSBgLmJlZm9yZWAgdGFtYmnDqW4gZXN0w6FuIGRpc3BvbmlibGVzIGVuIGBtdXRhdGUoKWAsIGxhIHNpZ3VpZW50ZSBmdW5jacOzbiBxdWUgcHJlc2VudGFyZW1vcy4KCjxicj4KPGJyPgoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKIyMgYG11dGF0ZSgpYAoKRXN0YSBmdW5jacOzbiAobyB2ZXJibykgc2lydmUgcGFyYSAqKmNyZWFyIG51ZXZhcyB2YXJpYWJsZXMqKiAoY29sdW1uYXMpLiBMw7NnaWNhbWVudGUsIGVzIG11eSDDunRpbCBlbiBhbsOhbGlzaXMgZGUgZGF0b3MuCgoKQ3JlYW1vcyBsYSB2YXJpYWJsZTogYEdEUCA9IHBvcCpnZHBwZXJDYXBgCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KIy0gQ3JlYW1vcyBsYSB2YXJpYWJsZTogR0RQID0gcG9wKmdkcHBlckNhcAphYSA8LSBnYXBtaW5kZXIgJT4lIG11dGF0ZShHRFAgPSBwb3AqZ2RwUGVyY2FwKQpgYGAKClBvciBkZWZlY3RvLCBsYSBudWV2YSB2YXJpYWJsZSBjcmVhZGEgc2Ugc2l0dWFyw6EgYWwgZmluYWwgZGVsIGRhdGEgZnJhbWUsIGEgbm8gc2VyIHF1ZSB1c2Vtb3MgbG9zIGFyZ3VtZW50b3MgYC5hZnRlcmAgeSBgLmJlZm9yZWAKCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KYWEgPC0gZ2FwbWluZGVyICU+JSBtdXRhdGUoR0RQID0gcG9wKmdkcFBlcmNhcCwgLmFmdGVyID0gY291bnRyeSkKYWEgPC0gZ2FwbWluZGVyICU+JSBtdXRhdGUoR0RQID0gcG9wKmdkcFBlcmNhcCwgLmJlZm9yZSA9IGNvdW50cnkpCmBgYAoKYG11dGF0ZSgpYCB0YW1iacOpbiB0aWVuZSB1biBhcmd1bWVudG8oYC5rZWVwYCkgcGFyYSBjb250cm9sYXIgcXVlIHZhcmlhYmxlcyBwZXJtYW5lY2VuIGVuIGVsIGRhdGEgZnJhbWUuIFBvciBlamVtcGxvLCBlbiBlbCBjaHVuayBkZSBhYmFqbyBkZWphcmVtb3Mgc29sbyBsYXMgdmFyaWFibGVzIHVzYWRhcy4KCmBgYHtyfQphYSA8LSBnYXBtaW5kZXIgJT4lIG11dGF0ZShHRFAgPSBwb3AqZ2RwUGVyY2FwLCAua2VlcCA9ICJ1c2VkIikKYGBgCgo8YnI+CgotLS0tLS0tLS0tLS0tLS0tLS0tCgojIyBgc3VtbWFyaXNlKClgCgpFc3RhIGZ1bmNpw7NuIChvIHZlcmJvKSBzaXJ2ZSBwYXJhICoqUkVTVU1JUioqIChvICJjb2xhcHNhciBmaWxhcyIpLiBDb2dlIHVuYSB2YXJpYWJsZSBvIGdydXBvIGRlIHZhbG9yZXMgY29tbyBpbnB1dCB5IGRldnVlbHZlIHVuIHNvbG8gdmFsb3I7IHBvciBlamVtcGxvLCBoYXlhIGxhIG1lZGlhIGFyaXRtw6l0aWNhIChvIGVsIG3DrW5pbW8sIG8gZWwgbcOheGltbyAuLi4pIGRlIHVuYSBjb2x1bW5hL3ZhcmlhYmxlLgoKCjxicj4KCk9idGVuZ2Ftb3MgZGV0ZXJtaW5hZG9zIGVzdGFkw61zdGljb3MgZGUgKip1bmEgdmFyaWFibGUqKi4gUGFyYSBlc3RvIG5vIG5vcyBoYWNlIGZhbHRhIGBkcGx5cmAgcGVybyBjb252aWVuZSBpciBoYWJpdHXDoW5kb3NlIGEgc3Ugc2ludGF4aXMuCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KIy0gcmV0b3JuYXLDoSB1biDDum5pY28gdmFsb3I6IGxhIG1lZGlhIGdsb2JhbCBkZSBsYSB2LiAibGlmZUV4cCIKYWEgPC0gZ2FwbWluZGVyICU+JSBzdW1tYXJpc2UobWVkaWEgPSBtZWFuKGxpZmVFeHApKSAgCgojLSByZXRvcm5hcsOhIHVuIMO6bmljbyB2YWxvcjogZWwgbsO6bWVybyBkZSBmaWxhcwphYSA8LSBnYXBtaW5kZXIgJT4lIHN1bW1hcmlzZShOTiA9IG4oKSkgIAphYSA8LSBnYXBtaW5kZXIgJT4lIGNvdW50KCkgICAgICAgICAgICAgICAgIy0gbcOhcyBhZGVsYW50ZSB2ZXJlbW9zIGxhIHV0aWxpZGFkIGRlIGNvdW50KCkKCgojLSByZXRvcm5hcsOhIHVuIMO6bmljbyB2YWxvcjogbGEgZGVzdmlhY2nDs24gdMOtcGljYSBkZSBsYSB2LiAibGlmZUV4cCIKYWEgPC0gZ2FwbWluZGVyICU+JSBzdW1tYXJpc2UoZGVzdmlhY2lvbl90aXBpY2EgPSBzZChsaWZlRXhwKSkgIAoKIy0gcmV0b3JuYXLDoSB1biDDum5pY28gdmFsb3I6IGVsIG3DoXhpbW8gZGUgbGEgdmFyaWFibGUgInBvcCIKYWEgPC0gZ2FwbWluZGVyICU+JSBzdW1tYXJpc2UobWF4KHBvcCkpICAKCiMtIHJldG9ybmFyw6EgMiB2YWxvcmVzOiBsYSBtZWRpYSB5IHNkIGRlIGxhIHYuICJsaWZlRXhwIgphYSA8LSBnYXBtaW5kZXIgJT4lIHN1bW1hcmlzZShtZWFuKGxpZmVFeHApLCBzZChsaWZlRXhwKSkgIAoKIy0gcmV0b3JuYXLDoSAyIHZhbG9yZXM6IGxhcyBtZWRpYXMgZGUgImxpZmVFeHAiIHkgImdkcFBlcmNhcCIKYWEgPC0gZ2FwbWluZGVyICU+JSBzdW1tYXJpc2UobWVhbihsaWZlRXhwKSwgbWVhbihnZHBQZXJjYXApKSAgCmBgYAoKPGJyPgoKCiMjIyMgYGFjcm9zcygpYCB5IGB3aGVyZSgpYAoKQW50ZXMgZGUgcGFzYXIgYSB2ZXIgYGdyb3VwX2J5KClgLCB2YW1vcyBhIHV0aWxpemFyIGxhcyAyIG51ZXZhcyBmdW5jaW9uZXMgYGFjcm9zcygpYCB5IGB3aGVyZSgpYC4KCk11Y2hhcyB2ZWNlcyBlbiB1biB0cmFiYWpvIHNlIGhhbiBkZSBjYWxjdWxhciBlc3RhZMOtc3RpY29zIGRlICoqdG9kYXMgbGFzIHZhcmlhYmxlcyoqIGRlbCBkZi4gRXN0byBzZSBoYWNlIGNvbiBgc3VtbWFyaXNlKClgLCBQRVJPIHV0aWxpemFuZG8gdGFtYmnDqW4gdW5hIGZ1bmNpw7NuIGRlIGF5dWRhIChoZWxwZXIgZnVuY3Rpb24pOiBgYWNyb3NzKClgXltIYXN0YSBsYSBhcGFyaWNpw7NuIGRlIGRwbHlyIDEuMC4wIGVuIG1heW8gZGUgMjAyMCwgc2UgdXRpbGl6YWJhIGxhIGZ1bmNpw7NuIGBzdW1hcmlzZV9hbGwoKWBdLiBBIHZlY2VzIHRhbWJpw6luIGhheSBxdWUgdXNhciBvdHJhIGhlbHBlciBmdW5jdGlvbjogYHdoZXJlKClgLgoKTGEgc2ludGF4aXMgZGUgYGFjcm9zcygpYCBlczogIAoKYGFjcm9zcyguY29scyA9IGV2ZXJ5dGhpbmcoKSwgLmZucyA9IE5VTEwsIC4uLiwgLm5hbWVzID0gTlVMTClgOyBlcyBkZWNpciwgICAKCmBhY3Jvc3MoImNvbHVtbmFzIHNlbGVjY2lvbmFkYXMiLCAiZnVuY2lvbmVzIG8gY8OhbGN1bG9zIGEgcmVhbGl6YXIiLCAic2kgcXVpZXJlcyBjb250cm9sYXIgbG9zIG5vbWJyZXMiKSlgLiAKClNpIGFsIHNlbGVjY2lvbmFyIGxhcyBjb2x1bW5hcyB1dGlsaXphcyBhbGfDum4gY3JpdGVyaW8gbMOzZ2ljbywgY29tbyBwb3IgZWplbXBsbyBgaXMubnVtZXJpYygpYCwgZW50b25jZXMgbGEgc2ludGF4aXMgZGUgYGFjcm9zcygpYCBlcyB1biBwb2NvIGRpZmVyZW50ZTsgY29uY3JldGFtZW50ZSBzZXLDoTogCgpgYWNyb3NzKCB3aGVyZSgiY29sdW1uYXMgc2VsZWNjaW9uYWRhcyIpLCAiZnVuY2lvbmVzIG8gY8OhbGN1bG9zIGEgcmVhbGl6YXIiLCAic2kgcXVpZXJlcyBjb250cm9sYXIgbG9zIG5vbWJyZXMiKSlgCgpMb3MgZWplbXBsb3MgYXl1ZGFuIGEgZW50ZW5kZXJsbzoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBtZWRpYSBkZSBjYWRhIHVuYSBkZSBsYXMgNiB2YXJpYWJsZXMuIERldnVlbHZlIDIgd2FybmluZ3MgcG9ycXVlIGxhcyAyIHByaW1lcmFzIHNvbiB0ZXh0dWFsZXMuIE5vIHNlIHB1ZWRlIGNhbGN1bGFyIGxhIG1lZGlhIGRlIGNvbnRpbmVudCB5IGNvdW50cnkKZ2FwbWluZGVyICU+JSBzdW1tYXJpc2UoYWNyb3NzKGV2ZXJ5dGhpbmcoKSwgbWVhbikgKSAKCiMtIGNhbGN1bGFtb3MgbGEgbWVkaWEgZGUgdGVyY2VyYSBhIGxhIHNleHRhIHZhcmlhYmxlCmdhcG1pbmRlciAlPiUgc3VtbWFyaXNlKGFjcm9zcygzOjYsIG1lYW4pICkgCmBgYAoKTG8gcXVlIG9zIGRlY8OtYSBkZSBzZWxlY2Npb25hciBjb2x1bW5hcyBjb24gdW4gY3JpdGVyaW8gbMOzZ2ljbyB5IHVzYXIgYHdoZXJlKClgCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KZ2FwbWluZGVyICU+JSBzdW1tYXJpc2UoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCBtZWFuKSkgCgojLSBjb24gbG9zIG5vbWJyZXMgZGUgbG9zIGFyZ3VtZW50b3MgKG3DoXMgbGFyZ28gcGVybyBjb252aWVuZSB2ZXJsbyBkZSB2ZXogZW4gY3VhbmRvKQpnYXBtaW5kZXIgJT4lIHN1bW1hcmlzZShhY3Jvc3MoLmNvbHMgPSB3aGVyZShpcy5udW1lcmljKSwgLmZucyA9IG1lYW4pKSAKYGBgCgpWYW1vcyBhIGNhbGN1bGFyIGNvc2FzIHVuIHBvY28gbcOhcyBjb21wbGVqYXMuIEltYWdpbmEgcXVlIG5vIHPDs2xvIHF1aWVyZXMgY2FsY3VsYXIgbGEgbWVkaWEgc2lubyBxdWUgdGFtYmnDqW4gcXVpZXJlcyBjYWxjdWxhciBsYSBkZXN2aWFjacOzbiB0w61waWNhLiBTZWd1aXJlbW9zIHVzYW5kbyBgc3VtbWFyaXNlKClgIHkgYGFjcm9zcygpYC4gTG8gw7puaWNvIG51ZXZvIGVzIHF1ZSBjb21vIHZhbW9zIGEgYXBsaWNhciBkb3MgZnVuY2lvbmVzIChgbWVhbigpYCB5IGBzZCgpYCkgbGFzIHRlbmVtb3MgcXVlIHBvbmVyIGRlbnRybyBkZSBgbGlzdCgpYC4gVGllbmUgc2VudGlkbywgZXMgdW5hIGxpc3RhIGRlIGZ1bmNpb25lcyBhIGFwbGljYXIgYSBsYXMgY29sdW1uYXMgcXVlIHNlbGVjY2lvbmVtb3MgY29uIGBhY3Jvc3MoKWAKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBjYWxjdWxhbW9zIGxhIG1lZGlhIHkgZGVzdmlhY2nDs24gdMOtcGljYSBkZSBsYXMgY29sdW1uYXMgMyBhIDYuCmdhcG1pbmRlciAlPiUgc3VtbWFyaXNlKGFjcm9zcygzOjYsIGxpc3QobWVkaWEgPSBtZWFuLCBkZXN2ID0gc2QpKSkKCiMtIGxvIG1pc21vLCBwZXJvIGV4cGxpY2l0YW5kbyBsb3Mgbm9tYnJlcyBkZSBsb3MgYXJndW1lbnRvcwpnYXBtaW5kZXIgJT4lIHN1bW1hcmlzZShhY3Jvc3MoLmNvbHMgPSAzOjYsIC5mbnMgPSBsaXN0KG1lZGlhID0gbWVhbiwgZGVzdiA9IHNkKSApKQoKIy0gbG8gbWlzbW8gb3RyYSB2ZXosIHBlcm8gZWxpZ2llbmRvIGVsIG5vbWJyZSBkZSBsYXMgdmFyaWFibGVzIHF1ZSBzZSB2YW4gYWEgY3JlYXIgY29uIC5uYW1lcwpnYXBtaW5kZXIgJT4lIHN1bW1hcmlzZShhY3Jvc3MoMzo2LCBsaXN0KG1lZGlhID0gbWVhbiwgZGVzdiA9IHNkKSwgLm5hbWVzID0gIntmbn1fe2NvbH0iKSkKYGBgCgooISEhKUltYWdpbmEgcXVlIHF1aXNpw6lyYW1vcyBwcmVzZW50YXIgZW4gdW5hIHRhYmxhIGxvcyBhbnRlcmlvcmVzIHJlc3VsdGFkb3M7IHRlbmRyw61hbW9zIHF1ZSB1c2FyIGB0aWR5cjo6cGl2b3RfbG9uZ2VyKClgLiBMbyB2b3kgYSBoYWNlciBwb3IgdHJvem9zLiBOb3MgdmEgYSBjb3N0YXIgdW4gcG9jbzoKCmBgYHtyfQphYSA8LSBnYXBtaW5kZXIgJT4lIHN1bW1hcmlzZShhY3Jvc3MoMzo2LCBsaXN0KG1lZGlhID0gbWVhbiwgZGVzdiA9IHNkKSwgLm5hbWVzID0gIntmbn1fe2NvbH0iKSkgCgphYTEgPC0gYWEgJT4lIHBpdm90X2xvbmdlcigxOjgsIG5hbWVzX3RvID0gIm5hbWVzIiwgdmFsdWVzX3RvID0gInZhbHVlcyIpCmFhMiA8LSBhYTEgJT4lIHNlcGFyYXRlKG5hbWVzLCBpbnRvID0gYygib3BlcmFjaW9uIiwgInZhcmlhYmxlIiksIHNlcCA9ICAiXyIpICU+JSAKICAgICAgICAgICAgICAgc2VsZWN0KHZhcmlhYmxlLCBldmVyeXRoaW5nKCkpCmFhMyA8LSBhYTIgJT4lIHBpdm90X3dpZGVyKDE6MywgbmFtZXNfZnJvbSA9IG9wZXJhY2lvbiwgdmFsdWVzX2Zyb20gPSB2YWx1ZXMpCmBgYAoKTG8gcHJhY3RpY2FyZW1vcywgeSB2ZXJlbW9zIG3DqXRvZG9zIG3DoXMgc2VuY2lsbG9zIHBhcmEgaGFjZXIgdGFibGFzIGNvbiBlc3RhZMOtc3RpY29zIGRlc2NyaXB0aXZvcywgcGVybyBlc28gc2Vyw6EgZW4gZWwgdHV0b3JpYWwgZGVkaWNhZG8gYSB0YWJsYXM7IHBlcm8gbm8gb3Mgb2x2aWTDqWlzIGRlIGBhY3Jvc3MoKWAgcXVlIGx1ZWdvIHRlbmVtb3MgcXVlIHZvbHZlciBhIGVsbGEuCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIGBncm91cF9ieSgpYAoKQ29uIGVzdGEgZnVuY2nDs24geWEgZW1wZXphcmVtb3MgYSB2ZXIgbGEgcG90ZW5jaWEgZGUgZHBseXIuIEVuIGFuw6FsaXNpcyBkZSBkYXRvcyBtdWNoYXMgb3BlcmFjaW9uZXMgKG1lZGlhIGV0Yy4uKSBxdWVyZW1vcyBjYWxjdWxhcmxhcyBwYXJhIGRpc3RpbnRvcyBncnVwb3MgKGhvbWJyZSwgbXVqZXIgLi4uKS4gYGdyb3VwX2J5KClgIHBlcm1pdGUgaGFjZXJsby4KCmBncm91cF9ieSgpYGNvZ2UgdW4gZGYgeSBsbyBjb252aWVydGUgZW4gdW4gKioiZGYgYWdydXBhZG8iKiouIEVuIGVzZSBudWV2byAiZGYgYWdydXBhZG8iLCBsYXMgb3BlcmFjaW9uZXMgcXVlIGhhZ2Ftb3MgY29uIGBzdW1tYXJpc2UoKWAgc2UgaGFyw6FuIHBvciBzZXBhcmFkbyBwYXJhIGNhZGEgdW5vIGRlIGxvcyBncnVwb3MgcXVlIGhheWFtb3MgZGVmaW5pZG8uIEFob3JhIGxvIHZlbW9zLgoKU2ksIHBvciBlamVtcGxvLCBhZ3J1cGFtb3MgdW4gZGYgcG9yIHBhw61zZXMsIGFsIGVqZWN1dGFyIGBzdW1tYXJpc2UoKWAsIG5vcyByZXRvcm5hcsOhIHVuYSBmaWxhIGNvbiBlbCByZXN1bHRhZG8gcGFyYSBjYWRhIHBhw61zLiBFbiByZWFsaWRhZCwgcG9kZW1vcyBwZW5zYXIgcXVlIGBncm91cF9ieSgpYCBubyBoYWNlICJuYWRhIiwgcXVlIGVuIHJlYWxpZGFkIHNvbG8gY2FtYmlhIGxvIHF1ZSBoYWNlbiBsYXMgb3RyYXMgZnVuY2lvbmVzOiBhaG9yYSBsb3MgY8OhbGN1bG9zIHNlIGhhcsOhbiBwYXJhIGNhZGEgdW5vIGRlIGxvcyBncnVwb3MgcXVlIGRlZmluZSBgZ3JvdXBfYnkoKWAKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0KCkNvbW8gZGljZSBKZW5ueTogTGV04oCZcyBzdGFydCB3aXRoIHNpbXBsZSBjb3VudGluZy4gwr9DdWFudGFzIG9ic2VydmFjaW9uZXMocm93cykgdGVuZW1vcyBwb3IgY29udGluZW50ZT8KCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQojLSBjb2dlbW9zIGRmIHkgbG8gKGRlcylhZ3J1cGFtb3MgcG9yIGdydXBvcyBkZWZpbmlkb3MgcG9yIGxhIHZhcmlhYmxlICJjb250aW5lbnQiOyBvc2VhLCBoYWJyw6EgNSBncnVwb3MKIy0gZGVzcHXDqXMgY29uIHN1bW1hcmlzZSgpIGNhbGN1bGFyZW1vcyBlbCBuwrogZGUgb2JzZXJ2YWNpb25lcyBlbiBjYWRhIGNvbnRpbmVudGUgbyBncnVwbzsgZXMgZGVjaXIsIG5vcyByZXRvcm5hcsOhIHVuIGRmIGNvbiB1bmEgZmlsYSBwb3IgY2FkYSBjb250aW5lbnRlCmFhIDwtIGdhcG1pbmRlciAlPiUgZ3JvdXBfYnkoY29udGluZW50KSAlPiUgc3VtbWFyaXNlKE5OID0gbigpKSAKYWEKYGBgCgpFc3RvIHRhbiBzZW5jaWxsbyB0YW1iacOpbiBzZSBwdWVkZSBoYWNlciBjb24gYGNvdW50KClgCgpgYGB7ciwgZXZhbCA9IEZBTFNFfQphYSA8LSBnYXBtaW5kZXIgJT4lIGdyb3VwX2J5KGNvbnRpbmVudCkgJT4lIGNvdW50KCkgCmFhIDwtIGdhcG1pbmRlciAlPiUgZ3JvdXBfYnkoY29udGluZW50KSAlPiUgY291bnQobmFtZSA9ICJOTiIpIAphYQpgYGAKCgo8YnI+CgrCv1kgY3VhbnRvcyBwYcOtc2VzIGhheSBlbiBsYSBiYXNlIGRlIGRhdG9zPyBQYXJhIGVzdGUgdGlwbyBkZSBjb3Nhcywgc2UgcHVlZGVuIHVzYXIgZnVuY2lvbmVzIGRlIFItYmFzZSwgcGVybyBkcGx5ciB0aWVuZSBtdWNoYXMgZnVuY2lvbmVzIGF1eGlsaWFyZXMuCgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gY29nZW1vcyBkZiB5IGxvIGFncnVwYW1vcyBwb3IgImNvbnRpbmVudCIsIAojLSBkZXNwdcOpcyBjYWxjdWxhbW9zIDIgY29zYXM6IGVsIG7Dum1lcm8gZGUgb2JzZXJ2YWNpb25lcyBvIHJvd3MKIy0geSBlbCBuw7ptZXJvIGRlIHBhw61zZXMgZW4gY2FkYSBjb250aW5lbnRlIChOTl9jb3VudHJpZXMpCmFhIDwtIGdhcG1pbmRlciAlPiUgZ3JvdXBfYnkoY29udGluZW50KSAlPiUgIAogICAgICAgICAgc3VtbWFyaXplKE5OID0gbigpLCAKICAgICAgICAgICAgICAgICAgICBOTl9jb3VudHJpZXMgPSBuX2Rpc3RpbmN0KGNvdW50cnkpKSAKYWEKYGBgCgoKPGJyPgoKQ2FsY3VsZW1vcyBsYSBlc3BlcmFuemEgZGUgdmlkYSBtZWRpYSBwb3IgY29udGluZW50ZQoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gY29nZW1vcyBkZiB5IGxvIGFncnVwYW1vcyBwb3IgImNvbnRpbmVudCIsIGRlc3B1w6lzIGNhbGN1bGFtb3MgbGEgbWVkaWEgZGUgImxpZmVFeHAiCmFhIDwtIGdhcG1pbmRlciAlPiUgZ3JvdXBfYnkoY29udGluZW50KSAlPiUgIAogICAgICAgICAgICAgICAgICAgIHN1bW1hcml6ZShtZWFuKGxpZmVFeHApKSAKYWEKYGBgCgpHdWF1ISBIYXkgcXVlIGlyc2UgYSB2aXZpciBhIE9jZWFuw61hISEKCgo8YnI+CgpDYWxjdWxlbW9zIGxhIGVzcGVyYW56YSBkZSB2aWRhIG1lZGlhIHBvciBjb250aW5lbnRlIGVuIGVsIHByaW1lciBwZXJpb2RvICgxOTUyKQoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gY29nZW1vcyBkZiB5IGZpbHRyYW1vcyBwYXJhIHF1ZWRhcm5vcyBjb24gbGFzIG9ic2VydmFjaW9uZXMgZGUgMTk1MgojLSBkZXNwdcOpcyBsbyBhZ3J1cGFtb3MgcG9yICJjb250aW5lbnQiLCAKIy0gZGVzcHXDqXMgY2FsY3VsYW1vcyBsYSBtZWRpYSBkZSAibGlmZUV4cCIKZ2FwbWluZGVyICU+JSBmaWx0ZXIoeWVhciA9PSAiMTk1MiIpICU+JSAgCiAgICAgICAgICAgICAgZ3JvdXBfYnkoY29udGluZW50KSAlPiUgIAogICAgICAgICAgICAgIHN1bW1hcml6ZShtZWFuKGxpZmVFeHApKSAKCmBgYAoKSGFicsOtYSBzaWRvIG1lam9yIGVuIGx1Z2FyIGRlIHBvbmVyIGBmaWx0ZXIoeWVhciA9PSAiMTk1MiIpYCBoYWJlciBwdWVzdG8gYGZpbHRlcih5ZWFyID09IG1pbih5ZWFyKSlgCgpHdWF1ISBIYWJyw61hIHF1ZSBoYWJlciB2aXZpZG8gZW4gT2NlYW7DrWEgKGVuIDE5NTIpISEKCjxicj4KClNlIHB1ZWRlbiBjYWxjdWxhciB2YXJpb3MgZXN0YWTDrXN0aWNvcyBhIGxhIHZlegoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CiMtIGNvZ2Vtb3MgZGYgeSBmaWx0cmFtb3MgKGNvZ2Vtb3MpIGxhcyBvYnNlcnZhY2lvbmVzIGRlIDE5NTIgeSAyMDA3CiMtIGFncnVwYW1vcyBwb3IgImNvbnRpbmVudCIsIAojLSBkZXNwdcOpcyBjYWxjdWxhbW9zIGxhIG1lZGlhIGRlICJsaWZlRXhwIiB5IGRlICJnZHBQZXJjYXAiCmdhcG1pbmRlciAlPiUgZmlsdGVyKHllYXIgJWluJSBjKDE5NTIsIDIwMDcpKSAlPiUgIAogICAgICAgICAgICAgZ3JvdXBfYnkoY29udGluZW50LCB5ZWFyKSAlPiUgIAogICAgICAgICAgICAgc3VtbWFyaXplKG1lYW4obGlmZUV4cCksIG1lYW4oZ2RwUGVyY2FwKSkgCmBgYAoKVmFtb3MgYSBoYWNlciBjw6FsY3Vsb3MgY2FkYSB2ZXogbcOhcyBjb21wbGVqb3M6CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KIy0gY29nZW1vcyBkZiB5IGxvIGFncnVwYW1vcyBwb3IgImNvbnRpbmVudCIgeSAieWVhciIsIAojLSBkZXNwdcOpcyBjYWxjdWxhbW9zIGxhIG1lZGlhIGRlICJsaWZlRXhwIiB5IGRlICJnZHBQZXJjYXAiCmdhcG1pbmRlciAlPiUgZmlsdGVyKHllYXIgJWluJSBjKDE5NTIsIDIwMDcpKSAlPiUKICAgICAgICAgICAgICBncm91cF9ieShjb250aW5lbnQsIHllYXIpICU+JSAKICAgICAgICAgICAgICAjLSBkZXNwdcOpcyBjYWxjdWxhbW9zIGxhIG1lZGlhIGRlICJsaWZlRXhwIiAKICAgICAgICAgICAgICBzdW1tYXJpc2UobWVkaWEgPSBtZWFuKGxpZmVFeHApKQpgYGAKCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KIy0gVm95IGEgY3JlYXIgdW4gbnVldm8gZGY6ICJnYXBtaW5kZXJfZ3IiIG8gImdhcG1pbmRlciBhZ3J1cGFkbyIKZ2FwbWluZGVyX2dyIDwtIGdhcG1pbmRlciAlPiUgZmlsdGVyKHllYXIgJWluJSBjKDE5NTIsIDIwMDcpKSAlPiUKICAgICAgICAgICAgICAgICBncm91cF9ieShjb250aW5lbnQsIHllYXIpIAojLSB5IHNvYnJlICJnYXBtaW5kZXJfZ3IiIGlyZW1vcyBoYWNpZW5kbyBjw6FsY3Vsb3MKICAKIy0gc2kgcXVlcmVtb3MgY2FsY3VsYXIgbGEgbWVkaWEgZGUgdmFyaWFzIHZhcmlhYmxlcyB0ZW5lbW9zIHF1ZSB1c2FyIGFjcm9zcygpCmdhcG1pbmRlcl9nciAlPiUgc3VtbWFyaXNlKGFjcm9zcyhjKGxpZmVFeHAsIGdkcFBlcmNhcCksIG1lYW4pKQoKIy0gc2kgcXVlcmVtb3MgY2FsY3VsYXIgbGEgbWVkaWEgZGUgdG9kYXMgbGFzIHZhcmlhYmxlcyBudW3DqXJpY2FzIHRlbmVtb3MgcXVlIHVzYXIgYWNyb3NzKCkgeSB3aGVyZSgpCmdhcG1pbmRlcl9nciAlPiUgc3VtbWFyaXNlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgbWVhbikpCgojLSBzaSBxdWVyZW1vcyBjYWxjdWxhciBsYSBtZWRpYSB5IGxhIG1lZGlhbmEsIGhheSBxdWUgdXNhciBsaXN0KCkKZ2FwbWluZGVyX2dyICU+JSBzdW1tYXJpc2UoYWNyb3NzKGMobGlmZUV4cCwgZ2RwUGVyY2FwKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBsaXN0IChtZWRpYSA9IG1lYW4sIG1lZGlhbmEgPSBtZWRpYW4pICkpCgojLSBzaSBwb25lbW9zIGxvcyBub21icmVzIGRlIGxvcyBhcmd1bWVudG9zIHF1ZWRhcsOtYSBjb21vCmdhcG1pbmRlcl9nciAlPiUgc3VtbWFyaXNlKGFjcm9zcyguY29scyA9IGMobGlmZUV4cCwgZ2RwUGVyY2FwKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAuZm5zID0gbGlzdCAobWVkaWEgPSBtZWFuLCBtZWRpYW5hID0gbWVkaWFuKSkpCgojLSBhZGVtw6FzLCBwb2RlbW9zIGNvbnRyb2xhciBlbCBub21icmUgZGUgbGFzIHZhcmlhYmxlcyBjcmVhZGFzIGNvbiBlbCBhcmd1bWVudG8gLm5hbWVzCmdhcG1pbmRlcl9nciAlPiUgc3VtbWFyaXNlKGFjcm9zcyhjKGxpZmVFeHAsIGdkcFBlcmNhcCksIAogICAgICAgICAgICAgICAgICAgICAgICBsaXN0IChtZWRpYSA9IG1lYW4sIG1lZGlhbmEgPSBtZWRpYW4pLCAKICAgICAgICAgICAgICAgICAgICAgICAgLm5hbWVzID0gIntmbn1fe2NvbH0iKSkKYGBgCgo8YnI+CgojIyMjIFByZWd1bnRhcyBkZSB2ZXJkYWQKCkJ1ZW5vLCBwdWVzIHlhIGNvbm9jw6lpcyBsbyBwcmluY2lwYWwsIGxvIGLDoXNpY28geSBtw6FzIGltcG9ydGFudGUgZGUgYGRwbHlyYCwgc29sbyBxdWVkYSBpciBjb2dpZW5kbyBwcsOhY3RpY2EgeSBjb25maWFuemEsIGFzw60gcXVlIHBhcmEgZWxsbyB0b2NhIGhhY2VyIHVuYSBzZXJpZSBkZSAqKnByZWd1bnRhcyBkZSB2ZXJkYWQhISoqLiBQb3IgZWplbXBsbzogCgoxLiAgwr9lbiBxdWUgY29udGluZW50ZSBoYSBhdW1lbnRhZG8gbcOhcyBsYSBlc3BlcmFuemEgZGUgdmlkYSBlbiBlbCBwZXJpb2RvIDE5NTItMjAwNz8KCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQojLSBjb2dlbW9zIGRmIHkgbG8gYWdydXBhbW9zIHBvciAiY29udGluZW50IiwgZGVzcHXDqXMgY2FsY3VsYW1vcyBsYSBtZWRpYSBkZSAibGlmZUV4cCIKZ2FwbWluZGVyICU+JSAKICBmaWx0ZXIoeWVhciAlaW4lIGMoMTk1MiwgMjAwNykpICU+JSAgCiAgZ3JvdXBfYnkoY29udGluZW50LCB5ZWFyKSAlPiUgCiAgc3VtbWFyaXplKG1lZGlhID0gbWVhbihsaWZlRXhwKSkgJT4lIHVuZ3JvdXAoKQpgYGAKCgpDYXNpLCBwZXJvIG5vISEgU8OzbG8gaGVtb3MgY29uc2VndWlkbyB2ZXIgbGEgZXNwZXJhbnphIGRlIHZpZGEgcG9yIGNvbnRpbmVudGUgZW4gMTk1MiB5IDIwMDcuRW4gcmVhbGlkYWQgZXN0byB5YSBsbyBoaWNpbW9zIGFudGVzLiBGYWx0YSByZXN0YXIuIAoKUXVpesOhcyBwb2Ryw61hbW9zIGNhbGN1bGFyIGVsIG3DoXhpbW8sIGVsIG3DrW5pbW8geSByZXN0YXJsb3MuIE5vLCBwb3JxdWUgc3Vwb25kcsOtYW1vcyBxdWUgImxpZmVFeHAiIHNpZW1wcmUgYXVtZW50YS4gVmFtb3MgcXVlIGVsIHRpZW1wbyBhcHJlbWlhOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gcHJpbWVyIGludGVudG86IHNlIHB1ZWRlIGhhY2VyIGRlIHVuYSB2ZXosIHBlcm8gdmFtb3MgYSBwYXJ0aXIgZWwgY8OzZGlnbyBlbiAyIHRyb3pvcwphYSA8LSBnYXBtaW5kZXIgJT4lIGZpbHRlcih5ZWFyICVpbiUgYygxOTUyLCAyMDA3KSkgJT4lICAKICBncm91cF9ieShjb250aW5lbnQsIHllYXIpICU+JSAKICBzdW1tYXJpemUobWVkaWEgPSBtZWFuKGxpZmVFeHApKSAlPiUgdW5ncm91cCgpIAoKYWExIDwtIGFhICU+JSBncm91cF9ieShjb250aW5lbnQpICU+JSAKICBzdW1tYXJpc2UobWluX2wgPSBtaW4obWVkaWEpLCBtYXhfbCA9IG1heChtZWRpYSkpICU+JSAKICBtdXRhdGUoZGlmID0gbWF4X2wtbWluX2wpICU+JSAKICBhcnJhbmdlKGRlc2MoZGlmKSkKCmFhMQpgYGAKCkFzaWEgc29uIGxvcyBnYW5hZG9yZXMuIEVuIHByb21lZGlvLCBlbiBBc8OtYSBzZSBtZWpvcsOzIGxhIGVzcGVyYW56YSBkZSB2aWRhIGVuIDI0LjQgYcOxb3MgZW50cmUgMTk1MiB5IDIwMDcuCgpMbyBkZSByZXN0YXIgZWwgbcOheGltbyB5IGVsIG3DrW5pbW8gaGEgZnVuY2lvbmFkbywgUEVSTyBwb2Ryw61hIG5vIGhhYmVybG8gaGVjaG8gc2kgaHViaWVzZSBoYWJpZG8gYWxnw7puIGNvbnRpbmVudGUgZW4gZWwgcXVlIGVuIGxhIGVzcGVyYW56YSBkZSB2aWRhLCBlbiBsdWdhciBkZSBoYWJlciBhdW1lbnRhZG8sIGh1Ymllc2UgYmFqYWRvLiBFbnRvbmNlcywgwr9jw7NtbyBsbyBoYWNlbW9zPwoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gc2VndW5kbyBpbnRlbnRvOiBzZSBwdWVkZSBoYWNlciBkZSB1bmEgdmV6LCBwZXJvIHZhbW9zIGEgcGFydGlyIGVsIGPDs2RpZ28gZW4gMiB0cm96b3MKYWEgPC0gZ2FwbWluZGVyICU+JSBmaWx0ZXIoeWVhciAlaW4lIGMoMTk1MiwgMjAwNykpICU+JSAgCiAgICAgICAgIGdyb3VwX2J5KGNvbnRpbmVudCwgeWVhcikgJT4lIAogICAgICAgICBzdW1tYXJpemUobWVkaWEgPSBtZWFuKGxpZmVFeHApKSAlPiUgdW5ncm91cCgpIAoKIy0gdXNhbW9zIGxhZygpCmFhMSA8LSBhYSAlPiUgZ3JvdXBfYnkoY29udGluZW50KSAlPiUgCiAgICAgICAgICAgICAgYXJyYW5nZSh5ZWFyKSAlPiUKICAgICAgICAgICAgICBtdXRhdGUodmFyaWFjX2wgPSBtZWRpYSAtIGxhZyhtZWRpYSkpCgojLSBtb3N0cmFtb3MgbG9zIHJlc3VsdGFkb3MKYWExICU+JSBmaWx0ZXIoeWVhciA9PSAyMDA3KSAlPiUgYXJyYW5nZShkZXNjKHZhcmlhY19sKSkKYGBgCgoKU8OtLCBBc2lhIGVzIGxhIGdhbmFkb3JhLiBFbiBwcm9tZWRpbywgZW4gQXPDrWEgc2UgbWVqb3LDsyBsYSBlc3BlcmFuemEgZGUgdmlkYSBlbiAyNCBhw7FvcyBlbnRyZSAxOTUyIHkgMjAwNy4gCgpPdHJhIGZvcm1hIGRlIG9idGVuZXIgZWwgbWlzbW8gcmVzdWx0YWRvOgoKYGBge3J9CiMtIGVzdGEgcGFydGUgZXMgY29tw7puCmFhIDwtIGdhcG1pbmRlciAlPiUgCiAgZmlsdGVyKHllYXIgJWluJSBjKDE5NTIsIDIwMDcpKSAlPiUgIAogIGdyb3VwX2J5KGNvbnRpbmVudCwgeWVhcikgJT4lIAogIHN1bW1hcml6ZShtZWRpYSA9IG1lYW4obGlmZUV4cCkpICU+JSB1bmdyb3VwKCkKCiMtIHBlcm8gYWhvcmEgdXNhbW9zIHBpdm90X3dpZGVyKCkKYWEgJT4lIHBpdm90X3dpZGVyKG5hbWVzX2Zyb20gPSB5ZWFyLCB2YWx1ZXNfZnJvbSA9IG1lZGlhKSAlPiUgCiAgICAgbXV0YXRlKGRpZl9sID0gYDIwMDdgIC0gYDE5NTJgKSAlPiUgCiAgICAgYXJyYW5nZShkZXNjKGRpZl9sKSkKYGBgCgo8YnI+CgoKRWwgY2h1bmsgZGUgYWJham8sIMK/cXXDqSBoYWNlPyDCv3F1w6kgc2UgZXN0w6EgY2FsY3VsYW5kbz86CgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KYWEgPC0gZ2FwbWluZGVyICU+JSAKICBncm91cF9ieShjb250aW5lbnQsIHllYXIpICU+JSAKICBzZWxlY3QoY29udGluZW50LCB5ZWFyLCBsaWZlRXhwKSAlPiUgCiAgc3VtbWFyaXNlKG1lYW5fbGlmZSA9IG1lYW4obGlmZUV4cCkpICU+JSAKICBhcnJhbmdlKHllYXIpICU+JSAKICBtdXRhdGUoaW5jcmVfbWVhbl9saWZlXzAgPSBtZWFuX2xpZmUgLSBmaXJzdChtZWFuX2xpZmUpKSAlPiUgCiAgbXV0YXRlKGluY3JlX21lYW5fbGlmZV90ID0gbWVhbl9saWZlIC0gbGFnKG1lYW5fbGlmZSkpICU+JSAKICBhcnJhbmdlKGNvbnRpbmVudCkKCiMtIHBvciBlamVtcGxvIHZlYW1vcyBlbCByZXN1bHRhZG8gcGFyYSBFdXJvcGUKYWEgJT4lIGZpbHRlcihjb250aW5lbnQgPT0gIkV1cm9wZSIpCmBgYAoKClNlZCBjb25zY2llbnRlcyBkZSBxdWUgbGEgc29sdWNpb25lcyBhIHVuYSBwcmVndW50YSBubyBzYWxlIGEgbGEgcHJpbWVyYSwgYSB2ZWNlcyBoYXkgcXVlIGNhbGVudGFyc2UgZWwgY2FwOgoKPiBCcmVhayB0aGUgY29kZSBpbnRvIHBpZWNlcywgc3RhcnRpbmcgYXQgdGhlIHRvcCwgYW5kIGluc3BlY3QgdGhlIGludGVybWVkaWF0ZSByZXN1bHRzLiBUaGF04oCZcyBjZXJ0YWlubHkgaG93IEkgd2FzIGFibGUgdG8gd3JpdGUgc3VjaCBhIHRoaW5nLiBUaGVzZSBjb21tYW5kcyBkbyBub3QgbGVhcCBmdWxseSBmb3JtZWQgb3V0IG9mIGFueW9uZeKAmXMgZm9yZWhlYWQg4oCTIHRoZXkgYXJlIGJ1aWx0IHVwIGdyYWR1YWxseSwgd2l0aCBsb3RzIG9mIGVycm9ycyBhbmQgcmVmaW5lbWVudHMgYWxvbmcgdGhlIHdheS4gSXMgdGhlIHN0YXRlbWVudCBhYm92ZSByZWFsbHkgaGFyZCBmb3IgeW91IHRvIHJlYWQ/IElmIHllcywgdGhlbiBieSBhbGwgbWVhbnMgYnJlYWsgaXQgaW50byBwaWVjZXMgYW5kIG1ha2Ugc29tZSBpbnRlcm1lZGlhdGUgb2JqZWN0cy4gWW91ciBjb2RlIHNob3VsZCBiZSBlYXN5IHRvIHdyaXRlIGFuZCByZWFkIHdoZW4geW914oCZcmUgZG9uZS4gICAgLS0tLSBKZW5ueSBCcnlhbgoKPGJyPgoKT3RyYXMgY3Vlc3Rpb25lcyBxdWUgcG9kZW1vcyByZXNvbHZlciBjb24gZHBseXIgc29icmUgbGEgZXNwZXJhbnphIGRlIHZpZGE6CgotIMK/Q8OzbW8gaGEgZXZvbHVjaW9uYWRvIGxhIGVzcGVyYW56YSBkZSB2aWRhIGVuIFNwYWluIGx1c3RybyBhIGx1c3Rybz8KCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CiMtIHZhcmlhY2nDs24gZGUgbGlmZUV4cCBlbiBTcGFpbiBhw7FvIGEgYcOxbyAoYnVlbm8gbHVzdHJvIGEgbHVzdHJvKQphYSA8LSBnYXBtaW5kZXIgJT4lIAogIGdyb3VwX2J5KGNvdW50cnkpICU+JSAKICBzZWxlY3QoY291bnRyeSwgeWVhciwgbGlmZUV4cCkgJT4lIAogIG11dGF0ZShsaWZlRXhwX2dhaW5fY2FkYV9sdXN0cm8gPSBsaWZlRXhwIC0gbGFnKGxpZmVFeHApKSAlPiUgCiAgZmlsdGVyKGNvdW50cnkgPT0gIlNwYWluIiApCmFhCmBgYAoKPGJyPgoKLSDCv1kgbGEgdmFyaWFjacOzbiBhY3VtdWxhZGE/IEbDoWNpbCEhIFPDs2xvIHRlbmRyw61hbW9zIHF1ZSBzdW1hciBvIGFjdW11bGFyIGxhIHZhcmlhYmxlICJsaWZlRXhwX2dhaW5fY2FkYV9sdXN0cm8iIHF1ZSBoZW1vcyBnZW5lcmFkbyBhbnRlcmlvcm1lbnRlLCBhc8OtIHF1ZSBzw7NsbyBoYWJyw61hIHF1ZSBhw7FhZGlyIHVuYSBsaW5lYSBhIG51ZXN0cm8gY8OzZGlnbzoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CiMtIGdhbmFuY2lhIGFjdW11bGFkYQphYSA8LSBnYXBtaW5kZXIgJT4lIAogIGdyb3VwX2J5KGNvdW50cnkpICU+JSAKICBzZWxlY3QoY291bnRyeSwgeWVhciwgbGlmZUV4cCkgJT4lIAogIG11dGF0ZShsaWZlRXhwX2dhaW5fY2FkYV9sdXN0cm8gPSBsaWZlRXhwIC0gbGFnKGxpZmVFeHApKSAlPiUgCiAgIy0tLSAyIGZpbGFzIG51ZXZhczogaWZlbHNlKCkgIHkgY3Vtc3VtKCkKICBtdXRhdGUobGlmZUV4cF9nYWluX2NhZGFfbHVzdHJvMiA9IGlmZWxzZShpcy5uYShsaWZlRXhwX2dhaW5fY2FkYV9sdXN0cm8pLCAwLCBsaWZlRXhwX2dhaW5fY2FkYV9sdXN0cm8pKSAlPiUgCiAgbXV0YXRlKGxpZmVFeHBfZ2Fpbl9hY3VtdWxhZG8gPSBjdW1zdW0obGlmZUV4cF9nYWluX2NhZGFfbHVzdHJvMikpICU+JSAgIAogIGZpbHRlcihjb3VudHJ5ID09ICJTcGFpbiIpCmFhCmBgYAoKQWwgZmluYWwgcGFyYSBoYWNlcmxvIChjb21vIGhhYsOtYSBwZW5zYWRvKSBtZSBoYW4gaGVjaG8gZmFsdGEgMiBsaW5lYXMsIHBvcnF1ZSBsYSBwcmltZXJhIG9ic2VydmFjacOzbiBkZSAibGlmZUV4cF9nYWluX2NhZGFfbHVzdHJvIiBlcyB1biBOQSB5IGVzbyBoYWPDrWEgcXVlIGxhIGZ1bmNpw7NuIGBjdW1zdW0oKWAgbm8gZnVuY2lvbmFzZS4KCjxicj4KCi0gT3RyYSBmb3JtYSBkZSBoYWNlciBsbyBtaXNtbyBzZXLDrWEgKHNlIG1lIGhhIG9jdXJyaWRvIGRlc3B1w6lzKS4gQWRlbcOhcyBlcyBtw6FzIGbDoWNpbAoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gZ2FuYW5jaWEgYWN1bXVsYWRhIChvdHJhIGZvcm1hIGRlIGhhY2VyIGxvIG1pc21vKQphYSA8LSBnYXBtaW5kZXIgJT4lIAogIGdyb3VwX2J5KGNvdW50cnkpICU+JSAKICBzZWxlY3QoY291bnRyeSwgeWVhciwgbGlmZUV4cCkgJT4lIAogIG11dGF0ZShsaWZlRXhwX2dhaW5fYWN1bXVsYWRhID0gbGlmZUV4cCAtIGxpZmVFeHBbMV0pICAlPiUgCiAgZmlsdGVyKGNvdW50cnkgPT0gIlNwYWluIikKYGBgCgo8YnI+CgotIE9idGVuZXIsIHBhcmEgY2FkYSBwZXJpb2RvLCBsb3MgKDMpIHBhw61zZXMgZGUgQXNpYSBjb24gTUFZT1IgbGlmZUV4cC4gVXNhcmVtb3MgdW5hIHZhcmlhbnRlIGRlIGBzbGljZSgpYCwgY29uY3LDqXRhbWVudGUgYHNsaWNlX21heCgpYAoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CmFhIDwtIGdhcG1pbmRlciAlPiUKICBmaWx0ZXIoY29udGluZW50ID09ICJBc2lhIikgJT4lCiAgc2VsZWN0KHllYXIsIGNvdW50cnksIGxpZmVFeHApICU+JQogIGdyb3VwX2J5KHllYXIpICU+JQogIHNsaWNlX21heChuID0gMywgbGlmZUV4cCkgJT4lIAogIGFycmFuZ2UoeWVhcikgCmBgYAoKClBhcmEgb2J0ZW5lciBsb3MgNCBwYcOtc2VzIGNvbiAqKk1FTk9SKiogImxpZmVFeHAiIHPDs2xvIHRlbmRyw61hbW9zIHF1ZSBzdXN0aXR1aXIgbGEgcXVpbnRhIGxpbmVhIHBvciBgc2xpY2VfbWluKG4gPSA0LCBsaWZlRXhwKWAKCjxicj4KCi0gT2J0ZW5lciwgcGFyYSBjYWRhIHBlcmlvZG8sIGxvcyBwYcOtc2VzIGRlIEFzaWEgY29uIG1heW9yIHkgbWVub3IgbGlmZUV4cC4KCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CiMtIE9idGVuZXIsIHBhcmEgY2FkYSBwZXJpb2RvLCBsb3MgcGHDrXNlcyBkZSBBc2lhIGNvbiBtYXlvciB5IG1lbm9yIGxpZmVFeHAuCmFhIDwtIGdhcG1pbmRlciAlPiUKICBmaWx0ZXIoY29udGluZW50ID09ICJBc2lhIikgJT4lCiAgc2VsZWN0KHllYXIsIGNvbnRpbmVudCwgY291bnRyeSwgbGlmZUV4cCkgJT4lCiAgZ3JvdXBfYnkoeWVhcikgJT4lCiAgZmlsdGVyKG1pbl9yYW5rKGRlc2MobGlmZUV4cCkpIDwgMiB8IG1pbl9yYW5rKGxpZmVFeHApIDwgMikgJT4lIAogIGFycmFuZ2UoeWVhcikgCmBgYAoKCkxhcyAyIMO6bHRpbWFzIGZ1bmNpb25lcyBxdWUgaGVtb3MgdXNhZG86IGBzbGljZV9taW4oKWAgIHkgYG1pbl9yYW5rKClgIHNvbiBmdW5jaW9uZXMgZGUgZHBseXIgcGVybyBubyBzb24gc29uIGZ1bmNpb25lcyBwcmluY2lwYWxlcywgZW4gY2llcnRhIGZvcm1hIHNvbiBhdXhpbGlhcmVzLgoKPGJyPgoKClBvZMOpaXMgdmVyIGxhcyBmdW5jaW9uZXMgYXV4aWxpYXJlcyBxdWUgdGllbmUgZHBseXIgZW4gbGEgc2VndW5kYSBww6FnaW5hIGRlIFtDSEVBVCBTSEVFVF0oaHR0cHM6Ly93d3cucnN0dWRpby5jb20vcmVzb3VyY2VzL2NoZWF0c2hlZXRzLykuIHRhbWJpw6luIHNlIHB1ZWRlbiB1c2FyIGxhcyBmdW5jaW9uZXMgZGUgUi1iYXNlIG8gZGUgb3Ryb3MgcGFja2FnZXMuIEFxdcOtIHRlbsOpaXMgYWxndW5hcyBwb3NpYmlsaWRhZGVzIHNhY2FkYXMgZGUgdW4gW3R1dG9yaWFsIGRlIEhhZGxleV0oaHR0cHM6Ly93d3cuZHJvcGJveC5jb20vc2gvaThxbmx1d211aWVpY3hjL0FBQWd0OXRJS29JbTdXWktJeUsyNWxoNmE/cHJldmlldz1kcGx5ci10dXRvcmlhbC5wZGYpCgoKYGBgcgpUeXBlcyBvZiBzdW1tYXJ5IGZ1bmN0aW9uczoK4oCiIG1pbih4KSwgbWVkaWFuKHgpLCBtYXgoeCksIHF1YW50aWxlKHgsIHApIArigKIgbigpLCBuX2Rpc3RpbmN0KCksIHN1bSh4KSwgbWVhbih4KSAK4oCiIHN1bSh4ID4gMTApLCBtZWFuKHggPiAxMCkgCuKAoiBzZCh4KSwgdmFyKHgpLCBpcXIoeCksIG1hZCh4KQoKVHlwZXMgb2Ygd2luZG93IGZ1bmN0aW9uczoK4oCiIFJhbmtpbmcgYW5kIG9yZGVyaW5nIArigKIgT2Zmc2V0czogbGVhZCAmIGxhZyAK4oCiIEN1bXVsYXRpdmUgYWdncmVnYXRlcwrigKIgUm9sbGluZyBhZ2dyZWdhdGVzCgpFamVtcGxvczoK4oCiIFdhcyB0aGVyZSBhIGNoYW5nZT8gIHggIT0gbGFnKHgpCuKAoiBQZXJjZW50IGNoYW5nZT8gKHggLSBsYWcoeCkpIC8geArigKIgRm9sZC1jaGFuZ2U/IHggLyBsYWcoeCkK4oCiIFByZXZpb3VzbHkgZmFsc2UsIG5vdyB0cnVlPyAhbGFnKHgpICYgeAoK4oCiIElmIG9uZSBvZiB0aGUgc3BlY2lhbGlzZWQgdmVyYnMgZG9lc27igJl0IGRvIHdoYXQgeW91IG5lZWQsIHlvdSBjYW4gdXNlIGRvKCkKYGBgCgo8YnI+CgojIyMjIyBBIHZlciBzaSBlbnRlbmRlaXMgZXN0ZSBlamVtcGxvCgpVbmEgKipmdW5jacOzbiBhdXhpbGlhcioqIHF1ZSBlcyAqKm11eSDDunRpbCoqIGFsIHV0aWxpemFybGEganVudG8gYSBtdXRhdGU6IGBjYXNlX3doZW4oKWAuIAoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmFhIDwtIGdhcG1pbmRlciAlPiUKICBncm91cF9ieShjb250aW5lbnQsIHllYXIpICAlPiUKICBtdXRhdGUobWVkaWFfbGlmZUV4cCA9IG1lYW4obGlmZUV4cCkpICU+JSAKICBtdXRhdGUobWVkaWFfZ2RwUGVyY2FwID0gbWVhbihnZHBQZXJjYXApKSAlPiUgCiAgbXV0YXRlKEdPT0Rfb3JfQkFEID0gY2FzZV93aGVuKCAKICAgIGxpZmVFeHAgPiBtZWFuKGxpZmVFeHApICYgZ2RwUGVyY2FwID4gbWVhbihnZHBQZXJjYXApICB+ICJnb29kIiwKICAgIGxpZmVFeHAgPCBtZWFuKGxpZmVFeHApICYgZ2RwUGVyY2FwIDwgbWVhbihnZHBQZXJjYXApICB+ICJiYWQiICwKICAgIGxpZmVFeHAgPCBtZWFuKGxpZmVFeHApIHwgZ2RwUGVyY2FwIDwgbWVhbihnZHBQZXJjYXApICB+ICJtZWRpdW0iCiAgICApKSAlPiUKICBmaWx0ZXIoY291bnRyeSA9PSAiU3BhaW4iKQpgYGAKCjxicj4KCiMjIyMjIE3DoXMgZnVuY2lvbmVzIGF1eGlsaWFyZXMKCkhheSBhbGd1bmFzIHF1ZSBubyBxdWllcm8gb2x2aWRhcjoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQpkcGx5cjo6bnRpbGUoeCwgbikgOiBjYXRlZ29yaXplcyBhIHZlY3RvciBvZiB2YWx1ZXMgaW50byAibnRpbGVzIiBzdWNoIGFzIHF1YXJ0aWxlcyBpZiBuID0gNAoKZHBseXI6Om5fZGlzdGluY3QoeCk6ICBjb3VudHMgdW5pcXVlIHZhbHVlcyBpbiBhIHZlY3Rvcjsgc2ltaWxhciBhICBsZW5ndGgodW5pcXVlKHgpKQoKZHBseXI6OmJldHdlZW4oeCwgbGVmdCwgcmlnaHQpIDogaXMgYSBzaG9ydGN1dCBmb3IgeCA+PSBsZWZ0ICYgeCA8PSByaWdodCwgCgp0aWJibGU6OmFkZF9yb3coKSAgOiBhw7FhZGUgcm93cyBhIHVuIGRmCgp0aWJibGU6OnJvd25hbWVzX3RvX2NvbHVtbigpCmBgYAoKPGJyPgoKCiMjIE3DoXMgZGV0YWxsZXMgc29icmUgZHBseXIKCkxvcyB2ZXJib3MgYG11dGF0ZSgpYCB5IGBzdW1tYXJpc2UoKWAgeWEgc2FiZW1vcyBxdWUgcHVlZGVuIGhhY2VyIHVzbyBkZSBmdW5jaW9uZXMgY29tbyBgbWVhbigpYCwgYHNkKClgIGV0YyAuLi4gUG9yIGVqZW1wbG8gcG9kZW1vcyB0cmFuc2Zvcm1hciBsYXMgdmFyaWFibGVzIG51bcOpcmljYXMgYSBsb2dhcml0bW9zOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ2FwbWluZGVyICU+JSBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCBsb2cpKSAlPiUgaGVhZChuID0gMykKYGBgCgpTZSBwdWVkZW4gdXNhciB0aWR5X2hlbHBlcnMgcGFyYSBzZWxlY2Npb25hciBsYXMgY29sdW1uYXMgeSBhcGxpY2FyIG3DoXMgZGUgdW5hIGZ1bmNpw7NuOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ2FwbWluZGVyICU+JSBtdXRhdGUoYWNyb3NzKGMoc3RhcnRzX3dpdGgoImxpZmUiKSwgY29udGFpbnMoInBvIikpLCAuZm5zID0gIG1lYW4pKSAlPiUgaGVhZApgYGAKCm8gY2FsY3VsYXIgbGEgbWVkaWEgZGUgbGFzIHZhcmlhYmxlIG51bcOpcmljYXMgeSBjb250cm9sYXIgZWwgbm9tYnJlIGRlIGxhcyB2YXJpYWJsZXMgY3JlYWRhczoKCmBgYHtyLCBldmFsID0gRkFMU0V9CmdhcG1pbmRlciAlPiUgZ3JvdXBfYnkoY29udGluZW50KSAlPiUKICBzdW1tYXJpemUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCBtZWFuLCAubmFtZXMgPSAibWVhbl97Y29sfSIpKSAlPiUgaGVhZChuID0gMykKYGBgCgpTaSBxdWllcmVzIHNlbGVjY2lvbmFyIHRvZGFzIGxhcyB2YXJpYWJsZXMgdXNhIGBldmVyeXRoaW5nKClgCgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ2FwbWluZGVyICU+JSBncm91cF9ieShjb250aW5lbnQpICU+JQogIHN1bW1hcml6ZShhY3Jvc3MoZXZlcnl0aGluZygpLCBhcy5jaGFyYWN0ZXIsIC5uYW1lcyA9ICJDSEFSX3tjb2x9IikpICU+JSBoZWFkKG4gPSAzKQpgYGAKClBlcm8gdmFtb3MgYSB2ZXIgb3RyYXMgcG9zaWJpbGlkYWRlczoKCjxicj4KCiMjIyBtw6FzIGRlIHVuYSBjb25kaWNpw7NuIHBhcmEgc2VsZWNjaW9uYXIgbGFzIGNvbHVtbmFzOgoKYGBge3J9CmdhcG1pbmRlciAlPiUgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5kb3VibGUpICYgZW5kc193aXRoKCJjYXAiKSwgYXMuaW50ZWdlcikpICU+JSBoZWFkKG4gPSAzKQpgYGAKCjxicj4KCiMjIyB1c28gZGUgZnVuY2lvbmVzIHByb3BpYXMgKCEhISEpCgpUZW5lbW9zIHZhcmlhcyBwb3NpYmlsaWRhZGVzOgoKLSAxLiBkZWZpbmllbmRvIHByaW1lcm8gbGEgZnVuY2nDs246CgoKYGBge3J9CmRpdmlkaXJfMTAwIDwtIGZ1bmN0aW9uKHgpIHt4IC8gMTAwfSAgIy0gZGVmaW5vIHVuYSBmdW5jacOzbgoKZ2FwbWluZGVyICU+JSBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCAuZm5zID0gZGl2aWRpcl8xMDApKSAlPiUgaGVhZApgYGAKCi0gMi4gdXNhbmRvIGZvcm11bGFzIGFuw7NuaW1hcwoKYGBge3J9CmdhcG1pbmRlciAlPiUgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgLmZucyA9IGZ1bmN0aW9uKHgpIHt4IC8gMTAwfSkpICU+JSBoZWFkCmBgYAoKLSAzLiB1c2FuZG8gZsOzcm11bGFzCgpgYGB7cn0KIy0gY29uIGZvcm11bGFzCmdhcG1pbmRlciAlPiUgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgLmZucyA9IH4gLngvMTAwKSkgJT4lIGhlYWQKCmdhcG1pbmRlciAlPiUgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgLmZucyA9IH4gezEvc3FydCguKX0pKSAlPiUgaGVhZApgYGAKCnVzYXIgZm9ybXVsYXMgZmFjaWxpdGEgZWwgdXNvIGRlIGFyZ3VtZW50b3MgZGVudHJvIGRlIGxhIGZ1bmNpw7NuOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ2FwbWluZGVyICU+JSAKICBncm91cF9ieShjb250aW5lbnQsIHllYXIpICU+JSAKICBzdW1tYXJpc2UoYWNyb3NzKGMoImxpZmVFeHAiLCAiZ2RwUGVyY2FwIiksIAoJCQkJICAgbGlzdChtZWFuID0gfiBtZWFuKC54LCBuYS5ybSA9IFRSVUUsIHRyaW0gPSAwLjEpKSkpCmBgYAoKCiMjIyBjcmVhciBpbmRpY2VzIGRlIGdydXBvCgpBIHZlY2VzIGN1YW5kbyB0cmFiYWphcyBjb24gZGYgYWdydXBhZG9zIGVzIMO6dGlsIHNhYmVyIGEgcXVlIGdydXBvIHBlcnRlbmVjZSBjYWRhIG9ic2VydmFjacOzbi4gUHVlZGVzIGhhY2VybG8gZsOhY2lsbWVudGUgY29uOiAKCgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpnYXBtaW5kZXIgJT4lIGdyb3VwX2J5KHllYXIpICAlPiUgbXV0YXRlKGlkX2dydXBvID0gY3VyX2dyb3VwX2lkKCkpIApgYGAKCgoKCjxicj48YnI+CgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMgNS4gQ29tYmluYW5kbyAoam9pbmluZykgZGYncwoKPGJyPgoKT0ssIHlhIHNhYmVtb3MgbWFuZWphci9maWx0cmFyIGV0Yy4uLiB1biBjb25qdW50byBkZSBkYXRvcywgUEVSTyBtdWNoYXMgdmVjZXMgbG8gcXVlIGhheSBxdWUgaGFjZXIgZXMgdW5pciBvICoqY29tYmluYXIgdmFyaWFzIHRhYmxhcyoqIG8gY29uanVudG9zIGRlIGRhdG9zIChKb2luaW5ncyBlbiBpbmdsw6lzKS4gCgoKVmFtb3MgYSBhcHJlbmRlciBjb21vIGhhY2VybG8gdXNhbmRvIGRwbHlyLiBbW0FxdcOtXShodHRwczovL2NyYW4uci1wcm9qZWN0Lm9yZy93ZWIvcGFja2FnZXMvZHBseXIvdmlnbmV0dGVzL3R3by10YWJsZS5odG1sKV0gdGVuw6lpcyBsYSB2aWduZXR0ZSBkZSBkcGx5ciBwYXJhICJ0d28gdGFibGUgdmVyYnMiLCB0YW1iacOpbiBwb2TDqWlzIHZlciB1biB2w61kZW8gbXV5IGlsdXN0cmF0aXZvIFtbYXF1w61dKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9UUhYcHNNU0RzemcgKV0uIFRhbWJpw6luIHBvZMOpaXMgdXNhciBlbCBbdHV0b3JpYWwgZGUgSmVubnldKGh0dHA6Ly9zdGF0NTQ1LmNvbS9iaXQwMDFfZHBseXItY2hlYXRzaGVldC5odG1sKS4gTGEgW0NIRUFUIFNIRUVUXShodHRwczovL3d3dy5yc3R1ZGlvLmNvbS9yZXNvdXJjZXMvY2hlYXRzaGVldHMvKSBlcyBtdXktbXV5IGJ1ZW5hLgoKCkxvcyBlamVtcGxvcyBlIGltw6FnZW5lcyB1c2Fkb3MgZW4gZXN0YSBzZWNjacOzbiBzZSBiYXNhbiBlbiBsb3MgY3JlYWRvcyBwb3IgW01hcmEgQXZlcmljayAoXEBkYXRhYW5kbWUpXShodHRwczovL3R3aXR0ZXIuY29tL2RhdGFhbmRtZT9sYW5nPWVzKSBlbiBbZXN0ZSByZXBvXShodHRwczovL2dpdGh1Yi5jb20vYmF0cGlnYW5kbWUvdGlkeWV4cGxhaW4vdHJlZS9waXZvdCksIHF1ZSBhIHN1IHZleiBzZSBiYXNhcm9uIGVuIGxhIGlkZWEgZGUgW0dhcnJpY2sgQWRlbi1CdWllICBcQGdycnJja10oaHR0cHM6Ly90d2l0dGVyLmNvbS9ncnJyY2spCgoKPGJyPgoKIyMgKGRvcykgY2Fzb3Mgc2VuY2lsbG9zCgojIyMjIERvcyBjYXNvcyBpZGVhbGVzIChzZW5jaWxsb3MgZGUgdW5pcik6IGBiaW5kX2NvbHMoKWAgeSBgYmluZF9yb3dzKClgCgoxLiBTaSBsb3MgMiBkZnMgdGllbmVuICoqZXhhY3RhbWVudGUgbGFzIG1pc21hcyBmaWxhcyoqIG8gdW5pZGFkZXMgZGUgYW7DoWxpc2lzICggeSBhZGVtYXMgZW4gZWwgbWlzbW8gb3JkZW4pLiBFbiBlc3RlIGNhc28sIHNvbG8gaGFicsOtYSBxdWUganVudGFyIGVuIHVuYSBtaXNtYSB0YWJsYSBsYXMgY29sdW1uYXMgZGUgZGYxIHkgZGUgZGYyLiBFc3RvIGxvIHBvZGVtb3MgaGFjZXIgY29uIGBiaW5kX2NvbHMoKWAgKG8gY29uICoqYyoqYmluZCgpIGRlIFItYmFzZSkuCgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZGZfMSA8LSBpcmlzWyAsIDE6Ml0gIDsgZGZfMiA8LSBpcmlzWyAsIDM6NV0KCmRmXzEgPC0gaXJpcyAlPiUgc2VsZWN0KDE6MikgIDsgZGZfMiA8LSBpcmlzICU+JSBzZWxlY3QoMzo1KSAKCmRmXzMgPC0gYGJpbmRfY29sc2AoZGZfMSwgZGZfMikKCmlkZW50aWNhbChpcmlzLCBkZl8zKQpgYGAKCgoyLiBTaSBsb3MgMiBkZnMgdGllbmVuICoqZXhhY3RhbWVudGUgbGFzIG1pc21hcyBjb2x1bW5hcyoqICggeSBhZGVtYXMgZW4gZWwgbWlzbW8gb3JkZW4pLiBFbiBlc3RlIGNhc28sIHNlIHRyYXRhcsOtYSBzaW1wbGVtZW50ZSBkZSBqdW50YXIgdG9kYXMgbGFzIG9ic2VydmFjaW9uZXMgbyBmaWxhcyBkZSBsb3MgMiBkZidzLiBFc3RvIGxvIHBvZGVtb3MgaGFjZXIgY29uIGBiaW5kX3Jvd3MoKWAgKG8gY29uICoqcioqYmluZCgpIGRlIFItYmFzZSkKCmBgYHtyLCBldmFsID0gRkFMU0V9CmRmXzEgPC0gaXJpc1sxOjc1LCBdICA7IGRmXzIgPC0gaXJpc1s3NjoxNTAsIF0KCmRmXzEgPC0gaXJpcyAlPiUgc2xpY2UoMTo3NSkgIDsgZGZfMiA8LSBpcmlzICU+JSBzbGljZSg3NjoxNTApIAoKZGZfMyA8LSBgYmluZF9yb3dzYChkZl8xLCBkZl8yKQoKaWRlbnRpY2FsKGlyaXMsIGRmXzMpCmBgYAoKCjxicj4KCiMjIyMgT2x2aWRhbmRvIHlhIGxvcyAyIGNhc29zIGlkZWFsZXMgeSBzZW5jaWxsb3MKCjxicj4KCkVuIGRwbHlyIGhheSAzIHRpcG9zIGRlIGZ1bmNpb25lcyh2ZXJib3MpIHF1ZSBzZSBvY3VwYW4gZGUgZGlmZXJlbnRlcyBvcGVyYWNpb25lcyBwYXJhIHVuaXIgZGF0YXNldHM6CgogIC0gKipNdXRhdGluZyBqb2lucyoqLCBhw7FhZGUgbnVldmFzIHZhcmlhYmxlcyAobyBjb2x1bW5hcykgYSB1biBkYXRhZnJhbWUgKGRmMSkuIEVzdGFzIG51ZXZhcyBjb2x1bW5hcyB2aWVuZW4gZGUgdW4gc2VndW5kbyBkZjIgKGhheSB2YXJpYXMgbXV0YXRpbmcgam9pbnMsIGRlcGVuZGllbmRvIGRlbCBjcml0ZXJpbyBwYXJhIHNlbGVjY2lvbmFyIGxhcyBmaWxhcykKCiAgLSAqKkZpbHRlcmluZyBqb2lucyoqLCBmaWx0cmEgbGFzIGZpbGFzIChvYnNlcnZhY2lvbmVzKSBkZSB1biBkYXRhZnJhbWUgKGRmMSkgYmFzw6FuZG9zZSBlbiBzw60gbGFzIGZpbGFzIGRlIGRmMSBjb2luY2lkZW4gKG1hdGNoKSBvIG5vIGNvbiB1bmEgb2JzZXJ2YWNpw7NuIGRlbCBzZWd1bmRvIGRmMgoKICAtICoqU2V0IG9wZXJhdGlvbnMqKiwgY29tYmluYSBsYXMgb2JzZXJ2YWNpb25lcyBkZSBsb3MgZG9zIGRhdGFzZXRzIChkZjEgeSBkZjIpIGFzIGlmIHRoZXkgd2VyZSBzZXQgZWxlbWVudHMuCgo8YnI+CgpUb2RhcyBlc3RhcyBmdW5jaW9uZXMgdGllbmVuIHVuYSAqKmVzdHJ1Y3R1cmEgc2ltaWxhcioqOiBzdXMgZG9zIHByaW1lcm9zIGFyZ3VtZW50b3Mgc29uIDIgZGYncyAoZW4gcmVhbGlkYWQgdGFibGFzIGRlIGRhdG9zKTogZGYxIHkgZGYyLiBFbCBvdXRwdXQgZGUgbGEgZnVuY2nDs24gZXMgc2llbXByZSB1bmEgbnVldmEgdGFibGEgKGRlbCBtaXNtbyB0aXBvIHF1ZSBkZjEpLgoKCjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIE11dGF0aW5nIGpvaW5zCgpIYXkgKio0IHRpcG9zIGRlIG11dGF0aW5nIGpvaW5zKiouIFN1IHNpbnRheGlzIGVzIGlkw6ludGljYSwgc8OzbG8gc2UgZGlmZXJlbmNpYW4gZW4gcXVlIGxhcyBmaWxhcyBxdWUgc2Ugc2VsZWNjaW9uYW4gZGVwZW5kZW4gZGVsIGNyaXRlcmlvIHBhcmEgaGFjZXIgZWwgbWF0Y2g6CgogIC0gYGlubmVyX2pvaW4oZGYxLGRmMilgOiBSZXRvcm5hIHRvZGFzIGxhcyBjb2x1bW5hcyBkZSBkZjEgeSB0YW1iacOpbiBsYXMgZGUgZGYyLCBQRVJPICoqc29sbyByZXRvcm5hIGxhcyBmaWxhcyBkZSBkZjEgcXVlIHRpZW5lbiB1bmEgZXF1aXZhbGVuY2lhIGVuIGRmMioqLiAobGEgZXF1aXZhbGVuY2lhIHNlIGRlZmluZSBlbiBmdW5jacOzbiBkZWwgdmFsb3IgZGUgdW5hIHZhcmlhYmxlIG8gdmFyaWFibGVzIGNvbXVuZXMgZW4gZGYxIHkgZGYyKQogIAogIC0gYGxlZnRfam9pbihkZjEsZGYyKWA6IFJldG9ybmEgdG9kYXMgbGFzIGNvbHVtbmFzIGRlIGRmMSB5IHRhbWJpw6luIGxhcyBkZSBkZjI7IGVuIGN1YW50byBhIGxhcyBmaWxhcywgKipyZXRvcm5hIFRPREFTIGxhcyBmaWxhcyBkZSBkZjEqKi4gKFNpIGh1Ymllc2VuIHZhcmlvcyBtYXRjaGVzIGVudHJlIGRmMSBlIGRmMiBzZSByZXRvcm5hbiB0b2RhcyBsYXMgY29tYmluYWNpb25lcyEhISEpCiAgCiAtIGByaWd0aF9qb2luKGRmMSxkZjIpYDogUmV0b3JuYSB0b2RhcyBsYXMgY29sdW1uYXMgZGUgZGYxIHkgdGFtYmnDqW4gbGFzIGRlIGRmMjsgZW4gY3VhbnRvIGEgbGFzIGZpbGFzLCAqKnJldG9ybmEgVE9EQVMgbGFzIGZpbGFzIGRlIGRmMioqLiBEZSAqKmRmMioqISEgKFNpIGh1Ymllc2VuIHZhcmlvcyBtYXRjaGVzIGVudHJlIGRmMSB5IGRmMiBzZSByZXRvcm5hbiB0b2RhcyBsYXMgY29tYmluYWNpb25lcyEhISEpICAKICAKCiAtIGBmdWxsX2pvaW4oZGYxLGRmMilgOiBSZXRvcm5hIHRvZGFzIGxhcyBjb2x1bW5hcyBkZSBkZjEgeSB0YW1iacOpbiBsYXMgZGUgZGYyOyBlbiBjdWFudG8gYSBsYXMgZmlsYXMsICoqcmV0b3JuYSBUT0RBUyBsYXMgZmlsYXMgZGUgZGYxIHkgZGUgZGYyKiouIE9zZWEsIHJldG9ybmEgVE9EQVMgbGFzIGZpbGFzIHkgVE9EQVMgbGFzIGNvbHVtbmFzIGRlIGxhcyAyIHRhYmxhcy4gKERvbmRlIG5vIGhheSBtYXRjaGVzIHJldG9ybmEgTkEncykKICAKICAgIAo8YnI+CgojIyMjIEVqZW1wbG9zIGRlIG11dGF0aW5nIGpvaW5zIAoKU2VhbiBsb3Mgc2lndWllbnRlcyAyIGRhdGFmcmFtZXMgKHRpYmJsZXMpOgoKYGBge3J9CnggPC0gdGliYmxlKGlkID0gMTozLCB4ID0gcGFzdGUwKCJ4IiwgMTozKSkKCnkgPC0gdGliYmxlKGlkID0gKDE6NClbLTNdLCB5ID0gcGFzdGUwKCJ5IiwgKDE6NClbLTNdKSkKYGBgCgoKPGJyPgoKIyMjIyMgSW5uZXIgSm9pbgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KIy0gb25seSBpbmNsdWRlcyBvYnNlcnZhdGlvbnMgdGhhdCBtYXRjaCBpbiBib3RoIHggYW5kIHkKZGZfaW5uZXIgPC0gaW5uZXJfam9pbih4LCB5KQpgYGAKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiNzUlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18wNl9pbm5lci1qb2luLnBuZyIpKQpgYGAKCgo8YnI+CgojIyMjIyBMZWZ0IEpvaW4KCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQojLSBpbmNsdWRlcyBhbGwgb2JzZXJ2YXRpb25zIGluIHgsIHJlZ2FyZGxlc3Mgb2Ygd2hldGhlciB0aGV5IG1hdGNoIG9yIG5vdC4gCiMtIFRoaXMgaXMgdGhlIG1vc3QgY29tbW9ubHkgdXNlZCBqb2luIGJlY2F1c2UgaXQgZW5zdXJlcyB0aGF0IHlvdSBkb27igJl0IGxvc2Ugb2JzZXJ2YXRpb25zIGZyb20geW91ciBwcmltYXJ5IHRhYmxlLgpkZl9sZWZ0X2pvaW4gPC0gbGVmdF9qb2luKHgsIHkpCmBgYAoKCmBgYHtyICwgZWNobz1GQUxTRSwgZXZhbCA9IFRSVUUsIGZpZy5hc3AgPSA0LzIsIG91dC53aWR0aCA9ICI3NSUiLCBmaWcuYWxpZ24gPSAiY2VudGVyIn0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzA3X2xlZnQtam9pbi5wbmciKSkKYGBgCgoKCjxicj4KCiMjIyMjIFJpZ2h0IEpvaW50CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQojLSBpbmNsdWRlcyBhbGwgb2JzZXJ2YXRpb25zIGluIHkuIAojLSBJdOKAmXMgZXF1aXZhbGVudCB0byBsZWZ0X2pvaW4oeSwgeCksIGJ1dCB0aGUgY29sdW1ucyB3aWxsIGJlIG9yZGVyZWQgZGlmZmVyZW50bHkuCmRmX3JpZ2h0X2pvaW4gPC0gcmlnaHRfam9pbih4LCB5KQpgYGAKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiNzUlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18wOF9yaWd0aC1qb2luLnBuZyIpKQpgYGAKCjxicj4KCiMjIyMjIEZ1bGwgSm9pbnQKCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQojLSBmdWxsX2pvaW4oKSBpbmNsdWRlcyBhbGwgb2JzZXJ2YXRpb25zIGZyb20geCBhbmQgeQpkZl9mdWxsX2pvaW4gPC0gZnVsbF9qb2luKHgsIHkpCmBgYAoKPGJyPgoKCmBgYHtyICwgZWNobz1GQUxTRSwgZXZhbCA9IFRSVUUsIGZpZy5hc3AgPSA0LzIsIG91dC53aWR0aCA9ICI3NSUiLCBmaWcuYWxpZ24gPSAiY2VudGVyIn0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzA5X2Z1bGwtam9pbi5wbmciKSkKYGBgCgoKIyMjIyMgMiBwcmVjaXNpb25lcyBzb2JyZSBsYXMgbXV0YXRpbmcgam9pbnMKCkxhcyAobGVmdCwgcmlnaHQgYW5kIGZ1bGwpIGpvaW5zIHNlIGxsYW1hbiBjb2xlY3RpdmFtZW50ZSBjb21vICJvdXRlciBqb2lucyIuIEN1YW5kbyB1bmEgZmlsYSBubyB0aWVuZSBtYXRjaCBlbiB1bmEgb3V0ZXIgam9pbiwgbGFzIG51ZXZhcyB2YXJpYWJsZXMgcXVlIHNlIGNyZWFuIHNlIGxsZW5hbiBjb24gTkEncy4KCkxhcyBtdXRhdGluZyBqb2lucyBzZSB1c2FuIHByaW5jaXBhbG1lbnRlIHBhcmEgYcOxYWRpciBjb2x1bW5hcywgKipQRVJPKiogZW4gZWwgcHJvY2VzbyBwdWVkZW4gZ2VuZXJhcnNlIG51ZXZhcyBmaWxhczogc2kgdW4gbWF0Y2ggbm8gZXMgw7puaWNvLCBzZSBhw7FhZGlyw6FuIHRvZGFzIGxhcyBjb21iaW5hY2lvbmVzIHBvc2libGVzIChlbCBwcm9kdWN0byBjYXJ0ZXNpYW5vKSBkZSBsYXMgbWF0Y2hpbmcgb2JzZXJ2YXRpb25zLiBWZWFtb3MgdW4gZWplbXBsbyBjb24gdW5hIGxlZnRfam9pbjoKCmBgYHtyfQp4IDwtIHRpYmJsZShpZCA9IDE6MywgeCA9IHBhc3RlMCgieCIsIDE6MykpCgp5IDwtIHRpYmJsZShpZCA9IGMoMTo0LDIpWy0zXSwgeSA9IHBhc3RlMCgieSIsIGMoMTo1KVstM10pKQpgYGAKPGJyPgoKIyMjIyMgbGVmdF9qb2luKCkgZW4gbGEgcXVlIHNlIGNyZWFuIG51ZXZhcyBmaWxhcwoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRX0KZGZfbGVmdF9qb2luIDwtIGxlZnRfam9pbih4LCB5KQpgYGAKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiNzUlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18xMF9sZWZ0LWpvaW4tZXh0cmEucG5nIikpCmBgYAoKCjxicj4KCiMjIyMgSW1wb3J0YW50ZSDCv0PDs21vIGRlY2lyIGEgbGFzIGZ1bmNpb25lcyBsYSBjb2x1bW5hcyAobyBjb2x1bW5hcykgcXVlIHNlIHVzYXLDoW4gcGFyYSBoYWNlciBsb3MgbWF0Y2hpbmc/CgpQb2RlbW9zKERFQkVNT1MpIGVsZWdpciBsYXMgY29sdW1uYXMgKG8gdmFyaWFibGVzKSBxdWUgbm9zIHNlcnZpcsOhbiBwYXJhIHVuaXIgbG9zIDIgZGYncy4gRXN0YXMgY29sdW1uYXMgcXVlIHNlIHVzYW4gcGFyYSBwYXJhIGhhbGxhciBsb3MgbWF0Y2hpbmdzIHkgcXVlIHBvciB0YW50byBub3MgcGVybWl0ZW4gZnVzaW9uYXIgbG9zIDIgZGYncyBzZSBsbGFtYW4gImtleXMiLiAKCkxhIG9wY2nDs24gZGUgbGFzIGZ1bmNpb25lcyBwYXJhIHNlbGVjY2lvbmFyIGVzdGFzIGNvbHVtbmFzICJrZXlzIiBlcyBgYnkgPWAuIAoKICAtIHNpIHBvbmVtb3MgYGxlZnRfam9pbihkZjEsIGRmMiwgYnkgPSAiWDEiKWAgc2UgaGFyw6EgdW5hIGxlZnRfam9pbiBzaWVuZG8gbGEgdmFyaWFibGUgIlgxIiBsYSBxdWUgaGFyw6EgZGUga2V5LiBTaSBsYXMgdmFyaWFibGVzIGtleSBubyBzZSBsbGFtYXNlbiBpZ3VhbCBlbiBsb3MgMiBkZidzIHNpZW1wcmUgcG9kZW1vcyByZW5vbWJyYXJsYXMgbyBoYWNlciBsbyBzaWd1aWVudGU6IGBsZWZ0X2pvaW4oZGYxLCBkZjIsIGJ5ID0gYygiWDEiID0gIkQ0IilgCiAgCiAgLSAgVGFtYmnDqW4gc2UgcHVlZGVuIGZ1c2lvbmFyIHRhYmxhcyB1c2FuZG8gZG9zIGtleXM7IHBvciBlamVtcGxvOiAgYGxlZnRfam9pbihkZjEsIGRmMiwgYnkgPSBjKCJYMSIsICJYMiIpKWAgLiBTaSBubyBzZSBsbGFtYXNlbiBpZ3VhbCBsYXMgdmFyaWFibGVzIGVuIGRmMSB5IGRmMiBoYXLDrWFtb3MgYGxlZnRfam9pbihkZjEsIGRmMiwgYnkgPSBjKCJYMSIgPSAiRDQiLCAiWDIiID0gIkQ3IikpYAoKPGJyPgo8YnI+CgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyBGaWx0ZXJpbmcgam9pbnMKCgpGaWx0ZXJpbmcgam9pbnMgc29uIHNpbWlsYXJlcyBhIGxvcyBhbnRlcmlvcmVzIChNdXRhdGluZyBqb2lucyk7IG8gc2VhLCBoYWNlbiBtYWNodGluZyBjb24gbGFzIGZpbGFzIGRlIGxhIG1pc21hIG1hbmVyYSwgKipQRVJPIGFmZWN0YW4gYSBsYXMgZmlsYXMqKiwgTk8gYSBsYXMgY29sdW1uYXMuIEhheSBmaWx0ZXJpbmcgIGpvaW5zIGRlIDIgdGlwb3M6CgogIC0gYHNlbWlfam9pbihkZjEsZGYyKWA6IHJldG9ybmEgbGFzIG9ic2VydmFjaW9uZXMgZGUgZGYxIHF1ZSB0aWVuZW4gdW4gbWF0Y2ggZW4gZGYyLiBFbiBjdWFudG8gYSBsYXMgY29sdW1uYXMgc8OzbG8gcmV0b3JuYSBsYXMgY29sdW1uYXMgZGUgZGYxCiAgLSBgYW50aV9qb2luKGRmMSxkZjIpYDogcmV0b3JuYSBsYXMgb2JzZXJ2YWNpb25lcyBkZSBkZjEgcXVlIE5PIHRpZW5lbiB1biBtYXRjaCBlbiBkZjI7IG9zZWEsIHF1aXRhIGxhcyBvYnNlcnZhY2lvbmVzIGNvbiBtYXRjaC4gRW4gY3VhbnRvIGEgbGFzIGNvbHVtbmFzIHPDs2xvIHJldG9ybmEgbGFzIGNvbHVtbmFzIGRlIGRmMQoKCkxhIHNlbWlfam9pbiBzZSBkaWZlcmVuY2lhIGRlIGxhIGlubmVyX2pvaW4gZW4gcXVlIGxhIGlubmVyX2pvaW4gc29sbyByZXRvcm5hIHVuYSBmaWxhIGRlIGRmMSBwb3IgY2FkYSBtYXRjaGluZywgbWllbnRyYXMgcXVlICoqbGEgc2VtaV9qb2luIE5VTkNBIGR1cGxpY2EgZmlsYXMgZGUgZGYxKioKCgpMYXMgZmlsdGVyaW5nIGpvaW5zIHNvbiDDunRpbGVzIHBhcmEgZGlhZ25vc3RpY2FyIG1pc21hdGNoZXMuClNpIHF1aWVyZXMgc2FiZXIgc29icmUgbG9zIG1hdGNoZXMsIGhheiB1bmEgc2VtaV9qb2luKCkgb3IgYW50aV9qb2luKCkuIHNlbWlfam9pbigpIGFuZCBhbnRpX2pvaW4oKSBOVU5DQSBkdXBsaWNhbiBmaWxhczsgc29sbyBwdWVkZW4gcXVpdGFyIGZpbGFzLgogICAgCiAgIAoKPGJyPgoKIyMjIyMgc2VtaV9qb2luCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQpkZl9zZW1pX2pvaW4gPC0gIHNlbWlfam9pbih4LCB5LCBieSA9ICJpZCIpCmBgYAoKCmBgYHtyICwgZWNobz1GQUxTRSwgZXZhbCA9IFRSVUUsIGZpZy5hc3AgPSA0LzIsIG91dC53aWR0aCA9ICI3NSUiLCBmaWcuYWxpZ24gPSAiY2VudGVyIn0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzExX3NlbWktam9pbi5wbmciKSkKYGBgCgoKPGJyPgoKIyMjIyMgYW50aV9qb2luCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQpkZl9hbnRpX2pvaW4gPC0gIGFudGlfam9pbih4LCB5LCBieSA9ICJpZCIpCmBgYAoKCmBgYHtyICwgZWNobz1GQUxTRSwgZXZhbCA9IFRSVUUsIGZpZy5hc3AgPSA0LzIsIG91dC53aWR0aCA9ICI3NSUiLCBmaWcuYWxpZ24gPSAiY2VudGVyIn0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzEyX2FudGktam9pbi5wbmciKSkKYGBgCgoKPGJyPgoKCiMjIyMjIGNvbXBhcmVtb3MgbGEgc2VtaV9qb2luIGNvbiBsYSBpbm5lcl9qb2luCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQpkZl9pbm5lciA8LSAgaW5uZXJfam9pbih4LCB5LCBieSA9ICJpZCIpCmRmX3NlbWlfam9pbiA8LSAgc2VtaV9qb2luKHgsIHksIGJ5ID0gImlkIikKYGBgCgoKYGBge3IgLCBlY2hvPUZBTFNFLCBldmFsID0gVFJVRSwgZmlnLmFzcCA9IDQvMiwgb3V0LndpZHRoID0gIjc1JSIsIGZpZy5hbGlnbiA9ICJjZW50ZXIifQprbml0cjo6aW5jbHVkZV9ncmFwaGljcyggIGMoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzA2X2lubmVyLWpvaW4ucG5nIiksIGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18xMV9zZW1pLWpvaW4ucG5nIikpICApCmBgYAoKCkxhIGlubmVyX2pvaW4gCgogIC0gKGkpIGHDsWFkZSB2YXJpYWJsZXMgZGVsIGRhdGEuZnJhbWUgeSBhbCBkYXRhLmZyYW1lIHggKGVuIGVzdGUgY2FzbyB5JHkpCiAgLSAoaWkpIHPDs2xvIHJldGllbmUgbGFzIHJvd3MgZGVsIGRhdGEuZnJhbWUgeCAqKnF1ZSB0aWVuZW4gdW4gbWF0Y2ggZW4geSoqIChlbiBlc3RlIGNhc28gbGFzIDIgcHJpbWVyYXMgZmlsYXMgZGUgeCkKICAKICAtIChpaWkpIFBFUk8gKiplbiBlc3RlIGVqZW1wbG8gQURFTcOBUyoqIGR1cGxpY2Egcm93cy4gbGEgcmF6w7NuIGVzIHF1ZSBoYXkgbWF0Y2hpbmcgZHVwbGljYWRvcywgaGF5IDIgdW5vcyBlbiB4IHkgb3Ryb3MgMiB1bm9zIGVuIGVsIGRhdGEuZnJhbWUgeSAoY29uIGRpc3RpbnRvcyB2YWxvcmVzIGRlIHkkeSkgKGFzw60gcXVlIHNhbGVuIDQgcm93cykKCgoKPGJyPgoKIyMjIyBJbXBvcnRhbnRlIMK/Q8OzbW8gZGVjaXIgYSBsYXMgZnVuY2lvbmVzIGxhIGNvbHVtbmEsIG8gY29sdW1uYXMsIHF1ZSBzZSB1c2Fyw6FuIHBhcmEgaGFjZXIgbG9zIG1hdGNoaW5nPwoKCkFsIGlndWFsIHF1ZSBjb24gbGFzIG11dGF0aW5nIGpvaW5zLCBlbiBsYXMgZmlsdGVyaW5nIGpvaW5zIHRhbWJpw6luIHBvZGVtb3MoREVCRU1PUykgZWxlZ2lyIGxhcyBjb2x1bW5hcyAobyB2YXJpYWJsZXMpIHF1ZSBub3Mgc2Vydmlyw6FuIHBhcmEgdW5pciBsb3MgMiBkZidzLiBFc3RhcyBjb2x1bW5hcyBxdWUgc2UgdXNhbiBwYXJhIHBhcmEgaGFsbGFyIGxvcyBtYXRjaGluZ3MgeSBxdWUgcG9yIHRhbnRvIHF1ZSBwZXJtaXRlbiBmdXNpb25hciBsb3MgMiBkZidzIHNlIGxsYW1hbiAia2V5cyIuIAoKTGEgb3BjacOzbiBkZSBsYXMgZnVuY2lvbmVzIHBhcmEgc2VsZWNjaW9uYXIgZXN0YXMgY29sdW1uYXMgImtleXMiIGVzIGBieSA9YC4gCgogIC0gc2kgcG9uZW1vcyBgc2VtaV9qb2luKGRmMSwgZGYyLCBieSA9ICJYMSIpYCBzZSBoYXLDoSB1bmEgc2VtaV9qb2luIHNpZW5kbyBsYSB2YXJpYWJsZSAiWDEiIGxhIHF1ZSBoYXLDoSBkZSBrZXkuIFNpIGxhcyB2YXJpYWJsZXMga2V5IG5vIHNlIGxsYW1hc2VuIGlndWFsIGVuIGxvcyAyIGRmJ3Mgc2llbXByZSBwb2RlbW9zIHJlbm9tYnJhcmxhcyBvIGhhY2VyIGBzZW1pX2pvaW4oZGYxLCBkZjIsIGJ5ID0gYygiWDEiID0gIkQ0IilgCiAgCiAgLSAgc2kgcG9uZW1vcyBgc2VtaV9qb2luKGRmMSwgZGYyLGJ5ID0gYygiWDEiLCAiWDIiKWAgaGFyw6EgZmFsdGEgcXVlIHVuYSByb3cgZGUgZGYxIHRlbmdhIGxvcyB2YWxvcmVzIHRhbnRvIGRlIFgxIGNvbW8gZGUgWDIgaWd1YWxlcyAgYSBsb3MgZGUgZXNhcyBtaXNtYXMgdmFyaWFibGVzIGVuIGRmMi4gU2kgbm8gc2UgbGxhbWFzZW4gaWd1YWwgbGFzIHZhcmlhYmxlcyBlbiBkZjEgeSBkZjIgaGFyw61hbW9zIGBieSA9IGMoIlgxIiA9ICJENCIsICJYMiIgPSAiRDciKWAKCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgo8YnI+CgojIyBTZXQgb3BlcmF0aW9ucwoKCkVzdGUgdGlwbyBkZSBqb2lucyBlcyBtw6FzIGVzdHJpY3RhOiAqKmhhY2UgZmFsdGEgcXVlIGxvcyAyIGRmJ3MgdGVuZ2FuIGxhcyBtaXNtYXMgdmFyaWFibGVzIChvIGNvbHVtbmFzKSoqLiBMb3MgMiBkZidzIHB1ZWRlbiB0ZW5lciBvYnNlcnZhY2lvbmVzKGZpbGFzKSBkaWZlcmVudGVzLCBQRVJPIGVzIG5lY2VzYXJpbyBxdWUgdGVuZ2FuIGxhcyBtaXNtYXMgdmFyaWFibGVzIChvIGNvbHVtbmFzKS4KCkNvbW8gbG9zIDIgZGYncyB0aWVuZW4gbGFzIG1pc21hcyBjb2x1bW5hcywgZW50b25jZXMgZXMgY29tbyBzaSBzZSB0cmF0YXNlbiBsb3MgZGZzIGNvbW8gY29uanVudG9zOgoKICAtIGBpbnRlcnNlY3QoZGYxLCBkZjIpYDogZGV2dWVsdmUgdW4gZGYgY29uIGxhcyBvYnNlcnZhY2lvbmVzIGNvbXVuZXMgZW4gZGYxIHkgZGYyCiAgLSBgdW5pb24oZGYxLCBkZjIpYDogZGV2dWVsdmUgbGEgdW5pw7NuOyBvIHNlYSwgbGFzIG9ic2VydmFjaW9uZXMgZGUgZGYxIHkgZGUgZGYyIChxdWl0YW5kbyBsYXMgcG9zaWJsZXMgZmlsYXMgZHVwbGljYWRhcykKICAtIGB1bmlvbl9hbGwoZGYxLCBkZjIpYDogZGV2dWVsdmUgbGEgdW5pw7NuIChzaW4gcXVpdGFyIGxvcyBkdXBsaWNhZG9zKQogIC0gYHNldGRpZmYoZGYxLCBkZjIpYDogZGV2dWVsdmUgbGFzIGZpbGFzIGVuIGRmMSBxdWUgbm8gZXN0w6FuIGVuIGRmMgogIAogIDxicj4KICAKICAtIGBzZXRlcXVhbChkZjEsZGYyYDogcmV0b3JuYSBUUlVFIHNpIGRmIHkgZGYyIHRpZW5lbiBleGFjdGFtZW50ZSBsYXMgbWlzbWFzIGZpbGFzIChkYSBpZ3VhbCBlbCBvcmRlbiBlbiBlbCBxdWUgZXN0w6luIGxhcyBmaWxhcykKCjxicj4KCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CnggPC0gdGliYmxlOjp0aWJibGUodjEgPSBjKDEsIDEsIDIpLCB2MiA9IGMoImEiICwgImIiLCAiYSIpKQp5IDwtIHRpYmJsZTo6dGliYmxlKHYxID0gYygxLCAyKSwgICAgdjIgPSBjKCJhIiAsICJiIikpCmBgYAoKPGJyPgoKIyMjIyMgaW50ZXJzZWNjacOzbgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmludGVyc2VjdCh4LCB5KQpgYGAKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiNzUlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18xM19pbnRlcnNlY3QucG5nIikpCmBgYAoKCjxicj4KCiMjIyMjIHVuacOzbgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CnVuaW9uKHgsIHkpCmBgYAoKCmBgYHtyICwgZWNobz1GQUxTRSwgZXZhbCA9IFRSVUUsIGZpZy5hc3AgPSA0LzIsIG91dC53aWR0aCA9ICI3NSUiLCBmaWcuYWxpZ24gPSAiY2VudGVyIn0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoaGVyZTo6aGVyZSgiaW1hZ2VuZXMiLCAidHRfMDVfaW1nXzE0X3VuaW9uLnBuZyIpKQpgYGAKCgo8YnI+CgojIyMjIyBzZXRkaWZmCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0Kc2V0ZGlmZih4LCB5KQpgYGAKCgpgYGB7ciAsIGVjaG89RkFMU0UsIGV2YWwgPSBUUlVFLCBmaWcuYXNwID0gNC8yLCBvdXQud2lkdGggPSAiNzUlIiwgZmlnLmFsaWduID0gImNlbnRlciJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKGhlcmU6OmhlcmUoImltYWdlbmVzIiwgInR0XzA1X2ltZ18xNV9zZXRkaWZmLnBuZyIpKQpgYGAKClB1ZWRlcyBwcm9iYXIgdMO6IG1pc21vIGEgY2FtYmlhciBlbCBvcmRlbiBkZWwgbG9zIGRmJ3MgZW4gYHNldGRpZmYoKWA6CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0Kc2V0ZGlmZih5LCB4KQpgYGAKCgoKCjxicj4KCiMjIyMjIHNldHF1YWwKClNpcnZlIHBhcmEgZGV0ZXJtaW5hciBzaSAyIGRmJ3Mgc29uIGlndWFsZXMgKHNpbiBpbXBvcnRhciBlbCBvcmRlbiBlbiBxdWUgZXN0w6luIGxhcyBmaWxhcykKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CnNldGVxdWFsKHgsIHkpCmBgYAoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IFRSVUV9CnNldGVxdWFsKHVuaW9uKHgsIHkpLCB1bmlvbih5LCB4KSkKYGBgCgo8YnI+CgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKPGJyPgoKIyBFc3BlcmFuZG8gYSBHT0RPVC9nZ3Bsb3QyCgo8YnI+CgpCdWVubywgcHVlcyBoZW1vcyB2aXN0byAiVE9ETyIgc29icmUgbWFuaXB1bGFjacOzbiBkZSBkYXRvcy4gRWwgcHLDs3hpbW8gdHV0b3JpYWwgdmEgZGUgdmlzdWFsaXphY2nDs246IGhhY2lhIGdncGxvdDIKCkFxdcOtIHVuIHBlcXVlw7FvIGF2YW5jZToKCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQpsaWJyYXJ5KCJnZ3Bsb3QyIikKbXlfcGxvdCA8LSBnZ3Bsb3QoZ2FwbWluZGVyLCBhZXMoeCA9IGNvbnRpbmVudCwgeSA9IGxpZmVFeHApKSArCiAgZ2VvbV9ib3hwbG90KG91dGxpZXIuY29sb3VyID0gImhvdHBpbmsiKSArCiAgZ2VvbV9qaXR0ZXIocG9zaXRpb24gPSBwb3NpdGlvbl9qaXR0ZXIod2lkdGggPSAwLjEsIGhlaWdodCA9IDApLCBhbHBoYSA9IDEvNCkgKwogIGxhYnModGl0bGUgPSAiRXhwZXJhbnphIGRlIHZpZGEgKHBvciBjb250aW5lbnRlKSIsCiAgICAgICBzdWJ0aXRsZSA9ICJEYXRvcyBkZSBnYXBtaW5kZXIuIDE5NTItMjAwNyhvYnNlcnZhY2lvbmVzIGNhZGEgNSBhw7FvcykiLAogICAgICAgY2FwdGlvbiA9ICJTb3VyY2U6IEdhcG1pbmRlci4gSmVubnkgQnJ5YW4gcm9ja3MgaW4gZ2FwbWluZGVyIHZpZ25ldHRlISEiLCAKICAgICAgIHggPSAiQ29udGluZW50ZSIsIHkgPSAiRXNwZXJhbnphIGRlIFZpZGEgKGxpZmVFeHApIikgCmBgYAoKPGJyPgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgZXZhbCA9IFRSVUV9Cm15X3Bsb3QKYGBgCgo8YnI+Cjxicj4KCgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQpnYXBtaW5kZXIyIDwtIGdhcG1pbmRlciAlPiUgbXV0YXRlKHllYXIgPSBhcy5mYWN0b3IoeWVhcikpCgpsaWJyYXJ5KCJnZ3Bsb3QyIikKbXlfcGxvdCA8LSBnZ3Bsb3QoZ2FwbWluZGVyMiwgYWVzKHggPSB5ZWFyLCB5ID0gbGlmZUV4cCkpICsKICBnZW9tX2JveHBsb3Qob3V0bGllci5jb2xvdXIgPSAiaG90cGluayIpICsKICBnZW9tX2ppdHRlcihwb3NpdGlvbiA9IHBvc2l0aW9uX2ppdHRlcih3aWR0aCA9IDAuMSwgaGVpZ2h0ID0gMCksIGFscGhhID0gMS80KSArCiAgbGFicyh0aXRsZSA9ICJFeHBlcmFuemEgZGUgdmlkYSAocG9yIGHDsW8pIiwKICAgICAgIHN1YnRpdGxlID0gIkRhdG9zIGRlIGdhcG1pbmRlci4gMTk1Mi0yMDA3KG9ic2VydmFjaW9uZXMgY2FkYSA1IGHDsW9zKSIsCiAgICAgICBjYXB0aW9uID0gIlNvdXJjZTogR2FwbWluZGVyLiBKZW5ueSBCcnlhbiByb2NrcyBpbiBnYXBtaW5kZXIgdmlnbmV0dGUhISIsIAogICAgICAgeCA9ICJQZXJpb2RvIiwgeSA9ICJFc3BlcmFuemEgZGUgVmlkYSAobGlmZUV4cCkiKSAKYGBgCgo8YnI+CgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgZXZhbCA9IFRSVUV9Cm15X3Bsb3QKYGBgCgo8YnI+CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBUUlVFfQpsaWJyYXJ5KCJnZ3Bsb3QyIikKbXlfcGxvdCA8LSBnZ3Bsb3QoZ2FwbWluZGVyLCBhZXMoeCA9IGdkcFBlcmNhcCwgeSA9IGxpZmVFeHAsIGNvbG91ciA9IGNvbnRpbmVudCkpICsKICBnZW9tX2ppdHRlcihwb3NpdGlvbiA9IHBvc2l0aW9uX2ppdHRlcih3aWR0aCA9IDAuMSwgaGVpZ2h0ID0gMCksIGFscGhhID0gMS80KSArCiAgbGFicyh0aXRsZSA9ICJFeHBlcmFuemEgZGUgdmlkYSB2cy4gR0RQIChwZXIgY8OhcGl0YSkiLAogICAgICAgc3VidGl0bGUgPSAiRGF0b3MgZGUgZ2FwbWluZGVyLiAxOTUyLTIwMDcob2JzZXJ2YWNpb25lcyBjYWRhIDUgYcOxb3MpIiwKICAgICAgIGNhcHRpb24gPSAiU291cmNlOiBHYXBtaW5kZXIuIEplbm55IEJyeWFuIHJvY2tzIGluIGdhcG1pbmRlciB2aWduZXR0ZSEhIiwgCiAgICAgICB4ID0gIkdEUCAocGVyIGPDoXBpdGEpIiwgeSA9ICJFc3BlcmFuemEgZGUgVmlkYSAobGlmZUV4cCkiKSAKYGBgCgo8YnI+CgpgYGB7ciwgZWNobyA9IEZBTFNFLCBldmFsID0gVFJVRX0KbXlfcGxvdApgYGAKCjxicj4KCi0tLS0tLS0tLS0tLS0tLQoKIyBUaWR5bG9nIHBhY2thZ2UKClVuYSBoZXJyYW1pZW50YSBxdWUgb3MgcHVlZGUgc2VyIGRlIHV0aWxpZGFkIHBhcmEgYXByZW5kZXIgZWwgdXNvIGRlIGBkcGx5cmAgZXMgZWwgcGFxdWV0ZSBbYHRpZHlsb2dgXShodHRwczovL2dpdGh1Yi5jb20vZWxiZXJzYi90aWR5bG9nKS4gRXN0ZSBwYXF1ZXRlIG5vcyBkYSBmZWVkYmFjayBpbnN0YW50w6FuZW8gc29icmUgcXXDqSBoYWNlbW9zIGN1YW5kbyB1c2Ftb3MgbGFzIHByaW5jaXBhbGVzIGZ1bmNpb25lcyBkZSBgZHBseXJgIHkgYHRpZHlyYC4gVmXDoW1vc2xvIGVuIGFjY2nDs246CgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gVFJVRSwgbWVzc2FnZSA9IFRSVUV9CmxpYnJhcnkoImRwbHlyIikKbGlicmFyeSgidGlkeXIiKQpsaWJyYXJ5KCJ0aWR5bG9nIiwgd2Fybi5jb25mbGljdHMgPSBGQUxTRSkKZmlsdGVyZWQgPC0gZmlsdGVyKG10Y2FycywgY3lsID09IDQpCm11dGF0ZWQgPC0gbXV0YXRlKG10Y2FycywgbmV3X3ZhciA9IHd0ICoqIDIpCmBgYAoKU2kgb3MgZmlqw6FpcywgY2FkYSB2ZXogcXVlIGVqZWN1dGFzIHVuYSBmdW5jacOzbiBkZSBgZHBseXJgIG5vcyBkZXZ1ZWx2ZSB1biBtZW5zYWplIGV4cGxpY8OhbmRvbm9zIHF1ZSBzZSBoYSBoZWNoby4gUG9yIGVqZW1wbG8gbGEgbGluZWEgZGUgY8OzZGlnbyBgZmlsdGVyZWQgPC0gZmlsdGVyKG10Y2FycywgY3lsID09IDQpYCBoYSBjcmVhZG8gdW4gbnVldm8gZGF0YS5mcmFtZSBkb25kZSBzZSBoYW4gZWxpbWluYWRvIDIxIGZpbGFzIGRlbCBkZiBvcmlnaW5hbDogYCM+IGZpbHRlcjogcmVtb3ZlZCAyMSByb3dzICg2NiUpLCAxMSByb3dzIHJlbWFpbmluZ2AuCgpMYSBzZWd1bmRhIGxpbmVhIGRlIGPDs2RpZ28gYG11dGF0ZWQgPC0gbXV0YXRlKG10Y2FycywgbmV3X3ZhciA9IHd0ICoqIDIpYCBoYSBjcmVhZG8gdW5hIG51ZXZhIHZhcmlhYmxlIGNvbiBgbXV0YXRlKClgOiBgIz4gbXV0YXRlOiBuZXcgdmFyaWFibGUgJ25ld192YXInIHdpdGggMjkgdW5pcXVlIHZhbHVlcyBhbmQgMCUgTkFgLgoKCgo8YnI+CgotLS0tLS0tLS0tLS0tLS0KCjxicj4KCiMgVGlkeXZlcnNlIHZzLiBCYXNlIFIKCgpUb2RvIGxvIHF1ZSBzZSBwdWVkZSBoYWNlciBjb24gZHBseXIsIHRpZHlyIGV0Yy4gLHRhbWJpw6luIHNlIHB1ZWRlIGhhY2VyIGNvbiBCYXNlLVIgcGVybyBkZSB1bmEgbWFuZXJhIG11Y2hvIG1lbm9zIGludHVpdGl2YS4gCgoKRWwgc2lndWllbnRlIGVqZW1wbG8gZXN0YSBzYWNhZG8gZGUgW2VzdGUgcG9zdF0oaHR0cHM6Ly93d3cuci1ibG9nZ2Vycy5jb20vd2hlbi1pLXVzZS1wbHlyZHBseXIvKS4gU29uIGRvcyB0cm96b3MgZGUgY8OzZGlnbyBxdWUgaGFjZW4gZXhhY3RhbWVudGUgbG8gbWlzbW86CgpDb24gZWwgdGlkeXZlcnNlOgoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmxpYnJhcnkoZHBseXIpCm10Y2FycyAlPiUgCiAgZ3JvdXBfYnkoY3lsLCBhbSkgJT4lCiAgc2VsZWN0KG1wZywgY3lsLCB3dCwgYW0pICU+JQogIHN1bW1hcmlzZShhdmdtcGcgPSBtZWFuKG1wZyksIGF2Z3d0ID0gbWVhbih3dCkpICU+JQogIGZpbHRlcihhdmdtcGcgPiAyMCkKYGBgCgpDb24gbGEgc2ludGF4aXMgZGUgYmFzZSBSOgoKCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQpmaWx0ZXIoCiAgc3VtbWFyaXNlKAogICAgc2VsZWN0KAogICAgICBncm91cF9ieShtdGNhcnMsIGN5bCwgYW0pLAogICAgICBtcGcsIGN5bCwgd3QsIGFtKSwKICAgIGF2Z21wZyA9IG1lYW4obXBnKSwgYXZnd3QgPSBtZWFuKHd0KSksCiAgYXZnbXBnID4gMjApCmBgYAoKTyBwdWVzdG8gZW4gaG9yaXpvbnRhbAoKYGBge3IsIGVjaG8gPSBUUlVFLCBldmFsID0gRkFMU0V9CmZpbHRlcihzdW1tYXJpc2Uoc2VsZWN0KGdyb3VwX2J5KG10Y2FycywgY3lsLCBhbSksICBtcGcsIGN5bCwgd3QsIGFtKSxhdmdtcGcgPSBtZWFuKG1wZyksIGF2Z3d0ID0gbWVhbih3dCkpLCBhdmdtcGcgPiAyMCkKYGBgCgo8YnI+CgoKIyMjIyMgT3Ryb3MgMiBlamVtcGxvcyBkZSBjb21wYXJhY2nDs24gdGlkeXZlcnNlIHZlcnN1cyBCYXNlLVI6CgpgYGB7ciwgZWNobyA9IFRSVUUsIGV2YWwgPSBGQUxTRX0KZGYgJT4lIGZpbHRlcihjb3VudHJ5ID09ICJTcGFpbiIpICU+JSAgc2VsZWN0KHllYXIsIGxpZmVFeHApCgpkZltkZiRjb3VudHJ5ID09ICJTcGFpbiIsIGMoInllYXIiLCAibGlmZUV4cCIpXSAKYGBgCgpFc3RlIMO6bHRpbW8gZWplbXBsbyBsbyBpbnRyb2R1emNvIHBvcnF1ZSBxdWllcm8gcmVjb3JkYXIgW2VzdGFzIHRyYXNwYXJlbmNpYXNdKGh0dHBzOi8vY2VyZWJyYWxtYXN0aWNhdGlvbi5naXRodWIuaW8vZG93bl93aXRoX29wcF9kcGx5ci5odG1sIy8pIHF1ZSBleHBsaWNhbiBxdWUgZWwgcGFxdWV0ZSBgZGJwbHlyYCB0cmFkdWNlIGV4cHJlc2lvbmVzIGRlIGRwbHlyIGEgU1FMLiBFbCBjw7NkaWdvIGRlIGxhcyB0cmFuc3BhcmVuY2lhcyBlc3TDoSBbYXF1w61dKGh0dHBzOi8vZ2l0aHViLmNvbS9DZXJlYnJhbE1hc3RpY2F0aW9uL1ByZXNlbnRhdGlvbnMpLCB5IFthcXXDrV0oaHR0cHM6Ly9kYi5yc3R1ZGlvLmNvbS9kcGx5ci8pIHB1ZWRlcyBhcHJlbmRlciBjb21vIGhhY2VyIHF1ZXJpZXMgU1FMIGEgdW4gZGF0YWJhc2UgdXRpbGl6YW5kbyBsYSBzaW50YXhpcyBkZSBkcGx5ci4KCmBgYHtyLCBlY2hvID0gVFJVRSwgZXZhbCA9IEZBTFNFfQojLSBjb24gdGlkeXZlcnNlCmRmX2NhcnMgJT4lIAogIHNlbGVjdChsb25nbmFtZSwgY3lsLCBocCkgJT4lCiAgbXV0YXRlKCBzaG9ydG5hbWUgPSB3b3JkKGxvbmduYW1lLCAxKSApICU+JQogIHNlbGVjdCggLSBsb25nbmFtZSkgIC0+CmRmX2NhcnNfbGltaXRlZApoZWFkKCBkZl9jYXJzX2xpbWl0ZWQgKQoKIy0gY29uIGRwbHlyIFBFUk8gc2luICU+JSAKaGVhZCgKICBzZWxlY3QoCiAgICBtdXRhdGUoIAogICAgICBzZWxlY3QoZGZfY2FycywgbG9uZ25hbWUsIGN5bCwgaHApICwgCiAgICAgIHNob3J0bmFtZSA9IHdvcmQobG9uZ25hbWUsIDEpIAogICAgICApLCAKICAgIC1sb25nbmFtZSAKICAgICkKKQpgYGAKCgo8YnI+Cjxicj4KCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKPGJyPgoKIyBCaWJsaW9ncmFmw61hCgotIFtJbnRyb2R1Y2Npw7NuIGEgZHBseXJdKGh0dHA6Ly9zdGF0NTQ1LmNvbS9ibG9jazAwOV9kcGx5ci1pbnRyby5odG1sKS4gSmVubnkgQnJ5YW4gbm9zIGN1ZW50YSAoZW4gc3VzIEZBTlRBU1RJQ09TIG1hdGVyaWFsZXMgcGFyYSBlbCBjdXJzbyBTVEFUNTQ1KSBsb3MgcnVkaW1lbnRvcyBkZSBkcGx5ci4gQnVlbmEgcGFydGUgZGUgZXN0ZSB0dXRvcmlhbCBzZSBiYXNhIGVuIGVsIHN1eW8uCgotIFtkcGx5ciBmdW5jdGlvbnMgZm9yIGEgc2luZ2xlIGRhdGFzZXRdKGh0dHA6Ly9zdGF0NTQ1LmNvbS9ibG9jazAxMF9kcGx5ci1lbmQtc2luZ2xlLXRhYmxlLmh0bWwpLiBPdHJvIG1hdGVyaWFsIGRlIEplbm55IGRlbCBjdXJzbyBTVEFUNTQ1IHNvYnJlIGRwbHlyCgotIFtkcGx5ciBDSEVBVCBTSEVFVF0oaHR0cHM6Ly93d3cucnN0dWRpby5jb20vcmVzb3VyY2VzL2NoZWF0c2hlZXRzLykuIEZBTlRBU1RJUVXDiVJSSU1BISEhIEltcHJlc2lvbmFudGUhIQoKLSBbQ8OhcGl0dWxvIGRlIFI0RFMgc29icmUgImRwbHlyIl0oaHR0cDovL3I0ZHMuaGFkLmNvLm56L3RyYW5zZm9ybS5odG1sKS4gRGUgSGFkbGV5LiBGYW50w6FzdGljbyBsaWJybyB5IGZhbnTDoXN0aWNvIGNhcMOtdHVsbyBzb2JyZSBtYW5lam8gZGUgZGF0b3MgKGRwbHlyKS4KCi0gW0EgbmV3IGRhdGEgcHJvY2Vzc2luZyB3b3JrZmxvdyBmb3IgUjogZHBseXIsIG1hZ3JpdHRyLCB0aWR5ciwgZ2dwbG90Ml0oaHR0cDovL3pldnJvc3MuY29tL2Jsb2cvMjAxNS8wMS8xMy9hLW5ldy1kYXRhLXByb2Nlc3Npbmctd29ya2Zsb3ctZm9yLXItZHBseXItbWFncml0dHItdGlkeXItZ2dwbG90Mi8pLiBQb3N0IGRlIHVuIG51ZXZvIGNvbnZlbmNpZG8gZGUgbGFzIGJvbmRhZGVzIGRlbCB0aWR5dmVyc2UuIFVuIGVqZW1wbG8gc2VuY2lsbG8gcGVybyBpbHVzdHJhdGl2by4KCi0gW1dpZGUgJiBMb25nIERhdGFdKGh0dHBzOi8vc3RhbmZvcmQuZWR1L35lamRlbXlyL3ItdHV0b3JpYWxzL3dpZGUtYW5kLWxvbmcvKSBQb3N0IHF1ZSBleHBsaWNhIGxvcyBiZW5lZmljaW9zIGRlIHRyYWJhamFyIGNvbiBkYXRvcyBlbiBmb3JtYXRvIGxvbmcgKHRpZHkpLgoKLSBbTGEgYmlibGlhIGRlbCB0aWR5IGRhdGFdKGZ0cDovL2NyYW4uci1wcm9qZWN0Lm9yZy9wdWIvUi93ZWIvcGFja2FnZXMvdGlkeXIvdmlnbmV0dGVzL3RpZHktZGF0YS5odG1sKS4gVmlnbmV0dGUgZGUgdGlkeXIgcGFja2FnZSBlc2NyaXRhIHBvciBIYWRsZXkgV2lja2hhbSBxdWUgZXhwbGljYSBjb24gTVVDSE8gZGV0YWxsZSBxdWUgc29uIGxvcyB0aWR5IGRhdGEuCgotIFtMZXNzZXIga25vd24gZHBseXIgZnVuY3Rpb25zXShodHRwczovL3N0YXRpc3RpY2Fsb2Rkc2FuZGVuZHMud29yZHByZXNzLmNvbS8yMDE5LzA4LzMwL2xlc3Nlci1rbm93bi1kcGx5ci1mdW5jdGlvbnMvKS4gVW4gcG9zdCBmYW50w6FzdGljbyBzb2JyZSBhbGd1bmFzIGZ1bmNpb25lcyBubyB0YW4gY29ub2NpZGFzIGRlIGBkcGx5cmAuCgotIFtWacOxZXRhcyBvZmljaWFsZXMgZGUgZHBseXIgMS4wLjBdKGh0dHBzOi8vZHBseXIudGlkeXZlcnNlLm9yZy9pbmRleC5odG1sKS4gRXN0w6FuIGVuIGxhIHNlY2Npw7NuICJBcnTDrWN1bG9zIi4gRXhwbGljYW4gbXV5IGJpZW4gZWwgZnVuY2lvbmFtaWVudG8gZGUgbGFzIG51ZXZhcyBmdW5jaW9uZXMgZGUgZHBseXIgMS4wLjAKCi0gRHVyYW50ZSBlbCBwcm9jZXNvIGRlIGFudW5jaW8gb2ZpY2lhbCBkZSBkcGx5ciAxLjAuMCBzZSBwdWJsaWNhcm9uIHVuYSBzZXJpZSBkZSBwb3N0cyBvZmljaWFsZXMgZGVsIHRpZHl2ZXJzZSwgY29uY3JldGFtZW50ZSBbYXF1w61dKGh0dHBzOi8vd3d3LnRpZHl2ZXJzZS5vcmcvYmxvZy8yMDIwLzA2L2RwbHlyLTEtMC0wLyksIFthcXXDrV0oaHR0cHM6Ly93d3cudGlkeXZlcnNlLm9yZy9ibG9nLzIwMjAvMDQvZHBseXItMS0wLTAtYW5kLXZjdHJzLyksIFthcXXDrV0oaHR0cHM6Ly93d3cudGlkeXZlcnNlLm9yZy9ibG9nLzIwMjAvMDMvZHBseXItMS0wLTAtc2VsZWN0LXJlbmFtZS1yZWxvY2F0ZS8pLCBbYXF1w61dKGh0dHBzOi8vd3d3LnRpZHl2ZXJzZS5vcmcvYmxvZy8yMDIwLzA1L2RwbHlyLTEtMC0wLWxhc3QtbWludXRlLWFkZGl0aW9ucy8pIHkgW2FxdcOtXShodHRwczovL3d3dy50aWR5dmVyc2Uub3JnL2Jsb2cvMjAyMC8wMy9kcGx5ci0xLTAtMC1zdW1tYXJpc2UvKS5Fc3TDoW4gbXV5IGJpZW4uCgotIEtlaXRoIE1jTnVsdHkgdGFtYmnDqW4gbWUgYXl1ZG8gYSBlbnRlbmRlciBkcGx5ciAxLjAwOiBbYXF1w61dKGh0dHBzOi8vdG93YXJkc2RhdGFzY2llbmNlLmNvbS93aGF0LXlvdS1uZWVkLXRvLWtub3ctYWJvdXQtdGhlLW5ldy1kcGx5ci0xLTAtMC03ZWFhYWY2ZDc4YWMpIHkgW2FxdcOtXShodHRwczovL3Rvd2FyZHNkYXRhc2NpZW5jZS5jb20vZml2ZS10aWR5dmVyc2UtdHJpY2tzLXlvdS1tYXktbm90LWtub3ctYWJvdXQtYzUwMjZkNWExOWRhKQoKLSBSZWJlY2NhIEJhcnRlciB0YW1iacOpbiBoaXpvIHVuIFttdXkgYnVlbiBwb3N0XShodHRwOi8vd3d3LnJlYmVjY2FiYXJ0ZXIuY29tL2Jsb2cvMjAyMC0wNy0wOS1hY3Jvc3MvKS4gCgotIFF1aXrDoXMgZWwgdHV0b3JpYWwgbcOhcyBjb21wbGV0byBxdWUgaGUgdmlzdG8gbG8gaGl6byBlbCBlcXVpcG8gZGUgVGhpbmtSOiBbYXF1w61dKGh0dHBzOi8vdGhpbmtyLmZyL2hleS1xdW9pLWRlLW5ldWYtZHBseXItbGUtcG9pbnQtc3VyLWxhLXYxLykuIEVzdMOhIGVuIGZyYW5jw6lzLgoKCgoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCgo=