Data Structures and Data Attribute Types

Data comes in many different forms.

Some of the most common data structures are

Other forms include

We will work mostly with tables.

Many other forms can be reduced to tables.

Stevens (1945) classifies scales of measurement for attributes, or variables, as

These are sometimes grouped as

These can be viewed as semantic classifications

Computational considerations often classify variables as

Another consideration is that some scales may be cyclic:

These distinctions can be important in choosing visual representations.

Other typologies include one proposed by Mosteller and Tukey (1977):

  1. Names
  2. Grades (ordered labels like beginner, intermediate, advanced)
  3. Ranks (orders with 1 being the smallest or largest, 2 the next smallest or largest, and so on)
  4. Counted fractions (bound by 0 and 1)
  5. Counts (non-negative integers)
  6. Amounts (non-negative real numbers)
  7. Balances (any real number)

Data Types in R

R variables can be of different types.

The most common types are

Factors can be

factors are more efficient and powerful for representing nominal or ordinal data than character data but can take a bit more getting used to.

Membership predicates and coercion functions are

Predicate Coersion
is.numeric as.numeric
is.integer as.integer
is.character as.character
is.factor as.factor
is.ordered as.ordered

Conversion of factors with numeric-looking labels to numeric data should always go through as.character first.

Data Frames: Organizing Cases and Variables

Tabular data in R is usually stored as a data frame.

A data frame is a collection of variables, each with a value for every case or observacion.

The faithful data set is a data frame:

class(faithful)
## [1] "data.frame"
names(faithful)
## [1] "eruptions" "waiting"

Most tools we work with in R use data organized in data frames.

Our plot() and lm() expressions from the introductory section can also we written as

plot(waiting ~ eruptions, data = faithful,
     xlab = "Eruption time (min)",
     ylab = "Waiting time to next eruption (min)")
fit <- lm(waiting ~ eruptions, data = faithful)

plot() only uses the data argument when the plot is specified as a formula, like

waiting ~ eruptions

Examining the Data in a Data Frame

head() provides an idea of what the raw data looks like:

head(faithful)
##   eruptions waiting
## 1     3.600      79
## 2     1.800      54
## 3     3.333      74
## 4     2.283      62
## 5     4.533      85
## 6     2.883      55

str() is also useful for an overview of an object’s structure:

str(faithful)
## 'data.frame':    272 obs. of  2 variables:
##  $ eruptions: num  3.6 1.8 3.33 2.28 4.53 ...
##  $ waiting  : num  79 54 74 62 85 55 88 85 51 85 ...

Another useful function available from the dplyr or tibble packages is glimpse():

library(dplyr)
glimpse(faithful)
## Rows: 272
## Columns: 2
## $ eruptions <dbl> 3.600, 1.800, 3.333, 2.283, 4.533, 2.883, 4.700, 3.600, 1.95…
## $ waiting   <dbl> 79, 54, 74, 62, 85, 55, 88, 85, 51, 85, 54, 84, 78, 47, 83, …

summary() shows basic statistical properties of the variables:

summary(faithful)
##    eruptions        waiting    
##  Min.   :1.600   Min.   :43.0  
##  1st Qu.:2.163   1st Qu.:58.0  
##  Median :4.000   Median :76.0  
##  Mean   :3.488   Mean   :70.9  
##  3rd Qu.:4.454   3rd Qu.:82.0  
##  Max.   :5.100   Max.   :96.0

summary() with a character variable and a factor variable:

ffl <- mutate(faithful,
              type = ifelse(eruptions < 3, "short", "long"),
              ftype = factor(type))
summary(ffl)
##    eruptions        waiting         type             ftype    
##  Min.   :1.600   Min.   :43.0   Length:272         long :175  
##  1st Qu.:2.163   1st Qu.:58.0   Class :character   short: 97  
##  Median :4.000   Median :76.0   Mode  :character              
##  Mean   :3.488   Mean   :70.9                                 
##  3rd Qu.:4.454   3rd Qu.:82.0                                 
##  Max.   :5.100   Max.   :96.0

Variables in a Data Frame

A Data frame is a list, or vector, of variables:

length(faithful)
## [1] 2

The dollar sign $ can be used to examine individual variables:

class(faithful$eruptions)
## [1] "numeric"
class(faithful$waiting)
## [1] "numeric"

The variables can also be extracted by numerical or character index using the element extraction operation [[:

class(faithful[[1]])
## [1] "numeric"
class(faithful[["waiting"]])
## [1] "numeric"

Dimensions

The numbers of rows and columns can be obtained using nrow() and ncol():

ncol(faithful)
## [1] 2
nrow(faithful)
## [1] 272

dim() returns a vector of the dimensions:

dim(faithful)
## [1] 272   2

Simple Visualizations

plot has a method for data frames that tries to provide a reasonable default visualization for numeric data frames:

plot(faithful)

Sample Data Sets

The datasets package in the base R distribution contains a number of data sets you can explore.

Another package with a useful collection of data sets is dslabs.

Many other data sets are contained in contributed packages as examples.

There are also many contributed packages designed specifically for making particular data sets available.

Tidy Data

The useful concept of a tidy data frame was introduced fairly recently and is described in a chapter in R for Data Science.

The idea is that

Tidy data is computationally convenient, and many of the tools we will use are designed around tidy data frames.

A large range of these tools can be accessed by loading the tidyverse package:

library(tidyverse)

But for now I will load the needed packages individually.

The term tidy is a little unfortunate.

  • Data that is not tidy isn’t necessarily bad.
  • For human reading, and for some computations, data in a wider format can be better.
  • For other computations data in a longer, or narrower, format can be better.

Tibbles

Many tools in the tidyverse produce slightly enhanced data frames called tibbles:

library(tibble)
faithful_tbl <- as_tibble(faithful)
class(faithful_tbl)
## [1] "tbl_df"     "tbl"        "data.frame"

Tibbles print differently from standard data frames:

faithful_tbl
## # A tibble: 272 × 2
##    eruptions waiting
##        <dbl>   <dbl>
##  1      3.6       79
##  2      1.8       54
##  3      3.33      74
##  4      2.28      62
##  5      4.53      85
##  6      2.88      55
##  7      4.7       88
##  8      3.6       85
##  9      1.95      51
## 10      4.35      85
## # ℹ 262 more rows

For the most part data frames and tibbles can be used interchangeably.

Tidying Data

Many data sets are in tidy form already.

If they are not, they can be put into tidy form.

The tools for this are part of data technologies.

The tasks involved are part of what is sometimes called data wrangling.

An Example: Global Average Surface Temperatures

Among many data sets available at https://data.giss.nasa.gov/gistemp/ is data on monthly global average surface temperatures over a number of years.

These data from 2017 were used for the widely cited Bloomberg hottest year visualization.

The current data are available in a formatted text file at

https://data.giss.nasa.gov/gistemp/tabledata_v4/GLB.Ts+dSST.txt

or as a CSV (comma-separated values) file at

https://data.giss.nasa.gov/gistemp/tabledata_v4/GLB.Ts+dSST.csv

The first few lines of the CSV file:

    Land-Ocean: Global Means
    Year,Jan,Feb,Mar,Apr,May,Jun,Jul,Aug,Sep,Oct,Nov,Dec,J-D,D-N,DJF,MAM,JJA,SON
    1880,-.19,-.24,-.09,-.16,-.10,-.21,-.18,-.10,-.14,-.23,-.21,-.18,-.17,***,***,-.12,-.16,-.19
    1881,-.20,-.14,.03,.05,.06,-.19,.00,-.04,-.15,-.22,-.19,-.07,-.09,-.10,-.17,.05,-.08,-.19
    1882,.16,.14,.05,-.17,-.14,-.23,-.16,-.07,-.14,-.23,-.16,-.35,-.11,-.09,.08,-.09,-.15,-.18
    1883,-.29,-.37,-.12,-.18,-.17,-.08,-.06,-.14,-.21,-.11,-.23,-.11,-.17,-.19,-.34,-.16,-.09,-.18

The CSV file is a little easier (for a computer program) to read in, so we will work with that.

The numbers in the CSV file represent deviations in degrees Celcius from the average temperature for the base period 1951-1980.

The file available on January 15, 2024, is now available locally.

We can make sure it has been downloaded to our working directory with

if (! file.exists("GLB.Ts+dSST.csv"))
    download.file("https://stat.uiowa.edu/~luke/data/GLB.Ts+dSST.csv",
                  "GLB.Ts+dSST.csv")

Assuming this locally available file has been downloaded, we can read in the data and drop some columns we don’t need with

library(readr)
gast <- read_csv("GLB.Ts+dSST.csv", skip = 1)[1 : 13]
  • The function read_csv is from the readr package, which is part of the tidyverse.
  • An alternative is the base R function read.csv.

A look at the first few lines:

head(gast, 5)
## # A tibble: 5 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep   Oct   Nov   Dec
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
## 1  1880 -0.19 -0.24 -0.09 -0.16 -0.1  -0.21 -0.18 -0.1  -0.14 -0.23 -0.21 -0.18
## 2  1881 -0.2  -0.14  0.03  0.05  0.06 -0.19  0    -0.04 -0.15 -0.22 -0.19 -0.07
## 3  1882  0.16  0.14  0.05 -0.17 -0.14 -0.23 -0.16 -0.07 -0.14 -0.23 -0.16 -0.35
## 4  1883 -0.29 -0.37 -0.12 -0.18 -0.17 -0.08 -0.06 -0.14 -0.21 -0.11 -0.23 -0.11
## 5  1884 -0.13 -0.07 -0.36 -0.4  -0.34 -0.36 -0.3  -0.27 -0.27 -0.25 -0.33 -0.3

And the last few lines:

tail(gast, 5)
## # A tibble: 5 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep   Oct   Nov   Dec
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
## 1  2019  0.93  0.95  1.17  1.01  0.85  0.9   0.94  0.95  0.92  1     0.99  1.09
## 2  2020  1.17  1.24  1.17  1.13  1.01  0.92  0.9   0.87  0.98  0.88  1.1   0.8 
## 3  2021  0.81  0.64  0.89  0.75  0.78  0.84  0.92  0.82  0.92  1     0.94  0.86
## 4  2022  0.91  0.89  1.05  0.84  0.84  0.92  0.94  0.95  0.89  0.96  0.72  0.8 
## 5  2023  0.87  0.98  1.2   1     0.94  1.08  1.19  1.19  1.48  1.34  1.43  1.37

The print() method for tibbles abbreviates the output.

It is neater and provides some useful additional information on variable data types.

But it shows only the first few rows and may not explicitly show some columns.

If some columns are skipped, you can ask to see more by calling print() explicitly:

print(tail(gast), width = 100)
## # A tibble: 6 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep   Oct   Nov   Dec
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
## 1  2018  0.82  0.85  0.88  0.89  0.82  0.77  0.82  0.77  0.8   1.01  0.82  0.91
## 2  2019  0.93  0.95  1.17  1.01  0.85  0.9   0.94  0.95  0.92  1     0.99  1.09
## 3  2020  1.17  1.24  1.17  1.13  1.01  0.92  0.9   0.87  0.98  0.88  1.1   0.8 
## 4  2021  0.81  0.64  0.89  0.75  0.78  0.84  0.92  0.82  0.92  1     0.94  0.86
## 5  2022  0.91  0.89  1.05  0.84  0.84  0.92  0.94  0.95  0.89  0.96  0.72  0.8 
## 6  2023  0.87  0.98  1.2   1     0.94  1.08  1.19  1.19  1.48  1.34  1.43  1.37

The format with one column per month is compact and useful for viewing and data entry.

But it is not in tidy format since

For obvious reasons this data format is often referred to as wide format.

The tidy, or long, format would have three variables: Year, Month, and Temp.

One way to put this data frame in tidy, or long, format uses pivot_longer from the tidyr package:

library(tidyr)
lgast <- pivot_longer(gast,
                      -Year,  ## specifies the columns to use -- all but Year
                      names_to = "Month",
                      values_to = "Temp")

The first few rows of the result:

head(lgast)
## # A tibble: 6 × 3
##    Year Month  Temp
##   <dbl> <chr> <dbl>
## 1  1880 Jan   -0.19
## 2  1880 Feb   -0.24
## 3  1880 Mar   -0.09
## 4  1880 Apr   -0.16
## 5  1880 May   -0.1 
## 6  1880 Jun   -0.21

During plotting it is likely that the Month variable will be converted to a factor.

By default, this will order levels alphabetically, which is not what we want:

levels(as.factor(lgast$Month))
##  [1] "Apr" "Aug" "Dec" "Feb" "Jan" "Jul" "Jun" "Mar" "May" "Nov" "Oct" "Sep"

We can guard against this by converting Month to a factor with the right levels now:

lgast <- mutate(lgast, Month = factor(Month, levels = month.abb))
levels(lgast$Month)
##  [1] "Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec"

We can now use this tidy version of the data to create a static version of the Bloomberg hottest year visualization.

The basic framework is set up with

library(ggplot2)
p <- ggplot(lgast) +
    ggtitle("Global Average Surface Temperatures") +
    theme(plot.title = element_text(hjust = 0.5))

ggplot objects only produce output when they are printed.

To see the plot in p we need to print it, for example by using a line with only p.

 p

Then add a layer with lines for each year (specified by the group argument to geom_line).

p + geom_line(aes(x = Month,
                  y = Temp,
                  group = Year))

We can use color to distingish the years.

p1 <- p +
    geom_line(aes(x = Month,
                  y = Temp,
                  color = Year,
                  group = Year),
              na.rm = TRUE)
p1

Saving the plot specification in the variable p1 will make it easier to experiment with color variations:

One way to highlight the past_year 2023:

lgast_last <- filter(lgast, Year == past_year)
p1 + geom_line(aes(x = Month,
                   y = Temp,
                   group = Year),
               linewidth = 1,
               color = "red",
               data = lgast_last,
               na.rm = TRUE)

A useful way to show more recent data in the context of the full data is to show the full data in grey and the more recent years in black:

lgast2k <- filter(lgast, Year >= 2000)
ggplot(lgast, aes(x = Month,
                  y = Temp,
                  group = Year)) +
    geom_line(color = "grey80") +
    theme_minimal() +
    geom_line(data = lgast2k)

If you want to update your plot later in the year then the current year’s entry may contain missing value indicators that you will have to deal with.

The New York Times on January 18, 2018, published another visualization of these data showing average yearly temperatures (via Google may work better).

To recreate this plot we first need to compute yearly average temperatures.

This is easy to do with the summarize and group_by functions from dpyr:

library(dplyr)
atemp <- lgast |>
    group_by(Year) |>
    summarize(AveTemp = mean(Temp, na.rm = TRUE))
head(atemp)
## # A tibble: 6 × 2
##    Year AveTemp
##   <dbl>   <dbl>
## 1  1880 -0.169 
## 2  1881 -0.0883
## 3  1882 -0.108 
## 4  1883 -0.172 
## 5  1884 -0.282 
## 6  1885 -0.334

Using na.rm = TRUE ensures that the mean is based on the available months if data for some months is missing.

A simple version of the plot is then produced by

ggplot(atemp) +
    geom_point(aes(x = Year, y = AveTemp))

A variation showing record years:

library(ggrepel)
atemp_rec <- filter(atemp, cummax(AveTemp) == AveTemp)
ggplot(atemp, aes(x = Year, y = AveTemp)) +
    geom_point() +
    geom_point(data = atemp_rec, color = "red") +
    geom_text_repel(aes(label = Year),
                    data = atemp_rec,
                    color = "blue")

Another variation on the Bloomberg plot showing just a few years 20 years apart:

lg_by_20 <-
    filter(lgast,
           Year %in% seq(2020, by = -20, len = 5)) |>
    mutate(Year = factor(Year))
ggplot(lg_by_20, aes(x = Month,
                     y = Temp,
                     group = Year,
                     color = Year)) +
    geom_line()

Converting Year to a factor results in a discrete color scale and legend.

Handling Missing Values

The data for 2019 available in early 2020 is also available locally.

Assuming this locally available file has been downloaded, we can read in the data and drop some columns we don’t need with

gast2019 <- read_csv("GLB.Ts+dSST-2019.csv", skip = 1)[1 : 13]

The last three columns are read as character variables:

head(gast2019, 5)
## # A tibble: 5 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep Oct   Nov   Dec  
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <chr> <chr> <chr>
## 1  1880 -0.17 -0.23 -0.08 -0.15 -0.08 -0.2  -0.17 -0.09 -0.13 -.22  -.20  -.16 
## 2  1881 -0.18 -0.13  0.04  0.06  0.08 -0.17  0.02 -0.02 -0.14 -.20  -.17  -.05 
## 3  1882  0.18  0.15  0.06 -0.15 -0.13 -0.21 -0.15 -0.06 -0.13 -.23  -.15  -.34 
## 4  1883 -0.28 -0.36 -0.11 -0.17 -0.16 -0.07 -0.05 -0.12 -0.2  -.10  -.22  -.10 
## 5  1884 -0.12 -0.07 -0.36 -0.39 -0.33 -0.34 -0.32 -0.27 -0.26 -.24  -.32  -.30

The reason is that data for October through December were not available:

tail(gast2019, 2)
## # A tibble: 2 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep Oct   Nov   Dec  
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <chr> <chr> <chr>
## 1  2018  0.82  0.85  0.9   0.89  0.83  0.78  0.83  0.76  0.81 1.02  .83   .92  
## 2  2019  0.94  0.96  1.18  1.02  0.86  0.93  0.95  0.94  0.93 ***   ***   ***

We want to convert these columns to numeric, with missing values represented as NA.

One option is to handle them individually:

gast2019$Oct <- as.numeric(gast2019$Oct)
## Warning: NAs introduced by coercion
tail(gast2019, 2)
## # A tibble: 2 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep   Oct Nov   Dec  
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <chr> <chr>
## 1  2018  0.82  0.85  0.9   0.89  0.83  0.78  0.83  0.76  0.81  1.02 .83   .92  
## 2  2019  0.94  0.96  1.18  1.02  0.86  0.93  0.95  0.94  0.93 NA    ***   ***

Another option is to convert all character columns to numeric with

gast2019 <- mutate(gast2019, across(where(is.character), as.numeric))
## Warning: There were 2 warnings in `mutate()`.
## The first warning was:
## ℹ In argument: `across(where(is.character), as.numeric)`.
## Caused by warning:
## ! NAs introduced by coercion
## ℹ Run `dplyr::last_dplyr_warnings()` to see the 1 remaining warning.
tail(gast2019, 2)
## # A tibble: 2 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep   Oct   Nov   Dec
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
## 1  2018  0.82  0.85  0.9   0.89  0.83  0.78  0.83  0.76  0.81  1.02  0.83  0.92
## 2  2019  0.94  0.96  1.18  1.02  0.86  0.93  0.95  0.94  0.93 NA    NA    NA

The warnings are benign and can be suppressed with the warning = FALSE chunk option.

Since we know the missing value pattern *** we can also avoid the need to fix the data after the fact by specifying this at read time:

gast2019 <- read_csv("GLB.Ts+dSST-2019.csv", na = "***", skip = 1)[1 : 13]
tail(gast2019, 2)
## # A tibble: 2 × 13
##    Year   Jan   Feb   Mar   Apr   May   Jun   Jul   Aug   Sep   Oct   Nov   Dec
##   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
## 1  2018  0.82  0.85  0.9   0.89  0.83  0.78  0.83  0.76  0.81  1.02  0.83  0.92
## 2  2019  0.94  0.96  1.18  1.02  0.86  0.93  0.95  0.94  0.93 NA    NA    NA

A plot highlighting the year 2019 shows only the months with available data:

lgast2019 <- gast2019 |>
    pivot_longer(-Year,
                 names_to = "Month",
                 values_to = "Temp") |>
    mutate(Month = factor(Month, levels = month.abb))
ggplot(lgast2019, aes(x = Month,
                      y = Temp,
                      group = Year)) +
    geom_line(color = "grey80",
              na.rm = TRUE) +
    geom_line(data = filter(lgast2019, Year == 2019),
              na.rm = TRUE)

Adding na.rm = TRUE in the geom_line calls suppresses warnings; the plot would be the same without these.

Outline of the tools used:

  • Data processing:
    • Reading: read_csv, read.csv;
    • Reshaping: pivot_longer;
    • Cleaning: is.character, as.numeric, mutate, across, factor;
    • Summarizing: group_by, summarize.
  • Visualization geometries:
    • geom_line for a line plot;
    • geom_point for a scatter plot.

Reading

Stevens’ classification of scales of measurement is described in a Wikipedia article.

A good introduction to the concept of tidy data is provided in a chapter in R for Data Science.

Interactive Tutorial

An interactive learnr tutorial for these notes is available.

You can run the tutorial with

STAT4580::runTutorial("datafrm")

Exercises

  1. Which of the Stevens classifications (nominal, ordinal, interval, ratio) best characterizes these variables:

    1. Daily maximal temperatures in Iowa City.
    2. Population counts for Iowa counties.
    3. Education level of job applicants using the Bureau of Labor Statistics classification.
    4. Major of UI students.
  1. Which of these data sets are in tidy form?

    1. The builtin data set co2
    2. The builtin data set BOD
    3. The who data set in package tidyr (tidyr::who)
    4. The mpg data set in package ggplot2 (ggplot2::mpg)

The next exercises use the data in the variable gapminder in the package gapminder. You can make it available with

data(gapminder, package = "gapminder")
  1. Use the function str to examine the value of the gapminder variable. How many cases are there in the data set? How many of the variables are factors?

  2. Use the functions class and names to find the class and variable names in the gapminder data.

  3. Use summary to compute summary information for the variables.

  4. Fill in the values for --- needed to produce plots of life expectancy against year for the countries in continent Oceania.

library(dplyr)
library(ggplot2)
data(gapminder, package = "gapminder")
ggplot(filter(gapminder, continent == "Oceania"),
       aes(x = ---, y = ---, color = country)) +
    geom_line()
LS0tCnRpdGxlOiAiRGF0YSBhbmQgRGF0YSBGcmFtZXMiCm91dHB1dDoKICBodG1sX2RvY3VtZW50OgogICAgdG9jOiB5ZXMKICAgIGNvZGVfZG93bmxvYWQ6IHRydWUKLS0tCgo8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9InN0YXQ0NTgwLmNzcyIgdHlwZT0idGV4dC9jc3MiIC8+CgpgYGB7ciBzZXR1cCwgaW5jbHVkZT1GQUxTRX0Kc291cmNlKGhlcmU6OmhlcmUoInNldHVwLlIiKSkKa25pdHI6Om9wdHNfY2h1bmskc2V0KGNvbGxhcHNlID0gVFJVRSwgbWVzc2FnZSA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgZmlnLmhlaWdodCA9IDUsIGZpZy53aWR0aCA9IDYsIGZpZy5hbGlnbiA9ICJjZW50ZXIiKQpvcHRpb25zKHdpZHRoID0gODApCmBgYAoKCiMjIERhdGEgU3RydWN0dXJlcyBhbmQgRGF0YSBBdHRyaWJ1dGUgVHlwZXMKCkRhdGEgY29tZXMgaW4gbWFueSBkaWZmZXJlbnQgZm9ybXMuCgpTb21lIG9mIHRoZSBtb3N0IGNvbW1vbiBkYXRhIHN0cnVjdHVyZXMgYXJlCgoqIHJlY3Rhbmd1bGFyIHRhYmxlcwoqIG5ldHdvcmtzIGFuZCB0cmVlcwoqIGdlb21ldHJpZXMsIHJlZ2lvbnMKCk90aGVyIGZvcm1zIGluY2x1ZGUKCiogY29sbGVjdGlvbnMgb2YgdGV4dAoqIHZpZGVvIGFuZCBhdWRpbyByZWNvcmRpbmdzCiogLi4uCgpXZSB3aWxsIHdvcmsgbW9zdGx5IHdpdGggdGFibGVzLgoKTWFueSBvdGhlciBmb3JtcyBjYW4gYmUgcmVkdWNlZCB0byB0YWJsZXMuCgpbU3RldmVuc10oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvU3RhbmxleV9TbWl0aF9TdGV2ZW5zKSAoMTk0NSkKIGNsYXNzaWZpZXMgW3NjYWxlcyBvZgogbWVhc3VyZW1lbnRdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0xldmVsX29mX21lYXN1cmVtZW50KSBmb3IKIGF0dHJpYnV0ZXMsIG9yIHZhcmlhYmxlcywgYXMKPCEtLSBwZXJtYW5lbnQgbGluazogaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3cvaW5kZXgucGhwP3RpdGxlPUxldmVsX29mX21lYXN1cmVtZW50Jm9sZGlkPTEwNjA3NzI4NDcgLS0+IAoKKiBub21pbmFsIChlLmcuIGhhaXIgY29sb3IpCiogb3JkaW5hbCAoZS5nLiBkaXNsaWtlLCBuZXV0cmFsLCBsaWtlKQoqIGludGVydmFsIChlLmcuIHRlbXBlcmF0dXJlKSAtLS0gY29tcGFyZSBieSBkaWZmZXJlbmNlCiogcmF0aW8gKGUuZy4gY291bnRzKSAtLS0gY29tcGFyZSBieSByYXRpbwoKVGhlc2UgYXJlIHNvbWV0aW1lcyBncm91cGVkIGFzCgoqIHF1YWxpdGF0aXZlOiBub21pbmFsCiogcXVhbnRpdGF0aXZlOiBvcmRpbmFsLCBpbnRlcnZhbCwgcmF0aW8KClRoZXNlIGNhbiBiZSB2aWV3ZWQgYXMgX3NlbWFudGljIGNsYXNzaWZpY2F0aW9uc18KCl9Db21wdXRhdGlvbmFsIGNvbnNpZGVyYXRpb25zXyBvZnRlbiBjbGFzc2lmeSB2YXJpYWJsZXMgYXMKCiogY2F0ZWdvcmljYWwKKiBpbnRlZ2VyLCBkaXNjcmV0ZQoqIHJlYWwsIGNvbnRpbnVvdXMKCkFub3RoZXIgY29uc2lkZXJhdGlvbiBpcyB0aGF0IHNvbWUgc2NhbGVzIG1heSBiZSBjeWNsaWM6CgoqIGhvdXJzIG9mIHRoZSBkYXkKKiBhbmdsZXMsIGxvbmdpdHVkZQoKVGhlc2UgZGlzdGluY3Rpb25zIGNhbiBiZSBpbXBvcnRhbnQgaW4gY2hvb3NpbmcgdmlzdWFsIHJlcHJlc2VudGF0aW9ucy4KCk90aGVyIHR5cG9sb2dpZXMgaW5jbHVkZSBvbmUgcHJvcG9zZWQgYnkgW01vc3RlbGxlciBhbmQgVHVrZXkKKDE5NzcpXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9MZXZlbF9vZl9tZWFzdXJlbWVudCNNb3N0ZWxsZXJfYW5kX1R1a2V5J3NfdHlwb2xvZ3lfKDE5NzcpKToKCjEuIE5hbWVzCjIuIEdyYWRlcyAob3JkZXJlZCBsYWJlbHMgbGlrZSBiZWdpbm5lciwgaW50ZXJtZWRpYXRlLCBhZHZhbmNlZCkKMy4gUmFua3MgKG9yZGVycyB3aXRoIDEgYmVpbmcgdGhlIHNtYWxsZXN0IG9yIGxhcmdlc3QsIDIgdGhlIG5leHQKICAgc21hbGxlc3Qgb3IgbGFyZ2VzdCwgYW5kIHNvIG9uKQo0LiBDb3VudGVkIGZyYWN0aW9ucyAoYm91bmQgYnkgMCBhbmQgMSkKNS4gQ291bnRzIChub24tbmVnYXRpdmUgaW50ZWdlcnMpCjYuIEFtb3VudHMgKG5vbi1uZWdhdGl2ZSByZWFsIG51bWJlcnMpCjcuIEJhbGFuY2VzIChhbnkgcmVhbCBudW1iZXIpCgoKIyMgRGF0YSBUeXBlcyBpbiBSCgpSIHZhcmlhYmxlcyBjYW4gYmUgb2YgZGlmZmVyZW50IHR5cGVzLgoKVGhlIG1vc3QgY29tbW9uIHR5cGVzIGFyZQoKKiBgbnVtZXJpY2AgZm9yIHJlYWwgbnVtYmVycwoqIGBpbnRlZ2VyYAoqIGBjaGFyYWN0ZXJgIGZvciB0ZXh0IGRhdGEgb3Igbm9taW5hbCBkYXRhCiogYGZhY3RvcmAgZm9yIG5vbWluYWwgb3Igb3JkaW5hbCBkYXRhCgpGYWN0b3JzIGNhbiBiZQoKKiB1bm9yZGVyZWQsIGZvciBub21pbmFsIGRhdGEKKiBvcmRlcmVkLCBmb3Igb3JkaW5hbCBkYXRhCgo8ZGl2IGNsYXNzID0gImFsZXJ0Ij4KYGZhY3RvcnNgIGFyZSBtb3JlIGVmZmljaWVudCBhbmQgcG93ZXJmdWwgZm9yIHJlcHJlc2VudGluZyBub21pbmFsIG9yCm9yZGluYWwgZGF0YSB0aGFuIGBjaGFyYWN0ZXJgIGRhdGEgYnV0IGNhbiB0YWtlIGEgYml0IG1vcmUgZ2V0dGluZwp1c2VkIHRvLgo8L2Rpdj4KCk1lbWJlcnNoaXAgcHJlZGljYXRlcyBhbmQgY29lcmNpb24gZnVuY3Rpb25zIGFyZQoKfCBQcmVkaWNhdGUgICAgICAgfCBDb2Vyc2lvbnwKfDotLS0tLS0tLS0tLS0tLS0tfDotLS0tLS0tLS0tLS0tLS0tLXwKfCBgaXMubnVtZXJpY2AgICB8IGBhcy5udW1lcmljYCAgICB8CnwgYGlzLmludGVnZXJgICAgfCBgYXMuaW50ZWdlcmAgICAgfAp8IGBpcy5jaGFyYWN0ZXJgIHwgYGFzLmNoYXJhY3RlcmAgIHwKfCBgaXMuZmFjdG9yYCAgICB8IGBhcy5mYWN0b3JgICAgICB8CnwgYGlzLm9yZGVyZWRgICAgfCBgYXMub3JkZXJlZGAgICAgfAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CkNvbnZlcnNpb24gb2YgZmFjdG9ycyB3aXRoIG51bWVyaWMtbG9va2luZyBsYWJlbHMgdG8gbnVtZXJpYyBkYXRhCnNob3VsZCBhbHdheXMgZ28gdGhyb3VnaCBgYXMuY2hhcmFjdGVyYCBmaXJzdC4KPC9kaXY+CgoKIyMgRGF0YSBGcmFtZXM6IE9yZ2FuaXppbmcgQ2FzZXMgYW5kIFZhcmlhYmxlcwoKVGFidWxhciBkYXRhIGluIFIgaXMgdXN1YWxseSBzdG9yZWQgYXMgYSBfZGF0YSBmcmFtZV8uCgpBIGRhdGEgZnJhbWUgaXMgYSBjb2xsZWN0aW9uIG9mIF92YXJpYWJsZXNfLCBlYWNoIHdpdGggYSB2YWx1ZSBmb3IKZXZlcnkgX2Nhc2VfIG9yIF9vYnNlcnZhY2lvbl8uCgpUaGUgYGZhaXRoZnVsYCBkYXRhIHNldCBpcyBhIGRhdGEgZnJhbWU6CgpgYGB7cn0KY2xhc3MoZmFpdGhmdWwpCm5hbWVzKGZhaXRoZnVsKQpgYGAKCk1vc3QgdG9vbHMgd2Ugd29yayB3aXRoIGluIFIgdXNlIGRhdGEgb3JnYW5pemVkIGluIGRhdGEgZnJhbWVzLgoKT3VyIGBwbG90KClgIGFuZCBgbG0oKWAgZXhwcmVzc2lvbnMgZnJvbSB0aGUKW2ludHJvZHVjdG9yeSBzZWN0aW9uXShpbnRyby5odG1sKQpjYW4gYWxzbyB3ZSB3cml0dGVuIGFzCgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpwbG90KHdhaXRpbmcgfiBlcnVwdGlvbnMsIGRhdGEgPSBmYWl0aGZ1bCwKICAgICB4bGFiID0gIkVydXB0aW9uIHRpbWUgKG1pbikiLAogICAgIHlsYWIgPSAiV2FpdGluZyB0aW1lIHRvIG5leHQgZXJ1cHRpb24gKG1pbikiKQpmaXQgPC0gbG0od2FpdGluZyB+IGVydXB0aW9ucywgZGF0YSA9IGZhaXRoZnVsKQpgYGAKCjxkaXYgY2xhc3MgPSAiYWxlcnQiPgpgcGxvdCgpYCBvbmx5IHVzZXMgdGhlIGBkYXRhYCBhcmd1bWVudCB3aGVuIHRoZSBwbG90IGlzIHNwZWNpZmllZCBhcyBhCl9mb3JtdWxhXywgbGlrZQoKYGBgcgp3YWl0aW5nIH4gZXJ1cHRpb25zCmBgYAo8L2Rpdj4KCgojIyBFeGFtaW5pbmcgdGhlIERhdGEgaW4gYSBEYXRhIEZyYW1lCgpgaGVhZCgpYCBwcm92aWRlcyBhbiBpZGVhIG9mIHdoYXQgdGhlIHJhdyBkYXRhIGxvb2tzIGxpa2U6CmBgYHtyfQpoZWFkKGZhaXRoZnVsKQpgYGAKCmBzdHIoKWAgaXMgYWxzbyB1c2VmdWwgZm9yIGFuIG92ZXJ2aWV3IG9mIGFuIG9iamVjdCdzIHN0cnVjdHVyZToKYGBge3J9CnN0cihmYWl0aGZ1bCkKYGBgCgpBbm90aGVyIHVzZWZ1bCBmdW5jdGlvbiBhdmFpbGFibGUgZnJvbSB0aGUgYGRwbHlyYCBvciBgdGliYmxlYApwYWNrYWdlcyBpcyBgZ2xpbXBzZSgpYDoKCmBgYHtyfQpsaWJyYXJ5KGRwbHlyKQpnbGltcHNlKGZhaXRoZnVsKQpgYGAKCmBzdW1tYXJ5KClgIHNob3dzIGJhc2ljIHN0YXRpc3RpY2FsIHByb3BlcnRpZXMgb2YgdGhlIHZhcmlhYmxlczoKCmBgYHtyfQpzdW1tYXJ5KGZhaXRoZnVsKQpgYGAKCmBzdW1tYXJ5KClgIHdpdGggYSBjaGFyYWN0ZXIgdmFyaWFibGUgYW5kIGEgZmFjdG9yIHZhcmlhYmxlOgoKYGBge3J9CmZmbCA8LSBtdXRhdGUoZmFpdGhmdWwsCiAgICAgICAgICAgICAgdHlwZSA9IGlmZWxzZShlcnVwdGlvbnMgPCAzLCAic2hvcnQiLCAibG9uZyIpLAogICAgICAgICAgICAgIGZ0eXBlID0gZmFjdG9yKHR5cGUpKQpzdW1tYXJ5KGZmbCkKYGBgCgoKIyMgVmFyaWFibGVzIGluIGEgRGF0YSBGcmFtZQoKQSBEYXRhIGZyYW1lIGlzIGEgbGlzdCwgb3IgdmVjdG9yLCBvZiB2YXJpYWJsZXM6CmBgYHtyfQpsZW5ndGgoZmFpdGhmdWwpCmBgYAoKVGhlIGRvbGxhciBzaWduIGAkYCBjYW4gYmUgdXNlZCB0byBleGFtaW5lIGluZGl2aWR1YWwgdmFyaWFibGVzOgpgYGB7cn0KY2xhc3MoZmFpdGhmdWwkZXJ1cHRpb25zKQpjbGFzcyhmYWl0aGZ1bCR3YWl0aW5nKQpgYGAKClRoZSB2YXJpYWJsZXMgY2FuIGFsc28gYmUgZXh0cmFjdGVkIGJ5IG51bWVyaWNhbCBvciBjaGFyYWN0ZXIgaW5kZXggdXNpbmcgdGhlCiAgZWxlbWVudCBleHRyYWN0aW9uIG9wZXJhdGlvbiBgW1tgOgpgYGB7cn0KY2xhc3MoZmFpdGhmdWxbWzFdXSkKY2xhc3MoZmFpdGhmdWxbWyJ3YWl0aW5nIl1dKQpgYGAKCgojIyBEaW1lbnNpb25zCgpUaGUgbnVtYmVycyBvZiByb3dzIGFuZCBjb2x1bW5zIGNhbiBiZSBvYnRhaW5lZCB1c2luZyBgbnJvdygpYCBhbmQgYG5jb2woKWA6CgpgYGB7cn0KbmNvbChmYWl0aGZ1bCkKbnJvdyhmYWl0aGZ1bCkKYGBgCgpgZGltKClgIHJldHVybnMgYSB2ZWN0b3Igb2YgdGhlIGRpbWVuc2lvbnM6CgpgYGB7cn0KZGltKGZhaXRoZnVsKQpgYGAKCgojIyBTaW1wbGUgVmlzdWFsaXphdGlvbnMKCmBwbG90YCBoYXMgYSBfbWV0aG9kXyBmb3IgZGF0YSBmcmFtZXMgdGhhdCB0cmllcyB0byBwcm92aWRlIGEKcmVhc29uYWJsZSBkZWZhdWx0IHZpc3VhbGl6YXRpb24gZm9yIG51bWVyaWMgZGF0YSBmcmFtZXM6CmBgYHtyfQpwbG90KGZhaXRoZnVsKQpgYGAKCgojIyBTYW1wbGUgRGF0YSBTZXRzCgpUaGUKW2BkYXRhc2V0c2BdKGh0dHA6Ly9zdGF0LmV0aHouY2gvUi1tYW51YWwvUi1kZXZlbC9saWJyYXJ5L2RhdGFzZXRzL2h0bWwvMDBJbmRleC5odG1sKQpwYWNrYWdlIGluIHRoZSBiYXNlIFIgZGlzdHJpYnV0aW9uIGNvbnRhaW5zIGEgbnVtYmVyIG9mIGRhdGEgc2V0cyB5b3UKY2FuIGV4cGxvcmUuCgpBbm90aGVyIHBhY2thZ2Ugd2l0aCBhIHVzZWZ1bCBjb2xsZWN0aW9uIG9mIGRhdGEgc2V0cyBpcyBbYGRzbGFic2BdKGh0dHBzOi8vY3Jhbi5yLXByb2plY3Qub3JnL3BhY2thZ2U9ZHNsYWJzKS4KCk1hbnkgb3RoZXIgZGF0YSBzZXRzIGFyZSBjb250YWluZWQgaW4gY29udHJpYnV0ZWQgcGFja2FnZXMgYXMgZXhhbXBsZXMuCgpUaGVyZSBhcmUgYWxzbyBtYW55IGNvbnRyaWJ1dGVkIHBhY2thZ2VzIGRlc2lnbmVkIHNwZWNpZmljYWxseSBmb3IKbWFraW5nIHBhcnRpY3VsYXIgZGF0YSBzZXRzIGF2YWlsYWJsZS4KCgojIyBUaWR5IERhdGEKClRoZSB1c2VmdWwgY29uY2VwdCBvZiBhIF90aWR5XyBkYXRhIGZyYW1lIHdhcyBpbnRyb2R1Y2VkIGZhaXJseQpbcmVjZW50bHldKGh0dHBzOi8vd3d3LmpzdGF0c29mdC5vcmcvYXJ0aWNsZS92aWV3L3YwNTlpMTApIGFuZCBpcwpkZXNjcmliZWQgaW4gYSBbY2hhcHRlciBpbiBfUiBmb3IgRGF0YQpTY2llbmNlX10oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei9kYXRhLXRpZHkuaHRtbCkuCgpUaGUgaWRlYSBpcyB0aGF0CgoqIGV2ZXJ5IG9ic2VydmF0aW9uIHNob3VsZCBjb3JyZXNwb25kIHRvIGEgc2luZ2xlIHJvdzsKKiBldmVyeSB2YXJpYWJsZSBzaG91bGQgY29ycmVzcG9uZCB0byBhIHNpbmdsZSBjb2x1bW4uCgpUaWR5IGRhdGEgaXMgY29tcHV0YXRpb25hbGx5IGNvbnZlbmllbnQsIGFuZCBtYW55IG9mIHRoZSB0b29scyB3ZSB3aWxsCnVzZSBhcmUgZGVzaWduZWQgYXJvdW5kIHRpZHkgZGF0YSBmcmFtZXMuCgpBIGxhcmdlIHJhbmdlIG9mIHRoZXNlIHRvb2xzIGNhbiBiZSBhY2Nlc3NlZCBieSBsb2FkaW5nIHRoZQpgdGlkeXZlcnNlYCBwYWNrYWdlOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KbGlicmFyeSh0aWR5dmVyc2UpCmBgYAoKQnV0IGZvciBub3cgSSB3aWxsIGxvYWQgdGhlIG5lZWRlZCBwYWNrYWdlcyBpbmRpdmlkdWFsbHkuCgo8ZGl2IGNsYXNzID0gImFsZXJ0Ij4KVGhlIHRlcm0gX3RpZHlfIGlzIGEgbGl0dGxlIHVuZm9ydHVuYXRlLgoKKiBEYXRhIHRoYXQgaXMgbm90IF90aWR5XyBpc24ndCBuZWNlc3NhcmlseSBfYmFkXy4KKiBGb3IgaHVtYW4gcmVhZGluZywgYW5kICBmb3Igc29tZSBjb21wdXRhdGlvbnMsIGRhdGEgaW4gYSB3aWRlciBmb3JtYXQgY2FuCiAgYmUgYmV0dGVyLgoqIEZvciBvdGhlciBjb21wdXRhdGlvbnMgZGF0YSBpbiBhIGxvbmdlciwgb3IgbmFycm93ZXIsIGZvcm1hdCBjYW4gYmUgYmV0dGVyLgo8L2Rpdj4KCgojIyBUaWJibGVzCgpNYW55IHRvb2xzIGluIHRoZSBgdGlkeXZlcnNlYCBwcm9kdWNlIHNsaWdodGx5IGVuaGFuY2VkIGRhdGEgZnJhbWVzCmNhbGxlZCBfdGliYmxlc186CgpgYGB7cn0KbGlicmFyeSh0aWJibGUpCmZhaXRoZnVsX3RibCA8LSBhc190aWJibGUoZmFpdGhmdWwpCmNsYXNzKGZhaXRoZnVsX3RibCkKYGBgClRpYmJsZXMgcHJpbnQgZGlmZmVyZW50bHkgZnJvbSBzdGFuZGFyZCBkYXRhIGZyYW1lczoKCmBgYHtyLCBoaWdobGlnaHQub3V0cHV0PWMoMSwgMywgMTQpfQpmYWl0aGZ1bF90YmwKYGBgCgpGb3IgdGhlIG1vc3QgcGFydCBkYXRhIGZyYW1lcyBhbmQgdGliYmxlcyBjYW4gYmUgdXNlZCBpbnRlcmNoYW5nZWFibHkuCgoKIyMgVGlkeWluZyBEYXRhCgpNYW55IGRhdGEgc2V0cyBhcmUgaW4gdGlkeSBmb3JtIGFscmVhZHkuCgpJZiB0aGV5IGFyZSBub3QsIHRoZXkgY2FuIGJlIHB1dCBpbnRvIHRpZHkgZm9ybS4KClRoZSB0b29scyBmb3IgdGhpcyBhcmUgcGFydCBvZiBfZGF0YSB0ZWNobm9sb2dpZXNfLgoKVGhlIHRhc2tzIGludm9sdmVkIGFyZSBwYXJ0IG9mIHdoYXQgaXMgc29tZXRpbWVzIGNhbGxlZCBfZGF0YQp3cmFuZ2xpbmdfLgoKCiMjIEFuIEV4YW1wbGU6IEdsb2JhbCBBdmVyYWdlIFN1cmZhY2UgVGVtcGVyYXR1cmVzCgpBbW9uZyBtYW55IGRhdGEgc2V0cyBhdmFpbGFibGUgYXQKPGh0dHBzOi8vZGF0YS5naXNzLm5hc2EuZ292L2dpc3RlbXAvPiBpcyBkYXRhIG9uIG1vbnRobHkgZ2xvYmFsCmF2ZXJhZ2Ugc3VyZmFjZSB0ZW1wZXJhdHVyZXMgb3ZlciBhIG51bWJlciBvZiB5ZWFycy4KCjwhLS0gcGF5d2FsbGVkIG5vdywgc28gdXNlIHd3dy9hcmNoaXZlLm9yZyAtLT4KClRoZXNlIGRhdGEgZnJvbSAyMDE3IHdlcmUgdXNlZCBmb3IgdGhlIHdpZGVseSBjaXRlZApbQmxvb21iZXJnIGhvdHRlc3QgeWVhciB2aXN1YWxpemF0aW9uXShodHRwczovL3dlYi5hcmNoaXZlLm9yZy93ZWIvMjAxOTAyMDIxOTQ0MzIvaHR0cHM6Ly93d3cuYmxvb21iZXJnLmNvbS9ncmFwaGljcy9ob3R0ZXN0LXllYXItb24tcmVjb3JkLykuCgpUaGUgY3VycmVudCBkYXRhIGFyZSBhdmFpbGFibGUgaW4gYSBmb3JtYXR0ZWQgdGV4dCBmaWxlIGF0Cgo8aHR0cHM6Ly9kYXRhLmdpc3MubmFzYS5nb3YvZ2lzdGVtcC90YWJsZWRhdGFfdjQvR0xCLlRzK2RTU1QudHh0PgoKb3IgYXMgYQpbX0NTVl8gKGNvbW1hLXNlcGFyYXRlZCB2YWx1ZXMpXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9Db21tYS1zZXBhcmF0ZWRfdmFsdWVzKSBmaWxlIGF0Cgo8aHR0cHM6Ly9kYXRhLmdpc3MubmFzYS5nb3YvZ2lzdGVtcC90YWJsZWRhdGFfdjQvR0xCLlRzK2RTU1QuY3N2PgoKYGBge3IgZG93bmxvYWQtR0xCLCBlY2hvID0gRkFMU0V9CmBgYAoKVGhlIGZpcnN0IGZldyBsaW5lcyBvZiB0aGUgQ1NWIGZpbGU6CgpgYGB7cn0KI3wgZWNobzogZmFsc2UKI3wgY29tbWVudDogIiAgICIKcmVhZExpbmVzKCJHTEIuVHMrZFNTVC5jc3YiLCA2KSB8PiB3cml0ZUxpbmVzKCkKYGBgCgpUaGUgQ1NWIGZpbGUgaXMgYSBsaXR0bGUgZWFzaWVyIChmb3IgYSBjb21wdXRlciBwcm9ncmFtKSB0byByZWFkIGluLApzbyB3ZSB3aWxsIHdvcmsgd2l0aCB0aGF0LgoKVGhlIG51bWJlcnMgaW4gdGhlIENTViBmaWxlIHJlcHJlc2VudCBkZXZpYXRpb25zIGluIGRlZ3JlZXMgQ2VsY2l1cwpmcm9tIHRoZSBhdmVyYWdlIHRlbXBlcmF0dXJlIGZvciB0aGUgYmFzZSBwZXJpb2QgMTk1MS0xOTgwLgoKVGhlIGZpbGUgYXZhaWxhYmxlIG9uIEphbnVhcnkgMTUsIDIwMjQsIGlzIG5vdyBhdmFpbGFibGUKW2xvY2FsbHldKGh0dHBzOi8vc3RhdC51aW93YS5lZHUvfmx1a2UvZGF0YS9HTEIuVHMrZFNTVC5jc3YpLgoKV2UgY2FuIG1ha2Ugc3VyZSBpdCBoYXMgYmVlbiBkb3dubG9hZGVkIHRvIG91ciB3b3JraW5nIGRpcmVjdG9yeSB3aXRoCgpgYGB7ciBkb3dubG9hZC1HTEJ9CmlmICghIGZpbGUuZXhpc3RzKCJHTEIuVHMrZFNTVC5jc3YiKSkKICAgIGRvd25sb2FkLmZpbGUoImh0dHBzOi8vc3RhdC51aW93YS5lZHUvfmx1a2UvZGF0YS9HTEIuVHMrZFNTVC5jc3YiLAogICAgICAgICAgICAgICAgICAiR0xCLlRzK2RTU1QuY3N2IikKYGBgCgpBc3N1bWluZyB0aGlzIGxvY2FsbHkgYXZhaWxhYmxlIGZpbGUgaGFzIGJlZW4gZG93bmxvYWRlZCwgd2UgY2FuIHJlYWQKaW4gdGhlIGRhdGEgYW5kIGRyb3Agc29tZSBjb2x1bW5zIHdlIGRvbid0IG5lZWQgd2l0aAoKYGBge3J9CmxpYnJhcnkocmVhZHIpCmdhc3QgPC0gcmVhZF9jc3YoIkdMQi5UcytkU1NULmNzdiIsIHNraXAgPSAxKVsxIDogMTNdCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CiogVGhlIGZ1bmN0aW9uIGByZWFkX2NzdmAgaXMgZnJvbSB0aGUgYHJlYWRyYCBwYWNrYWdlLCB3aGljaCBpcyBwYXJ0IG9mIHRoZQogIGB0aWR5dmVyc2VgLgoqIEFuIGFsdGVybmF0aXZlIGlzIHRoZSBiYXNlIFIgZnVuY3Rpb24gYHJlYWQuY3N2YC4KPC9kaXY+CgpBIGxvb2sgYXQgdGhlIGZpcnN0IGZldyBsaW5lczoKCmBgYHtyfQpoZWFkKGdhc3QsIDUpCmBgYAoKQW5kIHRoZSBsYXN0IGZldyBsaW5lczoKCmBgYHtyfQp0YWlsKGdhc3QsIDUpCmBgYAoKVGhlIGBwcmludCgpYCBtZXRob2QgZm9yIHRpYmJsZXMgYWJicmV2aWF0ZXMgdGhlIG91dHB1dC4KCkl0IGlzIG5lYXRlciBhbmQgcHJvdmlkZXMgc29tZSB1c2VmdWwgYWRkaXRpb25hbCBpbmZvcm1hdGlvbiBvbgp2YXJpYWJsZSBkYXRhIHR5cGVzLgoKQnV0IGl0IHNob3dzIG9ubHkgdGhlIGZpcnN0IGZldyByb3dzIGFuZCBtYXkgbm90IGV4cGxpY2l0bHkgc2hvdyBzb21lCmNvbHVtbnMuCgpJZiBzb21lIGNvbHVtbnMgYXJlIHNraXBwZWQsIHlvdSBjYW4gYXNrIHRvIHNlZSBtb3JlIGJ5IGNhbGxpbmcKYHByaW50KClgIGV4cGxpY2l0bHk6CgpgYGB7cn0KcHJpbnQodGFpbChnYXN0KSwgd2lkdGggPSAxMDApCmBgYAoKPCEtLQoqIFRoZSBsYXN0IHZhbHVlcyBpbiB0aGUgYEF1Z2AgLSBgRGVjYCBjb2x1bW5zIGFyZSBtaXNzaW5nIGFuZCBjb2RlZCBhcyBgKioqYC4KKiBUaGlzIGNhdXNlcyB0aGVzZSBjb2x1bW5zIHRvIGJlIHJlYWQgYXMgY2hhcmFjdGVyIHZlY3RvcnMgaW5kaWNhdGVkCiAgYnkgYDxjaHI+YC4KKiBUaGUgb3RoZXJzIGhhdmUgYmVlbiByZWFkIGFzIG51bWVyaWMsIGNvZGVkIGA8ZGJsPmAgKGZvciAKICBfZG91YmxlIHByZWNpc2lvbl8pLgoqIFdlIGNhbiBmaXggdGhlIHRoZXNlIGNvbHVtbnMgbm93IG9yIGRlYWwgd2l0aCB0aGVtIGxhdGVyLgoKT25lIHdheSB0byBmaXggdGhlbSBub3cgaXMgdG8gdXNlIGBtdXRhdGVfaWZgOgoKYGBge3IsIHdhcm5pbmcgPSBGQUxTRX0KZ2FzdCA8LSBtdXRhdGUoZ2FzdCwgYWNyb3NzKHdoZXJlKGlzLmNoYXJhY3RlciksIGFzLm51bWVyaWMpKQpoZWFkKGdhc3QpCmBgYAoKRWFjaCBvYnNlcnZhdGlvbiBjb25zaXN0cyBvZiBhIHllYXIsIGEgbW9udGgsIGFuZCBhIHRlbXBlcmF0dXJlLgotLT4KClRoZSBmb3JtYXQgd2l0aCBvbmUgY29sdW1uIHBlciBtb250aCBpcyBjb21wYWN0IGFuZCB1c2VmdWwgZm9yIHZpZXdpbmcKYW5kIGRhdGEgZW50cnkuCgpCdXQgaXQgaXMgbm90IGluIF90aWR5IGZvcm1hdF8gc2luY2UKCiogdGhlIG1vbnRobHkgdGVtcGVyYXR1cmUgIHZhcmlhYmxlIGlzIHNwcmVhZCBvdmVyIDEyIGNvbHVtbnM7CiogdGhlIG1vbnRoIHZhcmlhYmxlIGlzIGVuY29kZWQgaW4gdGhlIGNvbHVtbiBoZWFkaW5ncy4KCkZvciBvYnZpb3VzIHJlYXNvbnMgdGhpcyBkYXRhIGZvcm1hdCBpcyBvZnRlbiByZWZlcnJlZCB0byBhcyBfd2lkZQpmb3JtYXRfLgoKVGhlIHRpZHksIG9yIF9sb25nXywgZm9ybWF0IHdvdWxkIGhhdmUgdGhyZWUgdmFyaWFibGVzOiBgWWVhcmAsCmBNb250aGAsIGFuZCBgVGVtcGAuCgpPbmUgd2F5IHRvIHB1dCB0aGlzIGRhdGEgZnJhbWUgaW4gdGlkeSwgb3IgbG9uZywgZm9ybWF0IHVzZXMgYHBpdm90X2xvbmdlcmAKZnJvbSB0aGUgYHRpZHlyYCBwYWNrYWdlOgoKYGBge3J9CmxpYnJhcnkodGlkeXIpCmxnYXN0IDwtIHBpdm90X2xvbmdlcihnYXN0LAogICAgICAgICAgICAgICAgICAgICAgLVllYXIsICAjIyBzcGVjaWZpZXMgdGhlIGNvbHVtbnMgdG8gdXNlIC0tIGFsbCBidXQgWWVhcgogICAgICAgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAiTW9udGgiLAogICAgICAgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gIlRlbXAiKQpgYGAKClRoZSBmaXJzdCBmZXcgcm93cyBvZiB0aGUgcmVzdWx0OgoKYGBge3J9CmhlYWQobGdhc3QpCmBgYAoKRHVyaW5nIHBsb3R0aW5nIGl0IGlzIGxpa2VseSB0aGF0IHRoZSBgTW9udGhgIHZhcmlhYmxlIHdpbGwgYmUKY29udmVydGVkIHRvIGEgX2ZhY3Rvcl8uCgpCeSBkZWZhdWx0LCB0aGlzIHdpbGwgb3JkZXIgbGV2ZWxzIGFscGhhYmV0aWNhbGx5LCB3aGljaCBpcyBub3Qgd2hhdCB3ZSB3YW50OgoKYGBge3J9CmxldmVscyhhcy5mYWN0b3IobGdhc3QkTW9udGgpKQpgYGAKCldlIGNhbiBndWFyZCBhZ2FpbnN0IHRoaXMgYnkgY29udmVydGluZyBgTW9udGhgIHRvIGEgZmFjdG9yIHdpdGggdGhlCnJpZ2h0IGxldmVscyBub3c6CgpgYGB7cn0KbGdhc3QgPC0gbXV0YXRlKGxnYXN0LCBNb250aCA9IGZhY3RvcihNb250aCwgbGV2ZWxzID0gbW9udGguYWJiKSkKbGV2ZWxzKGxnYXN0JE1vbnRoKQpgYGAKCjwhLS0gcGF5d2FsbGVkIG5vdywgc28gdXNlIHd3dy9hcmNoaXZlLm9yZyAtLT4KCldlIGNhbiBub3cgdXNlIHRoaXMgdGlkeSB2ZXJzaW9uIG9mIHRoZSBkYXRhIHRvIGNyZWF0ZSBhIHN0YXRpYwp2ZXJzaW9uIG9mIHRoZSBbQmxvb21iZXJnIGhvdHRlc3QgeWVhcgp2aXN1YWxpemF0aW9uXShodHRwczovL3dlYi5hcmNoaXZlLm9yZy93ZWIvMjAxOTAyMDIxOTQ0MzIvaHR0cHM6Ly93d3cuYmxvb21iZXJnLmNvbS9ncmFwaGljcy9ob3R0ZXN0LXllYXItb24tcmVjb3JkLykuCgpUaGUgYmFzaWMgZnJhbWV3b3JrIGlzIHNldCB1cCB3aXRoCgpgYGB7cn0KbGlicmFyeShnZ3Bsb3QyKQpwIDwtIGdncGxvdChsZ2FzdCkgKwogICAgZ2d0aXRsZSgiR2xvYmFsIEF2ZXJhZ2UgU3VyZmFjZSBUZW1wZXJhdHVyZXMiKSArCiAgICB0aGVtZShwbG90LnRpdGxlID0gZWxlbWVudF90ZXh0KGhqdXN0ID0gMC41KSkKYGBgCgo8ZGl2IGNsYXNzID0gImFsZXJ0Ij4KYGdncGxvdGAgb2JqZWN0cyBvbmx5IHByb2R1Y2Ugb3V0cHV0IHdoZW4gdGhleSBhcmUgcHJpbnRlZC4KClRvIHNlZSB0aGUgcGxvdCBpbiBgcGAgd2UgbmVlZCB0byBwcmludCBpdCwgZm9yIGV4YW1wbGUgYnkgdXNpbmcgYQpsaW5lIHdpdGggb25seSBgcGAuCjwvZGl2PgoKYGBge3IgZ2FzdC1iYXNlLCBldmFsID0gRkFMU0V9CiBwCmBgYApgYGB7ciBnYXN0LWJhc2UsIGV2YWwgPSBUUlVFLCBlY2hvID0gRkFMU0V9CmBgYAoKVGhlbiBhZGQgYSBsYXllciB3aXRoIGxpbmVzIGZvciBlYWNoIHllYXIgKHNwZWNpZmllZCBieSB0aGUgYGdyb3VwYAphcmd1bWVudCB0byBgZ2VvbV9saW5lYCkuCgpgYGB7ciBnYXN0LWxpbmVzLCBldmFsID0gRkFMU0V9CnAgKyBnZW9tX2xpbmUoYWVzKHggPSBNb250aCwKICAgICAgICAgICAgICAgICAgeSA9IFRlbXAsCiAgICAgICAgICAgICAgICAgIGdyb3VwID0gWWVhcikpCmBgYApgYGB7ciBnYXN0LWxpbmVzLCBldmFsID0gVFJVRSwgZWNobyA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0V9CmBgYAoKV2UgY2FuIHVzZSBjb2xvciB0byBkaXN0aW5naXNoIHRoZSB5ZWFycy4KCmBgYHtyIGdhc3QtY29sb3IxLCBldmFsID0gRkFMU0V9CnAxIDwtIHAgKwogICAgZ2VvbV9saW5lKGFlcyh4ID0gTW9udGgsCiAgICAgICAgICAgICAgICAgIHkgPSBUZW1wLAogICAgICAgICAgICAgICAgICBjb2xvciA9IFllYXIsCiAgICAgICAgICAgICAgICAgIGdyb3VwID0gWWVhciksCiAgICAgICAgICAgICAgbmEucm0gPSBUUlVFKQpwMQpgYGAKU2F2aW5nIHRoZSBwbG90IHNwZWNpZmljYXRpb24gaW4gdGhlIHZhcmlhYmxlIGBwMWAgd2lsbCBtYWtlIGl0IGVhc2llcgp0byBleHBlcmltZW50IHdpdGggY29sb3IgdmFyaWF0aW9uczoKCmBgYHtyIGdhc3QtY29sb3IxLCBldmFsID0gVFJVRSwgZWNobyA9IEZBTFNFfQpgYGAKCmBgYHtyLCBpbmNsdWRlID0gRkFMU0V9CiMjIENvbXB1dGUgdGhlIHBhc3QgeWVhciBhbmQgbWFrZSBzdXJlIGl0IGlzIGluIHRoZSBmaWxlCmxpYnJhcnkobHVicmlkYXRlKQpwYXN0X3llYXIgPC0geWVhcih0b2RheSgpKSAtIDEKcGFzdF95ZWFyCnN0b3BpZm5vdChwYXN0X3llYXIgJWluJSBsZ2FzdCRZZWFyKQpgYGAKCk9uZSB3YXkgdG8gaGlnaGxpZ2h0IHRoZSBgcGFzdF95ZWFyYCBgciBwYXN0X3llYXJgOgoKYGBge3IgZ2FzdC1jb2xvcjIsIGV2YWwgPSBGQUxTRX0KbGdhc3RfbGFzdCA8LSBmaWx0ZXIobGdhc3QsIFllYXIgPT0gcGFzdF95ZWFyKQpwMSArIGdlb21fbGluZShhZXMoeCA9IE1vbnRoLAogICAgICAgICAgICAgICAgICAgeSA9IFRlbXAsCiAgICAgICAgICAgICAgICAgICBncm91cCA9IFllYXIpLAogICAgICAgICAgICAgICBsaW5ld2lkdGggPSAxLAogICAgICAgICAgICAgICBjb2xvciA9ICJyZWQiLAogICAgICAgICAgICAgICBkYXRhID0gbGdhc3RfbGFzdCwKICAgICAgICAgICAgICAgbmEucm0gPSBUUlVFKQpgYGAKYGBge3IgZ2FzdC1jb2xvcjIsIGVjaG8gPSBGQUxTRX0KYGBgCgpBIHVzZWZ1bCB3YXkgdG8gc2hvdyBtb3JlIHJlY2VudCBkYXRhIGluIHRoZSBjb250ZXh0IG9mIHRoZSBmdWxsIGRhdGEKaXMgdG8gc2hvdyB0aGUgZnVsbCBkYXRhIGluIGdyZXkgYW5kIHRoZSBtb3JlIHJlY2VudCB5ZWFycyBpbiBibGFjazoKCmBgYHtyIGdhc3QtZnVsbC1ncmV5LCBldmFsID0gRkFMU0V9CmxnYXN0MmsgPC0gZmlsdGVyKGxnYXN0LCBZZWFyID49IDIwMDApCmdncGxvdChsZ2FzdCwgYWVzKHggPSBNb250aCwKICAgICAgICAgICAgICAgICAgeSA9IFRlbXAsCiAgICAgICAgICAgICAgICAgIGdyb3VwID0gWWVhcikpICsKICAgIGdlb21fbGluZShjb2xvciA9ICJncmV5ODAiKSArCiAgICB0aGVtZV9taW5pbWFsKCkgKwogICAgZ2VvbV9saW5lKGRhdGEgPSBsZ2FzdDJrKQpgYGAKYGBge3IgZ2FzdC1mdWxsLWdyZXksIGVjaG8gPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFfQpgYGAKCklmIHlvdSB3YW50IHRvIHVwZGF0ZSB5b3VyIHBsb3QgbGF0ZXIgaW4gdGhlIHllYXIgdGhlbiB0aGUgY3VycmVudAp5ZWFyJ3MgZW50cnkgbWF5IGNvbnRhaW4gbWlzc2luZyB2YWx1ZSBpbmRpY2F0b3JzIHRoYXQgeW91IHdpbGwgaGF2ZQp0byBkZWFsIHdpdGguCjwhLS0KVGhlIHZlcnNpb24gb2YgdGhlIGRhdGEgW29uIHRoZQp3ZWJdKGh0dHBzOi8vZGF0YS5naXNzLm5hc2EuZ292L2dpc3RlbXAvdGFibGVkYXRhX3YzL0dMQi5UcytkU1NULnR4dCkKbWF5IGhhdmUgYmVlbiB1cGRhdGVkIHRvIGluY2x1ZGUgdGhlIG1pc3NpbmcgdmFsdWVzIGZvciBgciBwYXN0X3llYXJgLgpJZiB5b3Ugd2FudCB0byB1cGRhdGUgeW91ciBwbG90IGxhdGVyIGluIHRoZSB5ZWFyIHlvdSB3aWxsIHNlZQpzaW1pbGFyIG1pc3NpbmcgdmFsdWUgbWFya2VycyBmb3IgdGhlIHJlbWFpbmluZyBtb250aHMgb2YKYHIgKHBhc3RfeWVhciArIDEpYC4KLS0+CgpUaGUgTmV3IFlvcmsgVGltZXMgb24gSmFudWFyeSAxOCwgMjAxOCwgcHVibGlzaGVkClthbm90aGVyIHZpc3VhbGl6YXRpb25dKGh0dHBzOi8vd3d3Lm55dGltZXMuY29tL2ludGVyYWN0aXZlLzIwMTgvMDEvMTgvY2xpbWF0ZS9ob3R0ZXN0LXllYXItMjAxNy5odG1sKQpvZiB0aGVzZSBkYXRhIHNob3dpbmcgYXZlcmFnZSB5ZWFybHkgdGVtcGVyYXR1cmVzIChbdmlhIEdvb2dsZSBtYXkgd29yayBiZXR0ZXJdKGh0dHBzOi8vd3d3Lmdvb2dsZS5jb20vdXJsP3NhPXQmcmN0PWomcT0mZXNyYz1zJnNvdXJjZT13ZWImY2Q9JmNhZD1yamEmdWFjdD04JnZlZD0yYWhVS0V3al80UG1obk0zMUFoWEZqWWtFSGZReEFURVFGbm9FQ0FZUUFRJnVybD1odHRwcyUzQSUyRiUyRnd3dy5ueXRpbWVzLmNvbSUyRmludGVyYWN0aXZlJTJGMjAxOCUyRjAxJTJGMTglMkZjbGltYXRlJTJGaG90dGVzdC15ZWFyLTIwMTcuaHRtbCZ1c2c9QU92VmF3M3FvdXhIUUFKQ3FydF9GM2ZiTEVHRSkpLgoKVG8gcmVjcmVhdGUgdGhpcyBwbG90IHdlIGZpcnN0IG5lZWQgdG8gY29tcHV0ZSB5ZWFybHkgYXZlcmFnZSB0ZW1wZXJhdHVyZXMuCgpUaGlzIGlzIGVhc3kgdG8gZG8gd2l0aCB0aGUgYHN1bW1hcml6ZWAgYW5kIGBncm91cF9ieWAgZnVuY3Rpb25zIGZyb20gYGRweXJgOgoKYGBge3J9CmxpYnJhcnkoZHBseXIpCmF0ZW1wIDwtIGxnYXN0IHw+CiAgICBncm91cF9ieShZZWFyKSB8PgogICAgc3VtbWFyaXplKEF2ZVRlbXAgPSBtZWFuKFRlbXAsIG5hLnJtID0gVFJVRSkpCmhlYWQoYXRlbXApCmBgYAoKVXNpbmcgYG5hLnJtID0gVFJVRWAgZW5zdXJlcyB0aGF0IHRoZSBtZWFuIGlzIGJhc2VkIG9uIHRoZSBhdmFpbGFibGUKbW9udGhzIGlmIGRhdGEgZm9yIHNvbWUgbW9udGhzIGlzIG1pc3NpbmcuCgpBIHNpbXBsZSB2ZXJzaW9uIG9mIHRoZSBwbG90IGlzIHRoZW4gcHJvZHVjZWQgYnkKCmBgYHtyIGdhc3Qtbnl0LCBldmFsID0gRkFMU0V9CmdncGxvdChhdGVtcCkgKwogICAgZ2VvbV9wb2ludChhZXMoeCA9IFllYXIsIHkgPSBBdmVUZW1wKSkKYGBgCmBgYHtyIGdhc3Qtbnl0LCBlY2hvID0gRkFMU0V9CmBgYAoKQSB2YXJpYXRpb24gc2hvd2luZyByZWNvcmQgeWVhcnM6CgpgYGB7ciAgZ2FzdC1ueXQtcmVjLCBldmFsID0gRkFMU0V9CmxpYnJhcnkoZ2dyZXBlbCkKYXRlbXBfcmVjIDwtIGZpbHRlcihhdGVtcCwgY3VtbWF4KEF2ZVRlbXApID09IEF2ZVRlbXApCmdncGxvdChhdGVtcCwgYWVzKHggPSBZZWFyLCB5ID0gQXZlVGVtcCkpICsKICAgIGdlb21fcG9pbnQoKSArCiAgICBnZW9tX3BvaW50KGRhdGEgPSBhdGVtcF9yZWMsIGNvbG9yID0gInJlZCIpICsKICAgIGdlb21fdGV4dF9yZXBlbChhZXMobGFiZWwgPSBZZWFyKSwKICAgICAgICAgICAgICAgICAgICBkYXRhID0gYXRlbXBfcmVjLAogICAgICAgICAgICAgICAgICAgIGNvbG9yID0gImJsdWUiKQpgYGAKYGBge3IgZ2FzdC1ueXQtcmVjLCBlY2hvID0gRkFMU0V9CmBgYAoKQW5vdGhlciB2YXJpYXRpb24gb24gdGhlIEJsb29tYmVyZyBwbG90IHNob3dpbmcganVzdCBhIGZldyB5ZWFycyAyMAp5ZWFycyBhcGFydDoKCmBgYHtyIGdhc3Qtc2tpcCwgZXZhbCA9IEZBTFNFfQpsZ19ieV8yMCA8LQogICAgZmlsdGVyKGxnYXN0LAogICAgICAgICAgIFllYXIgJWluJSBzZXEoMjAyMCwgYnkgPSAtMjAsIGxlbiA9IDUpKSB8PgogICAgbXV0YXRlKFllYXIgPSBmYWN0b3IoWWVhcikpCmdncGxvdChsZ19ieV8yMCwgYWVzKHggPSBNb250aCwKICAgICAgICAgICAgICAgICAgICAgeSA9IFRlbXAsCiAgICAgICAgICAgICAgICAgICAgIGdyb3VwID0gWWVhciwKICAgICAgICAgICAgICAgICAgICAgY29sb3IgPSBZZWFyKSkgKwogICAgZ2VvbV9saW5lKCkKYGBgCkNvbnZlcnRpbmcgYFllYXJgIHRvIGEgZmFjdG9yIHJlc3VsdHMgaW4gYSBkaXNjcmV0ZSBjb2xvciBzY2FsZSBhbmQgbGVnZW5kLgpgYGB7ciBnYXN0LXNraXAsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgSGFuZGxpbmcgTWlzc2luZyBWYWx1ZXMKClRoZSBkYXRhIGZvciAyMDE5IGF2YWlsYWJsZSBpbiBlYXJseSAyMDIwIGlzIGFsc28gYXZhaWxhYmxlCltsb2NhbGx5XShodHRwczovL3N0YXQudWlvd2EuZWR1L35sdWtlL2RhdGEvR0xCLlRzK2RTU1QtMjAxOS5jc3YpLgoKYGBge3IsIGVjaG8gPSBGQUxTRX0KaWYgKCEgZmlsZS5leGlzdHMoIkdMQi5UcytkU1NULTIwMTkuY3N2IikpCiAgICBkb3dubG9hZC5maWxlKCJodHRwczovL3N0YXQudWlvd2EuZWR1L35sdWtlL2RhdGEvR0xCLlRzK2RTU1QtMjAxOS5jc3YiLAogICAgICAgICAgICAgICAgICAiR0xCLlRzK2RTU1QtMjAxOS5jc3YiKQpgYGAKCkFzc3VtaW5nIHRoaXMgbG9jYWxseSBhdmFpbGFibGUgZmlsZSBoYXMgYmVlbiBkb3dubG9hZGVkLCB3ZSBjYW4gcmVhZAppbiB0aGUgZGF0YSBhbmQgZHJvcCBzb21lIGNvbHVtbnMgd2UgZG9uJ3QgbmVlZCB3aXRoCgpgYGB7cn0KZ2FzdDIwMTkgPC0gcmVhZF9jc3YoIkdMQi5UcytkU1NULTIwMTkuY3N2Iiwgc2tpcCA9IDEpWzEgOiAxM10KYGBgCgpUaGUgbGFzdCB0aHJlZSBjb2x1bW5zIGFyZSByZWFkIGFzIGNoYXJhY3RlciB2YXJpYWJsZXM6CgpgYGB7cn0KaGVhZChnYXN0MjAxOSwgNSkKYGBgCgpUaGUgcmVhc29uIGlzIHRoYXQgZGF0YSBmb3IgT2N0b2JlciB0aHJvdWdoIERlY2VtYmVyIHdlcmUgbm90CmF2YWlsYWJsZToKCmBgYHtyfQp0YWlsKGdhc3QyMDE5LCAyKQpgYGAKCldlIHdhbnQgdG8gY29udmVydCB0aGVzZSBjb2x1bW5zIHRvIG51bWVyaWMsIHdpdGggbWlzc2luZyB2YWx1ZXMKcmVwcmVzZW50ZWQgYXMgYE5BYC4KCk9uZSBvcHRpb24gaXMgdG8gaGFuZGxlIHRoZW0gaW5kaXZpZHVhbGx5OgoKYGBge3J9Cmdhc3QyMDE5JE9jdCA8LSBhcy5udW1lcmljKGdhc3QyMDE5JE9jdCkKdGFpbChnYXN0MjAxOSwgMikKYGBgCgpBbm90aGVyIG9wdGlvbiBpcyB0byBjb252ZXJ0IGFsbCBjaGFyYWN0ZXIgY29sdW1ucyB0byBudW1lcmljIHdpdGgKCmBgYHtyfQpnYXN0MjAxOSA8LSBtdXRhdGUoZ2FzdDIwMTksIGFjcm9zcyh3aGVyZShpcy5jaGFyYWN0ZXIpLCBhcy5udW1lcmljKSkKdGFpbChnYXN0MjAxOSwgMikKYGBgCgpUaGUgd2FybmluZ3MgYXJlIGJlbmlnbiBhbmQgY2FuIGJlIHN1cHByZXNzZWQgd2l0aCB0aGUgYHdhcm5pbmcgPQpGQUxTRWAgY2h1bmsgb3B0aW9uLgoKU2luY2Ugd2Uga25vdyB0aGUgbWlzc2luZyB2YWx1ZSBwYXR0ZXJuIGAqKipgIHdlIGNhbiBhbHNvIGF2b2lkIHRoZQpuZWVkIHRvIGZpeCB0aGUgZGF0YSBhZnRlciB0aGUgZmFjdCBieSBzcGVjaWZ5aW5nIHRoaXMgYXQgcmVhZCB0aW1lOgoKYGBge3J9Cmdhc3QyMDE5IDwtIHJlYWRfY3N2KCJHTEIuVHMrZFNTVC0yMDE5LmNzdiIsIG5hID0gIioqKiIsIHNraXAgPSAxKVsxIDogMTNdCnRhaWwoZ2FzdDIwMTksIDIpCmBgYAoKQSBwbG90IGhpZ2hsaWdodGluZyB0aGUgeWVhciAyMDE5IHNob3dzIG9ubHkgdGhlIG1vbnRocyB3aXRoIGF2YWlsYWJsZQpkYXRhOgoKYGBge3IgZ2FzdC0yMDE5LCBldmFsID0gRkFMU0V9CmxnYXN0MjAxOSA8LSBnYXN0MjAxOSB8PgogICAgcGl2b3RfbG9uZ2VyKC1ZZWFyLAogICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gIk1vbnRoIiwKICAgICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAiVGVtcCIpIHw+CiAgICBtdXRhdGUoTW9udGggPSBmYWN0b3IoTW9udGgsIGxldmVscyA9IG1vbnRoLmFiYikpCmdncGxvdChsZ2FzdDIwMTksIGFlcyh4ID0gTW9udGgsCiAgICAgICAgICAgICAgICAgICAgICB5ID0gVGVtcCwKICAgICAgICAgICAgICAgICAgICAgIGdyb3VwID0gWWVhcikpICsKICAgIGdlb21fbGluZShjb2xvciA9ICJncmV5ODAiLAogICAgICAgICAgICAgIG5hLnJtID0gVFJVRSkgKwogICAgZ2VvbV9saW5lKGRhdGEgPSBmaWx0ZXIobGdhc3QyMDE5LCBZZWFyID09IDIwMTkpLAogICAgICAgICAgICAgIG5hLnJtID0gVFJVRSkKYGBgCgpBZGRpbmcgYG5hLnJtID0gVFJVRWAgaW4gdGhlIGBnZW9tX2xpbmVgIGNhbGxzIHN1cHByZXNzZXMgd2FybmluZ3M7CnRoZSBwbG90IHdvdWxkIGJlIHRoZSBzYW1lIHdpdGhvdXQgdGhlc2UuCgpgYGB7ciBnYXN0LTIwMTksIGVjaG8gPSBGQUxTRX0KYGBgCgo8ZGl2IGNsYXNzID0gImFsZXJ0Ij4KT3V0bGluZSBvZiB0aGUgdG9vbHMgdXNlZDoKCiogRGF0YSBwcm9jZXNzaW5nOgogICAgKiBSZWFkaW5nOiBgcmVhZF9jc3ZgLCBgcmVhZC5jc3ZgOwogICAgKiBSZXNoYXBpbmc6IGBwaXZvdF9sb25nZXJgOwogICAgKiBDbGVhbmluZzogYGlzLmNoYXJhY3RlcmAsIGBhcy5udW1lcmljYCwgYG11dGF0ZWAsIGBhY3Jvc3NgLCBgZmFjdG9yYDsKICAgICogU3VtbWFyaXppbmc6IGBncm91cF9ieWAsIGBzdW1tYXJpemVgLgoqIFZpc3VhbGl6YXRpb24gZ2VvbWV0cmllczoKICAgICogYGdlb21fbGluZWAgZm9yIGEgbGluZSBwbG90OwogICAgKiBgZ2VvbV9wb2ludGAgZm9yIGEgc2NhdHRlciBwbG90Lgo8L2Rpdj4KCgojIyBSZWFkaW5nCgpTdGV2ZW5zJyBjbGFzc2lmaWNhdGlvbiBvZiBzY2FsZXMgb2YgbWVhc3VyZW1lbnQgaXMgZGVzY3JpYmVkIGluIGEKW1dpa2lwZWRpYQphcnRpY2xlXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9MZXZlbF9vZl9tZWFzdXJlbWVudCkuCgpBIGdvb2QgaW50cm9kdWN0aW9uIHRvIHRoZSBjb25jZXB0IG9mIF90aWR5IGRhdGFfIGlzIHByb3ZpZGVkIGluIGEKW2NoYXB0ZXIgaW4gX1IgZm9yIERhdGEKU2NpZW5jZV9dKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovZGF0YS10aWR5Lmh0bWwpLgoKCiMjIEludGVyYWN0aXZlIFR1dG9yaWFsCgpBbiBpbnRlcmFjdGl2ZSBbYGxlYXJucmBdKGh0dHBzOi8vcnN0dWRpby5naXRodWIuaW8vbGVhcm5yLykgdHV0b3JpYWwKZm9yIHRoZXNlIG5vdGVzIGlzIFthdmFpbGFibGVdKGByIFdMTksoInR1dG9yaWFscy9kYXRhZnJtLlJtZCIpYCkuCgpZb3UgY2FuIHJ1biB0aGUgdHV0b3JpYWwgd2l0aAoKYGBge3IsIGV2YWwgPSBGQUxTRX0KU1RBVDQ1ODA6OnJ1blR1dG9yaWFsKCJkYXRhZnJtIikKYGBgCgoKIyMgRXhlcmNpc2VzCgoxLiBXaGljaCBvZiB0aGUgU3RldmVucyBjbGFzc2lmaWNhdGlvbnMgKG5vbWluYWwsIG9yZGluYWwsIGludGVydmFsLCByYXRpbykKICAgYmVzdCBjaGFyYWN0ZXJpemVzIHRoZXNlIHZhcmlhYmxlczoKCiAgICBhLiBEYWlseSBtYXhpbWFsIHRlbXBlcmF0dXJlcyBpbiBJb3dhIENpdHkuCiAgICBiLiBQb3B1bGF0aW9uIGNvdW50cyBmb3IgSW93YSBjb3VudGllcy4KICAgIGMuIEVkdWNhdGlvbiBsZXZlbCBvZiBqb2IgYXBwbGljYW50cyB1c2luZyB0aGUgW0J1cmVhdSBvZiBMYWJvcgogICAgICAgU3RhdGlzdGljcwogICAgICAgY2xhc3NpZmljYXRpb25dKGh0dHBzOi8vd3d3LmJscy5nb3YvY2FyZWVyb3V0bG9vay8yMDE0L2FydGljbGUvZWR1Y2F0aW9uLWxldmVsLWFuZC1qb2JzLmh0bSkuCiAgICBkLiBNYWpvciBvZiBVSSBzdHVkZW50cy4KCjwhLS0KV2hpY2ggYW5zd2VycyBhcmUgY29ycmVjdCBmb3IgZXhlcmNpc2UgMToKKiBhOiBpbnRlcnZhbDsgYjogcmF0aW87IGM6IG9yZGluYWw7IGQ6IG5vbWluYWwKICBhOiBub21pbmFsOyBiOiBpbnRlcnZhbDsgYzogb3JkaW5hbDsgZDogcmF0aW8KICBhOiBub21pbmFsOyBiOiBpbnRlcnZhbDsgYzogb3JkaW5hbDsgZDogcmF0aW8KICBhOiBpbnRlcnZhbDsgYjogcmF0aW87IGM6IG5vbWluYWw7IGQ6IG9yZGluYWwKZm10IDwtIGZ1bmN0aW9uKHgpCiAgICBwYXN0ZShsZXR0ZXJzW3NlcV9hbG9uZyh4KV0sIHgsIHNlcCA9ICI6ICIsIGNvbGxhcHNlID0gIjsgIikKc3RldiA8LSBjKCJub21pbmFsIiwgIm9yZGluYWwiLCAiaW50ZXJ2YWwiLCAicmF0aW8iKQpmbXQoc2FtcGxlKHN0ZXYsIDQpKQotLT4KCjIuIFdoaWNoIG9mIHRoZXNlIGRhdGEgc2V0cyBhcmUgaW4gdGlkeSBmb3JtPwoKICAgIGEuIFRoZSBidWlsdGluIGRhdGEgc2V0IGBjbzJgCiAgICBiLiBUaGUgYnVpbHRpbiBkYXRhIHNldCBgQk9EYAogICAgYy4gVGhlIGB3aG9gIGRhdGEgc2V0IGluIHBhY2thZ2UgYHRpZHlyYCAoYHRpZHlyOjp3aG9gKQogICAgZC4gVGhlIGBtcGdgIGRhdGEgc2V0IGluIHBhY2thZ2UgYGdncGxvdDJgIChgZ2dwbG90Mjo6bXBnYCkKCjwhLS0KV2hpY2ggYW5zd2VycyBhcmUgY29ycmVjdCBmb3IgZXhlcmNpc2UgMjoKKiBhOiBub3QgdGlkeTsgYjogdGlkeTsgYzogbm90IHRpZHk7IGQ6IHRpZHkKKiBhOiB0aWR5OyBiOiB0aWR5OyBjOiBub3QgdGlkeTsgZDogdGlkeQoqIGE6IG5vdCB0aWR5OyBiOiBub3QgdGlkeTsgYzogbm90IHRpZHk7IGQ6IHRpZHkKKiBhOiBub3QgdGlkeTsgYjogdGlkeTsgYzogdGlkeTsgZDogbm90IHRpZHkKLS0+CgpUaGUgbmV4dCBleGVyY2lzZXMgdXNlIHRoZSBkYXRhIGluIHRoZSB2YXJpYWJsZSBgZ2FwbWluZGVyYCBpbiB0aGUgcGFja2FnZQpgZ2FwbWluZGVyYC4gWW91IGNhbiBtYWtlIGl0IGF2YWlsYWJsZSB3aXRoCgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpkYXRhKGdhcG1pbmRlciwgcGFja2FnZSA9ICJnYXBtaW5kZXIiKQpgYGAKMy4gVXNlIHRoZSBmdW5jdGlvbiBgc3RyYCB0byBleGFtaW5lIHRoZSB2YWx1ZSBvZiB0aGUgZ2FwbWluZGVyCiAgIHZhcmlhYmxlLiAgSG93IG1hbnkgY2FzZXMgYXJlIHRoZXJlIGluIHRoZSBkYXRhIHNldD8gSG93IG1hbnkgb2YKICAgdGhlIHZhcmlhYmxlcyBhcmUgZmFjdG9ycz8KCjQuIFVzZSB0aGUgZnVuY3Rpb25zIGBjbGFzc2AgYW5kIGBuYW1lc2AgdG8gZmluZCB0aGUgY2xhc3MgYW5kCiAgIHZhcmlhYmxlIG5hbWVzIGluIHRoZSBgZ2FwbWluZGVyYCBkYXRhLgoKNS4gVXNlIGBzdW1tYXJ5YCB0byBjb21wdXRlIHN1bW1hcnkgaW5mb3JtYXRpb24gZm9yIHRoZSB2YXJpYWJsZXMuCgo2LiBGaWxsIGluIHRoZSB2YWx1ZXMgZm9yIGAtLS1gIG5lZWRlZCB0byBwcm9kdWNlIHBsb3RzIG9mIGxpZmUKICAgZXhwZWN0YW5jeSBhZ2FpbnN0IHllYXIgZm9yIHRoZSBjb3VudHJpZXMgaW4gY29udGluZW50IE9jZWFuaWEuCjwhLS0gIyMgbm9saW50IHN0YXJ0IC0tPgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGRwbHlyKQpsaWJyYXJ5KGdncGxvdDIpCmRhdGEoZ2FwbWluZGVyLCBwYWNrYWdlID0gImdhcG1pbmRlciIpCmdncGxvdChmaWx0ZXIoZ2FwbWluZGVyLCBjb250aW5lbnQgPT0gIk9jZWFuaWEiKSwKICAgICAgIGFlcyh4ID0gLS0tLCB5ID0gLS0tLCBjb2xvciA9IGNvdW50cnkpKSArCiAgICBnZW9tX2xpbmUoKQpgYGAKPCEtLSAjIyBub2xpbnQgZW5kIC0tPgo=