Basics

Objectives

Learn how to effecitvely use visualization for

  • exploring and understanding data
  • communicating and explaining insights

Learn how to use data technologies for

  • acquiring data
  • cleaning data
  • organizing data

Learn how to do this in ways that are

  • reproducible
  • reusable
  • shareable

Topics

Data visualization

  • some history of visualization
  • learning the basic graph types
  • how to create basic graphs in R
  • human perception, and how it affects visualization
  • using understanding of perception to guide evaluation and design
  • dynamic and interactive visualizations

Data technologies

  • basic data types
  • reshaping and transforming data
  • aggregating and summarizing data
  • merging several data sets
  • regular expressions for cleaning data
  • harvesting data from the web

Reproducible research and collaboration

  • literate programming and data analysis
  • version control for collaboration

Prerequisites

An introductory statistics course.

A regression course.

Strongly recommended: Prior exposure to basic use of statistical programming software, such as R or SAS, as obtained from a regression course.

Assessment

Quizzes

  • Short quizzes will be posted on ICON after most lectures.

Homework

  • Homework assignments will be due approximately once a week.
  • You will typically submit your work by pushing it to your GitLab repository by 5:00 PM on the due date.
  • Your homework solutions should be written as reports, using proper sentences and paragraphs to present your results.

Project

  • You will do a project developing a visual analysis of a data set of your choosing.
  • You can work on your own or in a group of up to three students.
  • Your project should represent about 10 hours of work for each student.
  • A one page proposal for your project is due on Monday, March 18.
  • A final report on your project is due on Friday, May 3.
  • Your project may be shared with the class through the class web page.

Your grade will be based on quizzes (10%), homework (70%) and the project (20%).

Tools

We will be using

  • R for computing and graphics
  • R Markdown for creating reproducible reports.
  • git and the UI GitLab service for revision control and submitting work.

You will need an editor or IDE; you can use

  • RStudio for editing and more
  • any other editor or IDE

To access these tools you can

For help installing your own a good place to start is https://happygitwithr.com/

First Steps: Do This Today!

Visit the UI GitLab site at https://research-git.uiowa.edu and log in with your HawkID.

Make sure you can access the UI IDAS Rstudio Notebook Server with your HawkID and password.

Make sure you are able to log into the CLAS Linux systems with your HawkID and password.

Look at the brief introduction to git or the beginning of https://happygitwithr.com to see what git is about and how to get started with it.

Make sure you have access to R and try someting like this:

with(faithful,
     plot(eruptions, waiting,
          xlab = "Eruption time (min)",
          ylab = "Waiting time to next eruption (min)"))

The result is a plot that looks like this:

Getting Set Up

Log into the UI GitLab site at https://research-git.uiowa.edu to get your GitLab account activated.

Decide where you want to work:

Setup needed for IDAS RStudio Server:

  • If you are registered then you should have an account now. If you add the course late you should have an account within a day.
  • Introduce yourself to Git.

Setup needed for CLAS Linux:

Setting up your own computer: (A good resource for help with this is https://happygitwithr.com):

  • Install the current version of R.

    • You might have older versions from other courses (e.g. from Anaconda).
    • You will need to add packages as we go along.
  • Install RStudio if you want to use it (highly recommended).

  • Install Git.

  • Introduce yourself to Git.

Even if you decide to use your own computer you should make sure you can use the RStudio server or CLAS systems as a backup.

Some Examples

Life Expectancy in the Americas in 2007

The data is from the GapMinder project.

library(dplyr)
library(ggplot2)
library(gapminder)
le_am_2007 <- filter(gapminder, year == 2007, continent == "Americas") |>
    mutate(country = reorder(country, lifeExp))
knitr::kable(select(le_am_2007, country, lifeExp),
             col.names = c("Country", "Life Expectancy (years)"),
             digits = 1, format = "html") |>
    kableExtra::kable_styling(bootstrap_options = "striped",
                              full_width = FALSE,
                              font_size = 14) |>
    kableExtra::scroll_box(height = "300px", width = "75%")
Country Life Expectancy (years)
Argentina 75.3
Bolivia 65.6
Brazil 72.4
Canada 80.7
Chile 78.6
Colombia 72.9
Costa Rica 78.8
Cuba 78.3
Dominican Republic 72.2
Ecuador 75.0
El Salvador 71.9
Guatemala 70.3
Haiti 60.9
Honduras 70.2
Jamaica 72.6
Mexico 76.2
Nicaragua 72.9
Panama 75.5
Paraguay 71.8
Peru 71.4
Puerto Rico 78.7
Trinidad and Tobago 69.8
United States 78.2
Uruguay 76.4
Venezuela 73.7

A dot plot:

thm <- theme_minimal() + theme(text = element_text(size = 16))
ggplot(le_am_2007, aes(y = country, x = lifeExp)) +
    geom_point(fill = "lightblue") +
    labs(x = "Life Expectancy (years)", y = NULL) +
    thm + ggtitle("Dot Plot")

A bar chart:

ggplot(le_am_2007, aes(x = lifeExp, y = country)) +
    geom_col(fill = "lightblue") +
    labs(x = "Life Expectancy (years)", y = NULL) +
    thm + ggtitle("Bar Chart")

Another (bad!) bar chart:

baseline <- 60
ticks <- c(0, 10, 20, 30)
ggplot(le_am_2007, aes(x = lifeExp - baseline, y = country)) +
    geom_col(fill = "lightblue") +
    labs(x = "Life Expectancy (years)", y = NULL) +
    scale_x_continuous(breaks = ticks, labels = ticks + baseline) +
    thm + ggtitle("Another Bar Chart")

We will look at:

  • How to create these views using code that makes them easily reproducible.

  • How to assess their advantages and disadvantages as visual representations of the data

A data set with more variables for more countries and years is available in the gapminder R package.

Data preparation steps:

  • Filter the larger data set down to the countries and year we want.

  • Select the country name and life expectancy variables.

We will look at how to carry out these steps with reproducible code.

Yearly Snowfall in Iowa City

How did the winter of 2018/9 compare to other years?

The data are available from a NOAA web serice API as a CSV file.

## # A tibble: 6 × 34
##    year month element VALUE1 VALUE2 VALUE3 VALUE4 VALUE5 VALUE6 VALUE7 VALUE8
##   <int> <int> <chr>    <int>  <int>  <int>  <int>  <int>  <int>  <int>  <int>
## 1  1893     1 TMAX       -17    -28   -150    -44    -17   -106    -56    -67
## 2  1893     1 TMIN       -67   -161   -233   -156   -156   -206   -111   -211
## 3  1893     1 PRCP         0      0      0     64      0     38      0      0
## 4  1893     1 SNOW         0      0      0     64      0     38      0      0
## 5  1893     2 TMAX        11    -56   -106   -122     33     28   -161    -94
## 6  1893     2 TMIN      -233   -206   -217   -267   -122   -211   -256   -278
## # ℹ 23 more variables: VALUE9 <int>, VALUE10 <int>, VALUE11 <int>,
## #   VALUE12 <int>, VALUE13 <int>, VALUE14 <int>, VALUE15 <int>, VALUE16 <int>,
## #   VALUE17 <int>, VALUE18 <int>, VALUE19 <int>, VALUE20 <int>, VALUE21 <int>,
## #   VALUE22 <int>, VALUE23 <int>, VALUE24 <int>, VALUE25 <int>, VALUE26 <int>,
## #   VALUE27 <int>, VALUE28 <int>, VALUE29 <int>, VALUE30 <int>, VALUE31 <int>

Data preparation steps:

  • Read in the CSV file.

  • Reshape the data to have columns date, TMAX, TMIN, SNOW and PRCP.

  • Filter out bogus dates created by the original format.

  • Convert units to more standard (American) ones (e.g. milimeters to inches).

Code is available here.

Internet Adoption Across the World

An example from Wilke (2019) with World Bank data.

The data are available in several formats (CSV, XML, Excel).

## # A tibble: 6 × 65
##   Country.Name Country.Code Indicator.Name      Indicator.Code X1960 X1961 X1962
##   <chr>        <chr>        <chr>               <chr>          <int> <lgl> <lgl>
## 1 Aruba        ABW          Individuals using … IT.NET.USER.ZS    NA NA    NA   
## 2 Afghanistan  AFG          Individuals using … IT.NET.USER.ZS    NA NA    NA   
## 3 Angola       AGO          Individuals using … IT.NET.USER.ZS    NA NA    NA   
## 4 Albania      ALB          Individuals using … IT.NET.USER.ZS    NA NA    NA   
## 5 Andorra      AND          Individuals using … IT.NET.USER.ZS    NA NA    NA   
## 6 Arab World   ARB          Individuals using … IT.NET.USER.ZS    NA NA    NA   
## # ℹ 58 more variables: X1963 <lgl>, X1964 <lgl>, X1965 <int>, X1966 <lgl>,
## #   X1967 <lgl>, X1968 <lgl>, X1969 <lgl>, X1970 <int>, X1971 <lgl>,
## #   X1972 <lgl>, X1973 <lgl>, X1974 <lgl>, X1975 <int>, X1976 <int>,
## #   X1977 <int>, X1978 <int>, X1979 <int>, X1980 <int>, X1981 <int>,
## #   X1982 <int>, X1983 <int>, X1984 <int>, X1985 <int>, X1986 <int>,
## #   X1987 <int>, X1988 <int>, X1989 <int>, X1990 <dbl>, X1991 <dbl>,
## #   X1992 <dbl>, X1993 <dbl>, X1994 <dbl>, X1995 <dbl>, X1996 <dbl>, …

Data preparation:

  • Read in the data.

  • Filter down to the countries we want.

  • Reshape to have columns country, year, and users.

Code is available here.

Iowa Wind Turbines

Data is available from the U.S. Wind Turbine Database.

library(sf)
data(US_counties_geoms, package = "dviz.supp")
wtfile <- "uswtdb_v6_1_20231128.csv.gz"
if (! file.exists(wtfile))
    download.file(paste0("https://stat.uiowa.edu/~luke/data/", wtfile), wtfile)
wind_turbines <- read.csv(wtfile)

sf_wt <- st_as_sf(wind_turbines, coords = c("xlong", "ylat"), crs = 4326)
sf_wt_IA <- filter(sf_wt, t_state == "IA")
sf_wt_IA <- mutate(sf_wt_IA,
                   p_year = ifelse(p_year > 0, p_year, NA),
                   year = cut(p_year,
                              breaks = c(0, 2005, 2010, 2015, 2020, Inf),
                              labels = c("before 2005", "2005-2009",
                                         "2010-2014", "2015-2020",
                                         "2021 and later"),
                              right = FALSE))

ggplot(filter(US_counties_geoms$lower48, STATEFP == 19)) +
    geom_sf() +
    geom_sf(data = sf_wt_IA, aes(fill = year),
            shape = 21, color = "black", stroke = 0.25) +
    scale_fill_viridis_d(direction = -1, na.value = "red") +
    ggthemes::theme_map() +
    ggtitle("A Map") +
    theme(legend.background = element_rect(fill = "transparent"),
          plot.title = element_text(size = 16))

There are two data sets:

  • Shape information for drawing the map.

  • Data on individual wind turbines.

Data preparation:

  • Read in the data.

  • Match up the projection used for the map and location data.

Iowa Population in 2010

data(US_census, package = "dviz.supp")
uscounties <- mutate(US_counties_geoms$lower48,
                     FIPS = as.numeric(paste0(STATEFP, COUNTYFP)))
uscounties <- left_join(uscounties,
                        select(US_census, FIPS, pop2010),
                        "FIPS")
## old-style crs object detected; please recreate object with a recent sf::st_crs()
ggplot(filter(uscounties, STATEFP == 19)) +
    geom_sf(aes(fill = pop2010)) +
    scale_fill_viridis_c(direction = -1, trans = "log10",
                         labels = scales::comma) +
    ggtitle("A Choropleth Map") +
    ggthemes::theme_map() +
    theme(legend.background = element_rect(fill = "transparent"),
          plot.title = element_text(size = 16))

Again there are two data sets:

  • Shape data for drawing the map.

  • County population data from the 2010 census.

Data preparation:

  • Merge or join the population data with the shape data.

Reproducibility

Reproducible Reports and Analyses

Preparing a report on a data analysis project usually involves

  • reading the data

  • wrangling the data into usable form

  • visualizing, summarizing, and modeling

  • writing a report that includes your results

To make your work reproducible for someone else, or for you when the data changes, it is best to use code for the entire workflow.

R Markdown is one technology that supports this.

Tools for Reproducibility

R Markdown files contain report text along with code to produce numerical and graphical results.

Tools are available to

  • convert an R Markdown file into a PDF or HTML report;

  • extract the code used to produce the computational and graphical results.

This page was generated from the R Markdown file intro.Rmd.

You will be creating R Markdown files like this for your homework and project.

Some R Markdown tutorials:

Code

Code for the Iowa City Snowfall Example

Read the data:

library(tidyverse)
library(lubridate)
if (! file.exists(here("ic_noaa.csv.gz")))
    download.file("http://www.stat.uiowa.edu/~luke/data/ic_noaa.csv.gz",
                  here("ic_noaa.csv.gz"))
ic_data_raw <- as_tibble(read.csv(here("ic_noaa.csv.gz"), head = TRUE))

Reshape from (very) wide to (too) long:

ic_data <- select(ic_data_raw, year, month, element, starts_with("VALUE"))
ic_data <- pivot_longer(ic_data,
                        names_to = "day",
                        values_to = "value", c(VALUE1 : VALUE31))

Extract the day as a number:

ic_data <- mutate(ic_data, day = as.integer(sub("VALUE", "", day)))

Reshape from too long to tidy with one row per day, keeping only the primary variables:

corevars <- c("TMAX", "TMIN", "PRCP", "SNOW", "SNWD")
ic_data <- filter(ic_data, element %in% corevars)
ic_data <- pivot_wider(ic_data, names_from = "element", values_from = "value")

Add a date variable for plotting and to help get rid of bogus days:

ic_data <- mutate(ic_data, date = lubridate::make_date(year, month, day))
ic_data <- filter(ic_data, ! is.na(date))

Make units more standard (American):

mm2in <- function(x) x / 25.4
C2F <- function(x) 32 + 1.8 * x
ic_data <- transmute(ic_data, year, month, day, date,
                     PRCP = mm2in(PRCP / 10),
                     SNOW = mm2in(SNOW),
                     SNWD = mm2in(SNWD),
                     TMIN = C2F(TMIN / 10),
                     TMAX = C2F(TMAX / 10))

Add a Month factor with abbreviated levels:

ic_data <- mutate(ic_data,
                  Month = lubridate::month(month, label = TRUE, abbr = TRUE))

Associate January through June with the winter starting in the previous year:

ic_data <- mutate(ic_data, wyear = ifelse(month <= 6, year - 1, year))

Compute the winter totals and the total for the 2018/9 winter:

ic_snow <- group_by(ic_data, wyear) |>
    summarize(snow = sum(SNOW, na.rm = TRUE))
ic_snow_2018 <- filter(ic_snow, wyear == 2018)$snow

Create the histogram and show the 2018/9 total:

ggplot(ic_snow) +
    geom_histogram(aes(x = snow), bins = 15, fill = "grey", color = "black") +
    geom_vline(xintercept = ic_snow_2018, color = "red", size = 2) +
    scale_x_continuous(name = "Total Annual Snowfall (inches)",
                       sec.axis = dup_axis(name = NULL,
                                           breaks = ic_snow_2018,
                                           labels = "2018/9")) +
    scale_y_continuous(expand = expansion(0, 0)) +
    ggtitle("A Histogram") +
    theme_minimal() +
    theme(text = element_text(size = 16))

Code for the Internet Example

From code for the Visualizing amounts chapter in Claus Wilke’s Fundamentals of Data Visualization.

Reading in the data:

base_url <- "http://api.worldbank.org/v2/en/indicator/"
if (file.exists("internet.xls")) {
    ## if we have to go with Excel
    ## xls_url <- paste0(base_url, "IT.NET.USER.ZS?downloadformat=excel")
    ## download.file(xls_url, "internet.xls")
    internet_raw <- readxl::read_xls("internet.xls",
                                     skip = 2, .name_repair = "universal")
    names(internet_raw) <- sub("\\.\\.\\.", "X", names(internet_raw))
} else {
    csvfile <- here("internet.csv")
    if (! file.exists(csvfile)) {
        csv_url <- paste0(base_url, "IT.NET.USER.ZS?downloadformat=csv")
        download.file(csv_url, "internet.zip")
        unzip("internet.zip")
        file.rename("API_IT.NET.USER.ZS_DS2_en_csv_v2_2255007.csv",
                    csvfile)
    }

    internet_raw <- read.csv(csvfile, skip = 4)
}

Reshape to longer format:

internet <- select(internet_raw, country = Country.Name, matches("X.+"))
internet <- pivot_longer(internet, -country,
                         names_to = "year", values_to = "users")
internet <- mutate(internet, year = as.integer(sub("X", "", year)))

Select some countries to include in the plot:

country_list <- c("United States", "China", "India", "Japan", "Algeria",
                  "Brazil", "Germany", "France", "United Kingdom", "Italy",
                  "New Zealand", "Canada", "Mexico", "Chile", "Argentina",
                  "Norway", "South Africa", "Kenya", "Israel", "Iceland")

internet_short <- filter(internet, country %in% country_list)
## internet_short <- mutate(internet_short,
##                          users = ifelse(is.na(users), 0, users))

Get ordering by 2017 levels:

intr17 <- filter(internet_short, year == 2017)
levs <- arrange(intr17, users)$country

The basic plot:

p_inet <- ggplot(filter(internet_short, year > 1993),
                 aes(x = year,
                     y = factor(country, levs),
                     fill = users)) +
    geom_tile(color = "white", size = 0.25)

Adjust color palette and guide:

p_inet <- p_inet +
    scale_fill_viridis_c(
        option = "A", begin = 0.05, end = 0.98,
        limits = c(0, 100),
        name = "internet users / 100 people",
        guide = guide_colorbar(
            direction = "horizontal",
            label.position = "bottom",
            title.position = "top",
            ticks = FALSE,
            barwidth = grid::unit(3.5, "in"),
            barheight = grid::unit(0.2, "in"),
            order = 1))

Adjust x and y scales:

p_inet <- p_inet +
    scale_x_continuous(expand = c(0, 0), name = NULL) +
    scale_y_discrete(name = NULL, position = "right")

Add layer for NA values:

p_inet <- p_inet +
    ggnewscale::new_scale_fill() +
    geom_tile(data = filter(internet_short, year > 1993, is.na(users)),
              aes(fill = "NA"), color = "white") +
    scale_fill_manual(values = "grey50") +
    guides(fill = guide_legend(title = "",
                               label.position = "bottom",
                               title.position = "top",
                               keyheight = grid::unit(0.2, "in"),
                               keywidth = grid::unit(0.2, "in"),
                               order = 2))

Final plot with title and theme adjustments:

p_inet +
    ggtitle("A Heat Map",
            "Countries ordered by percent internet users in 2017.") +
    theme(text = element_text(size = 16)) +
    theme(axis.line = element_blank(),
          axis.ticks = element_blank(),
          axis.ticks.length = grid::unit(1, "pt"),
          legend.position = "top",
          legend.justification = "left",
          legend.title.align = 0.5,
          legend.title = element_text(size = 12 * 12 / 14)
          )
LS0tCnRpdGxlOiAiRGF0YSBWaXN1YWxpemF0aW9uIGFuZCBEYXRhIFRlY2hub2xvZ2llcyIKb3V0cHV0OgogIGh0bWxfZG9jdW1lbnQ6CiAgICB0b2M6IHllcwogICAgY29kZV9mb2xkaW5nOiBzaG93CiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCi0tLQoKYGBge3Igc2V0dXAsIGluY2x1ZGUgPSBGQUxTRX0Kc291cmNlKGhlcmU6OmhlcmUoInNldHVwLlIiKSkKa25pdHI6Om9wdHNfY2h1bmskc2V0KGNvbGxhcHNlID0gVFJVRSwKICAgICAgICAgICAgICAgICAgICAgIGZpZy5oZWlnaHQgPSA1LCBmaWcud2lkdGggPSA2LCBmaWcuYWxpZ24gPSAiY2VudGVyIikKb3B0aW9ucyhodG1sdG9vbHMuZGlyLnZlcnNpb24gPSBGQUxTRSkKbGlicmFyeShnZ3Bsb3QyKQpsaWJyYXJ5KHRpZHl2ZXJzZSkKbGlicmFyeShoZXJlKQp0aGVtZV9zZXQodGhlbWVfbWluaW1hbCgpICsKICAgICAgICAgIHRoZW1lKHRleHQgPSBlbGVtZW50X3RleHQoc2l6ZSA9IDE2KSkgKwogICAgICAgICAgdGhlbWUocGFuZWwuYm9yZGVyID0gZWxlbWVudF9yZWN0KGNvbG9yID0gImdyZXkzMCIsIGZpbGwgPSBOQSkpKQpzZXQuc2VlZCgxMjM0NSkKYGBgCjxsaW5rIHJlbD0ic3R5bGVzaGVldCIgaHJlZj0ic3RhdDQ1ODAuY3NzIiB0eXBlPSJ0ZXh0L2NzcyIgLz4KCgojIEJhc2ljcwoKCiMjIE9iamVjdGl2ZXMKCkxlYXJuIGhvdyB0byBlZmZlY2l0dmVseSB1c2UgdmlzdWFsaXphdGlvbiBmb3IKCiogZXhwbG9yaW5nIGFuZCB1bmRlcnN0YW5kaW5nIGRhdGEKKiBjb21tdW5pY2F0aW5nIGFuZCBleHBsYWluaW5nIGluc2lnaHRzCgpMZWFybiBob3cgdG8gdXNlIGRhdGEgdGVjaG5vbG9naWVzIGZvcgoKKiBhY3F1aXJpbmcgZGF0YQoqIGNsZWFuaW5nIGRhdGEKKiBvcmdhbml6aW5nIGRhdGEKCkxlYXJuIGhvdyB0byBkbyB0aGlzIGluIHdheXMgdGhhdCBhcmUKCiogcmVwcm9kdWNpYmxlCiogcmV1c2FibGUKKiBzaGFyZWFibGUKCgojIyBUb3BpY3MKCkRhdGEgdmlzdWFsaXphdGlvbgoKKiBzb21lIGhpc3Rvcnkgb2YgdmlzdWFsaXphdGlvbgoqIGxlYXJuaW5nIHRoZSBiYXNpYyBncmFwaCB0eXBlcwoqIGhvdyB0byBjcmVhdGUgYmFzaWMgZ3JhcGhzIGluIFIKKiBodW1hbiBwZXJjZXB0aW9uLCBhbmQgaG93IGl0IGFmZmVjdHMgdmlzdWFsaXphdGlvbgoqIHVzaW5nIHVuZGVyc3RhbmRpbmcgb2YgcGVyY2VwdGlvbiB0byBndWlkZSBldmFsdWF0aW9uIGFuZCBkZXNpZ24KKiBkeW5hbWljIGFuZCBpbnRlcmFjdGl2ZSB2aXN1YWxpemF0aW9ucwoKRGF0YSB0ZWNobm9sb2dpZXMKCiogYmFzaWMgZGF0YSB0eXBlcwoqIHJlc2hhcGluZyBhbmQgdHJhbnNmb3JtaW5nIGRhdGEKKiBhZ2dyZWdhdGluZyBhbmQgc3VtbWFyaXppbmcgZGF0YQoqIG1lcmdpbmcgc2V2ZXJhbCBkYXRhIHNldHMKKiByZWd1bGFyIGV4cHJlc3Npb25zIGZvciBjbGVhbmluZyBkYXRhCiogaGFydmVzdGluZyBkYXRhIGZyb20gdGhlIHdlYgoKUmVwcm9kdWNpYmxlIHJlc2VhcmNoIGFuZCBjb2xsYWJvcmF0aW9uCgoqIGxpdGVyYXRlIHByb2dyYW1taW5nIGFuZCBkYXRhIGFuYWx5c2lzCiogdmVyc2lvbiBjb250cm9sIGZvciBjb2xsYWJvcmF0aW9uCgo8IS0tICFbXShpbWcvUjREUy1kYXRhLXNjaWVuY2UucG5nKSAtLT4KCgojIyBSZWNvbW1lbmRlZCBUZXh0IEJvb2tzCgo+IEtpZXJhbiBIZWFseSAoMjAxOCkgW19EYXRhIFZpc3VhbGl6YXRpb246IEEgcHJhY3RpY2FsCj4gaW50cm9kdWN0aW9uX10oaHR0cHM6Ly9zb2N2aXouY28vKSwgUHJpbmNldG9uCgo+IFBhdWwgTXVycmVsbCAoMjAwOSkuIFtfSW50cm9kdWN0aW9uIHRvIERhdGEKPiBUZWNobm9sb2dpZXNfXShodHRwOi8vd3d3LnN0YXQuYXVja2xhbmQuYWMubnovfnBhdWwvSXREVC8pLCBDaGFwbWFuCj4gJiBIYWxsL0NSQy4KCj4gSGFkbGV5IFdpY2toYW0sIE1pbmUgw4dldGlua2F5YS1SdW5kZWwsIGFuZCBHYXJyZXR0IEdyb2xlbXVuZCAoMjAyMyksCj4gW19SIGZvciBEYXRhIFNjaWVuY2UgKDJuZCBFZGl0aW9uKV9dKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovKSwKPiBPJ1JlaWxseS4KCj4gQ2xhdXMgTy4gV2lsa2UgKDIwMTkpIFtfRnVuZGFtZW50YWxzIG9mIERhdGEKPiBWaXN1YWxpemF0aW9uX10oaHR0cHM6Ly9jbGF1c3dpbGtlLmNvbS9kYXRhdml6LyksIE/igJlSZWlsbHksCj4gSW5jLiAoW0Jvb2sgc291cmNlIG9uCj4gR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vY2xhdXN3aWxrZS9kYXRhdml6KTsgW3N1cHBvcnRpbmcKPiBtYXRlcmlhbHMgb24gR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vY2xhdXN3aWxrZS9kdml6LnN1cHApKQoKCiMjIFByZXJlcXVpc2l0ZXMKCkFuIGludHJvZHVjdG9yeSBzdGF0aXN0aWNzIGNvdXJzZS4KCkEgcmVncmVzc2lvbiBjb3Vyc2UuCgpTdHJvbmdseSByZWNvbW1lbmRlZDogUHJpb3IgZXhwb3N1cmUgdG8gYmFzaWMgdXNlIG9mIHN0YXRpc3RpY2FsCnByb2dyYW1taW5nIHNvZnR3YXJlLCBzdWNoIGFzIFIgb3IgU0FTLCBhcyBvYnRhaW5lZCBmcm9tIGEgcmVncmVzc2lvbgpjb3Vyc2UuCgoKIyMgQXNzZXNzbWVudAoKUXVpenplcwoKKiBTaG9ydCBxdWl6emVzIHdpbGwgYmUgcG9zdGVkIG9uICoqSUNPTioqIGFmdGVyIG1vc3QgbGVjdHVyZXMuCgpIb21ld29yawoKKiBIb21ld29yayBhc3NpZ25tZW50cyB3aWxsIGJlIGR1ZSBhcHByb3hpbWF0ZWx5IG9uY2UgYSB3ZWVrLgoqIFlvdSB3aWxsIHR5cGljYWxseSBzdWJtaXQgeW91ciB3b3JrIGJ5IHB1c2hpbmcgaXQgdG8geW91cgogIEdpdExhYiByZXBvc2l0b3J5IGJ5IDU6MDAgUE0gb24gdGhlIGR1ZSBkYXRlLgoqIFlvdXIgaG9tZXdvcmsgc29sdXRpb25zIHNob3VsZCBiZSB3cml0dGVuIGFzIHJlcG9ydHMsIHVzaW5nIHByb3BlcgogIHNlbnRlbmNlcyBhbmQgcGFyYWdyYXBocyB0byBwcmVzZW50IHlvdXIgcmVzdWx0cy4KClByb2plY3QKCjwhLS0gZGF0ZXMgYXJlIGZvciAyMDIzIC0tPgoqIFlvdSB3aWxsIGRvIGEgcHJvamVjdCBkZXZlbG9waW5nIGEgdmlzdWFsIGFuYWx5c2lzIG9mIGEgZGF0YSBzZXQKICBvZiB5b3VyIGNob29zaW5nLgoqIFlvdSBjYW4gd29yayBvbiB5b3VyIG93biBvciBpbiBhIGdyb3VwIG9mIHVwIHRvIHRocmVlIHN0dWRlbnRzLgoqIFlvdXIgcHJvamVjdCBzaG91bGQgcmVwcmVzZW50IGFib3V0IDEwIGhvdXJzIG9mIHdvcmsgZm9yIGVhY2ggc3R1ZGVudC4KKiBBIG9uZSBwYWdlIHByb3Bvc2FsIGZvciB5b3VyIHByb2plY3QgaXMgZHVlIG9uIE1vbmRheSwgTWFyY2ggMTguCiogQSBmaW5hbCByZXBvcnQgb24geW91ciBwcm9qZWN0IGlzIGR1ZSBvbiBGcmlkYXksIE1heSAzLgoqIFlvdXIgcHJvamVjdCBtYXkgYmUgc2hhcmVkIHdpdGggdGhlIGNsYXNzIHRocm91Z2ggdGhlIGNsYXNzIHdlYgogIHBhZ2UuCgpZb3VyIGdyYWRlIHdpbGwgYmUgYmFzZWQgb24gcXVpenplcyAoMTAlKSwgaG9tZXdvcmsgKDcwJSkgYW5kIHRoZQpwcm9qZWN0ICgyMCUpLgoKCiMjIFRvb2xzCgpXZSB3aWxsIGJlIHVzaW5nCgoqIFtSXShodHRwczovL3d3dy5yLXByb2plY3Qub3JnKSBmb3IgY29tcHV0aW5nIGFuZCBncmFwaGljcwoqIFtSIE1hcmtkb3duXShodHRwczovL3JtYXJrZG93bi5yc3R1ZGlvLmNvbSkgZm9yIGNyZWF0aW5nCiAgcmVwcm9kdWNpYmxlIHJlcG9ydHMuCiogW2BnaXRgXShodHRwczovL2dpdC1zY20uY29tLykgYW5kIHRoZSBbVUkKICBHaXRMYWJdKGh0dHBzOi8vcmVzZWFyY2gtZ2l0LnVpb3dhLmVkdSkgc2VydmljZSBmb3IgcmV2aXNpb24gY29udHJvbAogIGFuZCBzdWJtaXR0aW5nIHdvcmsuCgpZb3Ugd2lsbCBuZWVkIGFuIGVkaXRvciBvciBJREU7IHlvdSBjYW4gdXNlCgoqIFtSU3R1ZGlvXShodHRwczovL3Bvc2l0LmNvL3Byb2R1Y3RzL29wZW4tc291cmNlL3JzdHVkaW8vKSBmb3IgZWRpdGluZyBhbmQgbW9yZQoqIGFueSBvdGhlciBlZGl0b3Igb3IgSURFCgpUbyBbYWNjZXNzIHRoZXNlIHRvb2xzXShhY2Nlc3MuaHRtbCkgeW91IGNhbgoKKiB1c2UgdGhlIFtVSSBJREFTIFJTdHVkaW8gTm90ZWJvb2sKICBTZXJ2ZXJdKGh0dHBzOi8vbm90ZWJvb2tzLmhwYy51aW93YS5lZHUvc3ByaW5nMjAyNC1zdGF0LTQ1ODAtMDAwMS9odWIvaG9tZSksCiogdXNlIHRoZSBDTEFTIExpbnV4IHN5c3RlbXMgdmlhIHRoZSBbRmFzdFggcmVtb3RlCiAgZGVza3RvcF0oaHR0cHM6Ly9mYXN0eC5kaXZtcy51aW93YS5lZHUpLAoqIG9yIGluc3RhbGwgeW91ciBvd24gb24geW91ciBjb21wdXRlcgoKRm9yIGhlbHAgaW5zdGFsbGluZyB5b3VyIG93biBhIGdvb2QgcGxhY2UgdG8gc3RhcnQgaXMKPGh0dHBzOi8vaGFwcHlnaXR3aXRoci5jb20vPgoKCiMjIEZpcnN0IFN0ZXBzOiBEbyBUaGlzIFRvZGF5IQoKVmlzaXQgdGhlIFtVSSBHaXRMYWIgc2l0ZV0oaHR0cHM6Ly9yZXNlYXJjaC1naXQudWlvd2EuZWR1KSBhdAo8aHR0cHM6Ly9yZXNlYXJjaC1naXQudWlvd2EuZWR1PiBhbmQgbG9nIGluIHdpdGggeW91ciBIYXdrSUQuCgpNYWtlIHN1cmUgeW91IGNhbiBhY2Nlc3MgdGhlIFVJIElEQVMgUnN0dWRpbyBOb3RlYm9vayBTZXJ2ZXIgd2l0aAp5b3VyIEhhd2tJRCBhbmQgcGFzc3dvcmQuCgoqIFRoZSBzZXJ2ZXIgaXMgYXZhaWxhYmxlIGF0CgogIDxodHRwczovL25vdGVib29rcy5ocGMudWlvd2EuZWR1L3NwcmluZzIwMjQtc3RhdC00NTgwLTAwMDEvaHViL2hvbWU+LgoKKiBJZiB5b3UgY2Fubm90IGxvZyBpbnRvIHRoZSBSU3R1ZGlvIHNlcnZlciwgcGxlYXNlIGxldCB5b3VyIFRBCiAgb3IgbWUga25vdyBpbW1lZGlhdGVseS4KCk1ha2Ugc3VyZSB5b3UgYXJlIGFibGUgdG8gbG9nIGludG8gdGhlIENMQVMgTGludXggc3lzdGVtcyB3aXRoIHlvdXIKSGF3a0lEIGFuZCBwYXNzd29yZC4KCiogVGhlIGVhc2llc3Qgd2F5IGlzIHRvIHVzZSB0aGUKICBbRmFzdFhdKGh0dHBzOi8vZmFzdHguZGl2bXMudWlvd2EuZWR1KSBjbGllbnQgYXQKICA8aHR0cHM6Ly9mYXN0eC5kaXZtcy51aW93YS5lZHU+LgoqIElmIHlvdSBjYW5ub3QgbG9nIGludG8gdGhlIENMQVMgd29ya3N0YXRpb25zLCBwbGVhc2UgbGV0IHlvdXIgVEEKICBvciBtZSBrbm93IGltbWVkaWF0ZWx5LgoKTG9vayBhdCB0aGUgW2JyaWVmIGludHJvZHVjdGlvbiB0byBnaXRdKGByIFdMTksoImdpdC5odG1sIilgKSBvciB0aGUKYmVnaW5uaW5nIG9mIDxodHRwczovL2hhcHB5Z2l0d2l0aHIuY29tPiB0byBzZWUgd2hhdCBgZ2l0YCBpcyBhYm91dAphbmQgaG93IHRvIGdldCBzdGFydGVkIHdpdGggaXQuCgo8IS0tCiogV29ya2luZyB3aXRoIHRoZSBVSSBHaXRMYWIgZnJvbSBSU3R1ZGlvIG9yIHRoZSBjb21tYW5kIGxpbmUgd2lsbCB1c2UKICBbU1NIXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9TZWN1cmVfU2hlbGwpIGNvbW11bmljYXRpb24uIFlvdQogIHdpbGwgbmVlZCB0byBkbyBhIGZldyB0aGluZ3MgdG8gc2V0IHRoaXMgdXA6CiAgCiAgICAqIFNldCB1cCBhbiBTU0gga2V5IGJ5IGZvbGxvd2luZyBbdGhlc2UKICAgICAgaW5zdHJ1Y3Rpb25zXShodHRwczovL2hhcHB5Z2l0d2l0aHIuY29tL3NzaC1rZXlzLmh0bWwpLiBBbHRlcm5hdGUKICAgICAgaW5zdHVjdGlvbnMgYXJlIFthdmFpbGFibGUKICAgICAgaGVyZV0oaHR0cHM6Ly93aWtpLnVpb3dhLmVkdS9kaXNwbGF5L2dpdGh1YmRvY3MvQ3JlYXRpbmcrU1NIK0tleXMpLgogICAgICAoRm9yIExpbnV4IGZvbGxvdyB0aGUgTWFjIGluc3RydWN0aW9ucy4pCiAgICAqIFRvIHVzZSBTU0ggdG8gY29tbXVuaWNhdGUgd2l0aCBVSSBHaXRMYWIgZnJvbSBvZmYtY2FtcHVzIHlvdQogICAgICB3aWxsIG5lZWQgdG8gdXNlIGEgW1ZQTiBjb25uZWN0aW9uXShodHRwczovL2l0cy51aW93YS5lZHUvdnBuKSwKLS0+CgpNYWtlIHN1cmUgeW91IGhhdmUgYWNjZXNzIHRvIFIgYW5kIHRyeSBzb21ldGluZyBsaWtlIHRoaXM6CgpgYGB7ciBnZXlzZXIsIGV2YWwgPSBGQUxTRX0Kd2l0aChmYWl0aGZ1bCwKICAgICBwbG90KGVydXB0aW9ucywgd2FpdGluZywKICAgICAgICAgIHhsYWIgPSAiRXJ1cHRpb24gdGltZSAobWluKSIsCiAgICAgICAgICB5bGFiID0gIldhaXRpbmcgdGltZSB0byBuZXh0IGVydXB0aW9uIChtaW4pIikpCmBgYAoKVGhlIHJlc3VsdCBpcyBhIHBsb3QgdGhhdCBsb29rcyBsaWtlIHRoaXM6CgpgYGB7ciwgcmVmLmxhYmVsPSJnZXlzZXIiLCBlY2hvPUZBTFNFfQpgYGAKCgojIyBHZXR0aW5nIFNldCBVcAoKTG9nIGludG8gdGhlIFtVSSBHaXRMYWIgc2l0ZV0oaHR0cHM6Ly9yZXNlYXJjaC1naXQudWlvd2EuZWR1KSBhdAo8aHR0cHM6Ly9yZXNlYXJjaC1naXQudWlvd2EuZWR1PiB0byBnZXQgeW91ciBHaXRMYWIgYWNjb3VudAphY3RpdmF0ZWQuCgpEZWNpZGUgd2hlcmUgeW91IHdhbnQgdG8gd29yazoKCiogW1VJIElEQVMgUlN0dWRpbyBOb3RlYm9vawogIFNlcnZlcl0oaHR0cHM6Ly9ub3RlYm9va3MuaHBjLnVpb3dhLmVkdS9zcHJpbmcyMDI0LXN0YXQtNDU4MC0wMDAxL2h1Yi9ob21lKQoqIFtGYXN0WF0oaHR0cHM6Ly9saW51eC5jbGFzLnVpb3dhLmVkdS9oZWxwL2Zhc3R4KSBmb3IgYWNjZXNzaW5nCiAgdGhlIENMQVMgTGludXggc3lzdGVtcyB2aWEgdGhlCiAgW3dlYiBpbnRlcmZhY2VdKGh0dHBzOi8vZmFzdHguZGl2bXMudWlvd2EuZWR1KSBvciB0aGUgCiAgW2Rlc2t0b3AgY2xpZW50XShodHRwczovL2xpbnV4LmNsYXMudWlvd2EuZWR1L2hlbHAvZmFzdHgvZGVza3RvcCkuCiogWW91ciBvd24gY29tcHV0ZXIuCgpTZXR1cCBuZWVkZWQgZm9yIElEQVMgUlN0dWRpbyBTZXJ2ZXI6CgoqIElmIHlvdSBhcmUgcmVnaXN0ZXJlZCB0aGVuIHlvdSBzaG91bGQgaGF2ZSBhbiBhY2NvdW50CiAgbm93LiBJZiB5b3UgYWRkIHRoZSBjb3Vyc2UgbGF0ZSB5b3Ugc2hvdWxkIGhhdmUgYW4gYWNjb3VudAogIHdpdGhpbiBhIGRheS4KKiBbSW50cm9kdWNlIHlvdXJzZWxmIHRvIEdpdF0oaHR0cHM6Ly9oYXBweWdpdHdpdGhyLmNvbS9oZWxsby1naXQuaHRtbCkuCgpTZXR1cCBuZWVkZWQgZm9yIENMQVMgTGludXg6CgoqIEluc3RhbGwgdGhlIFtkZXNrdG9wCiAgIGNsaWVudF0oaHR0cHM6Ly9saW51eC5jbGFzLnVpb3dhLmVkdS9oZWxwL2Zhc3R4L2Rlc2t0b3ApIGlmIHlvdQogICB3YW50IHRvIHVzZSBpdC4gT3RoZXJ3aXNlLCB1c2UgdGhlIFt3ZWIKICAgaW50ZXJmYWNlXShodHRwczovL2Zhc3R4LmRpdm1zLnVpb3dhLmVkdSkuCiogWW91ciBhY2NvdW50IHdpbGwgYmUgc2V0IHVwIGF1dG9tYXRpY2FsbHkgdGhlIGZpcnN0IHRpbWUgeW91IGxvZyBpbi4KKiBbSW50cm9kdWNlIHlvdXJzZWxmIHRvIEdpdF0oaHR0cHM6Ly9oYXBweWdpdHdpdGhyLmNvbS9oZWxsby1naXQuaHRtbCkuCgo8IS0tCiogU2V0IHVwIGFuIFNTSCBrZXkgaWYgeW91IGRvbid0IGhhdmUgb25lIGFscmVhZHkuCiogQWRkIHlvdXIgcHVibGljIFNTSCBrZXkgdG8geW91ciBVSSBHaXRIdWIgYWNjb3VudC4KLS0+CgpTZXR0aW5nIHVwIHlvdXIgb3duIGNvbXB1dGVyOiAoQSBnb29kIHJlc291cmNlIGZvciBoZWxwIHdpdGggdGhpcyBpcwo8aHR0cHM6Ly9oYXBweWdpdHdpdGhyLmNvbT4pOgoKKiBJbnN0YWxsIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgUi4KICAgICogWW91IG1pZ2h0IGhhdmUgb2xkZXIgdmVyc2lvbnMgZnJvbSBvdGhlciBjb3Vyc2VzIChlLmcuIGZyb20KICAgICAgW0FuYWNvbmRhXShodHRwczovL3d3dy5hbmFjb25kYS5jb20vKSkuCiAgICAqIFlvdSB3aWxsIG5lZWQgdG8gYWRkIHBhY2thZ2VzIGFzIHdlIGdvIGFsb25nLgoKKiBJbnN0YWxsIFJTdHVkaW8gaWYgeW91IHdhbnQgdG8gdXNlIGl0IChoaWdobHkgcmVjb21tZW5kZWQpLgoKKiBJbnN0YWxsIEdpdC4KCiogW0ludHJvZHVjZSB5b3Vyc2VsZiB0byBHaXRdKGh0dHBzOi8vaGFwcHlnaXR3aXRoci5jb20vaGVsbG8tZ2l0Lmh0bWwpLgoKPCEtLQogICAgKiBTZXQgdXAgYW4gU1NIIGtleSBpZiB5b3UgZG9uJ3QgaGF2ZSBvbmUgYWxyZWFkeS4KICAgICogQWRkIHlvdSBwdWJsaWMgU1NIIGtleSB0byB5b3VyIFVJIEdpdEh1YiBhY2NvdW50LgotLT4KCkV2ZW4gaWYgeW91IGRlY2lkZSB0byB1c2UgeW91ciBvd24gY29tcHV0ZXIgeW91IHNob3VsZCBtYWtlIHN1cmUgeW91IGNhbiB1c2UKdGhlIFJTdHVkaW8gc2VydmVyIG9yIENMQVMgc3lzdGVtcyBhcyBhIGJhY2t1cC4KCjwhLS0KCiMjIENoZWNraW5nIGFuZCBTZXR0aW5nIFVwIGFuIFNTSCBLZXkKCiogVXNpbmcgUlN0dWRpbzogQ2hlY2svY3JlYXRlIHlvdXIga2V5IGJ5IGdvaW5nIHRvICoqVG9vbHMqKiA+ICoqR2xvYmFsKiogPgogICAgICAqKk9wdGlvbnMqKiA+ICoqR0lUL1NWTioqLgoKKiBGcm9tIGEgc2hlbGwgY29tbWFuZCBsaW5lOiB1c2UgYHNzaC1rZXlnZW5gLgoKKiBPcGVuIHRlcm1pbmFsIG9yIHNoZWxsIGFuZCBkbzoKICAgIGBgYHtiYXNoLCBldmFsID0gRkFMU0V9CmNhdCA8ZmlsZW5hbWU+LnB1YgogICAgYGBgCiAgYW5kIGNvcHkgdG8gdGhlIGNsaXBib2FyZC4KCiogT24gdGhlIFtVSSBHaXRMYWIgd2ViIHBhZ2VdKGh0dHBzOi8vcmVzZWFyY2gtZ2l0LnVpb3dhLmVkdSkgZ28gdG8KICBgPHlvdXIgaWNvbiBhdCByaWdodD5gID4gKipTZXR0aW5ncyoqID4gKipTU0ggS2V5cyoqIGFuZCBwYXN0ZQogIGluIHRoZSBwdWJsaWMga2V5LgoKKiBDaGVjayB0aGF0IHlvdSBjYW4gY2xvbmUgeW91ciByZXBvc2l0b3J5CgogICAgKiBSU3R1ZGlvOiBnbyB0byAqKkZpbGUqKiA+ICoqTmV3IFByb2plY3QqKiA+ICoqVmVyc2lvbiBDb250cm9sKiogPiAqKkdpdCoqLgogICAgKiBGcm9tIGEgc2hlbGwgY29tbWFuZCBsaW5lIHVzZSBgZ2l0IGNsb25lYAoKKiBPbiBMaW51eCBhbmQgTWFjIE9TIHlvdSBjYW4gYXZvaWQgdGhlIG5lZWQgdG8gdHlwZSB5b3VyIHBhc3MgcGhyYXNlCiAgbWFueSB0aW1lcyBieSBjYWxsaW5nIGBzc2gtYWRkYCBpbiBhIHNoZWxsLiBUaGlzIHNob3VsZCByZW1lbWJlcgogIHlvdXIgcGFzcyBwaHJhc2UgdW50aWwgeW91IGxvZyBvdXQuCgoqIEluIHByaW5jaXBsZSB0aGlzIGlzIHBvc3NpYmxlIG9uIFdpbmRvd3MsIGJ1dCB0aGUgc2V0dXAgaXMgbW9yZQpjb21wbGljYXRlZC4gIFRoaXMKW2xpbmtdKGh0dHBzOi8vaGVscC5naXRodWIuY29tL2FydGljbGVzL3dvcmtpbmctd2l0aC1zc2gta2V5LXBhc3NwaHJhc2VzLyNwbGF0Zm9ybS13aW5kb3dzKQpwcm92aWRlcyBzb21lIGhpbnRzLgotLT4KCgojIFNvbWUgRXhhbXBsZXMKCgojIyBMaWZlIEV4cGVjdGFuY3kgaW4gdGhlIEFtZXJpY2FzIGluIDIwMDcKClRoZSBkYXRhIGlzIGZyb20gdGhlIFtHYXBNaW5kZXJdKGh0dHBzOi8vd3d3LmdhcG1pbmRlci5vcmcpIHByb2plY3QuCgpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCBjbGFzcy5zb3VyY2UgPSAiZm9sZC1oaWRlIn0KbGlicmFyeShkcGx5cikKbGlicmFyeShnZ3Bsb3QyKQpsaWJyYXJ5KGdhcG1pbmRlcikKbGVfYW1fMjAwNyA8LSBmaWx0ZXIoZ2FwbWluZGVyLCB5ZWFyID09IDIwMDcsIGNvbnRpbmVudCA9PSAiQW1lcmljYXMiKSB8PgogICAgbXV0YXRlKGNvdW50cnkgPSByZW9yZGVyKGNvdW50cnksIGxpZmVFeHApKQprbml0cjo6a2FibGUoc2VsZWN0KGxlX2FtXzIwMDcsIGNvdW50cnksIGxpZmVFeHApLAogICAgICAgICAgICAgY29sLm5hbWVzID0gYygiQ291bnRyeSIsICJMaWZlIEV4cGVjdGFuY3kgKHllYXJzKSIpLAogICAgICAgICAgICAgZGlnaXRzID0gMSwgZm9ybWF0ID0gImh0bWwiKSB8PgogICAga2FibGVFeHRyYTo6a2FibGVfc3R5bGluZyhib290c3RyYXBfb3B0aW9ucyA9ICJzdHJpcGVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZnVsbF93aWR0aCA9IEZBTFNFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb250X3NpemUgPSAxNCkgfD4KICAgIGthYmxlRXh0cmE6OnNjcm9sbF9ib3goaGVpZ2h0ID0gIjMwMHB4Iiwgd2lkdGggPSAiNzUlIikKYGBgCgpBIF9kb3QgcGxvdF86CgpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCBmaWcuaGVpZ2h0ID0gNiwgZmlnLndpZHRoID0gMTAsIGNsYXNzLnNvdXJjZSA9ICJmb2xkLWhpZGUifQp0aG0gPC0gdGhlbWVfbWluaW1hbCgpICsgdGhlbWUodGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTYpKQpnZ3Bsb3QobGVfYW1fMjAwNywgYWVzKHkgPSBjb3VudHJ5LCB4ID0gbGlmZUV4cCkpICsKICAgIGdlb21fcG9pbnQoZmlsbCA9ICJsaWdodGJsdWUiKSArCiAgICBsYWJzKHggPSAiTGlmZSBFeHBlY3RhbmN5ICh5ZWFycykiLCB5ID0gTlVMTCkgKwogICAgdGhtICsgZ2d0aXRsZSgiRG90IFBsb3QiKQpgYGAKCkEgX2JhciBjaGFydF86CgpgYGB7ciwgbWVzc2FnZSA9IEZBTFNFLCBmaWcuaGVpZ2h0ID0gNiwgZmlnLndpZHRoID0gMTAsIGNsYXNzLnNvdXJjZSA9ICJmb2xkLWhpZGUifQpnZ3Bsb3QobGVfYW1fMjAwNywgYWVzKHggPSBsaWZlRXhwLCB5ID0gY291bnRyeSkpICsKICAgIGdlb21fY29sKGZpbGwgPSAibGlnaHRibHVlIikgKwogICAgbGFicyh4ID0gIkxpZmUgRXhwZWN0YW5jeSAoeWVhcnMpIiwgeSA9IE5VTEwpICsKICAgIHRobSArIGdndGl0bGUoIkJhciBDaGFydCIpCmBgYAoKQW5vdGhlciAoYmFkISkgIF9iYXIgY2hhcnRfOgoKYGBge3IsIG1lc3NhZ2UgPSBGQUxTRSwgZmlnLmhlaWdodCA9IDYsIGZpZy53aWR0aCA9IDEwLCBjbGFzcy5zb3VyY2UgPSAiZm9sZC1oaWRlIn0KYmFzZWxpbmUgPC0gNjAKdGlja3MgPC0gYygwLCAxMCwgMjAsIDMwKQpnZ3Bsb3QobGVfYW1fMjAwNywgYWVzKHggPSBsaWZlRXhwIC0gYmFzZWxpbmUsIHkgPSBjb3VudHJ5KSkgKwogICAgZ2VvbV9jb2woZmlsbCA9ICJsaWdodGJsdWUiKSArCiAgICBsYWJzKHggPSAiTGlmZSBFeHBlY3RhbmN5ICh5ZWFycykiLCB5ID0gTlVMTCkgKwogICAgc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcyA9IHRpY2tzLCBsYWJlbHMgPSB0aWNrcyArIGJhc2VsaW5lKSArCiAgICB0aG0gKyBnZ3RpdGxlKCJBbm90aGVyIEJhciBDaGFydCIpCmBgYAoKV2Ugd2lsbCBsb29rIGF0OgoKKiBIb3cgdG8gY3JlYXRlIHRoZXNlIHZpZXdzIHVzaW5nIGNvZGUgdGhhdCBtYWtlcyB0aGVtIGVhc2lseSByZXByb2R1Y2libGUuCgoqIEhvdyB0byBhc3Nlc3MgdGhlaXIgYWR2YW50YWdlcyBhbmQgZGlzYWR2YW50YWdlcyBhcyB2aXN1YWwKICByZXByZXNlbnRhdGlvbnMgb2YgdGhlIGRhdGEKCkEgZGF0YSBzZXQgd2l0aCBtb3JlIHZhcmlhYmxlcyBmb3IgbW9yZSBjb3VudHJpZXMgYW5kIHllYXJzIGlzCmF2YWlsYWJsZSBpbiB0aGUgYGdhcG1pbmRlcmAgUiBwYWNrYWdlLgoKRGF0YSBwcmVwYXJhdGlvbiBzdGVwczoKCiogX0ZpbHRlcl8gdGhlIGxhcmdlciBkYXRhIHNldCBkb3duIHRvIHRoZSBjb3VudHJpZXMgYW5kIHllYXIgd2Ugd2FudC4KCiogX1NlbGVjdF8gdGhlIGNvdW50cnkgbmFtZSBhbmQgbGlmZSBleHBlY3RhbmN5IHZhcmlhYmxlcy4KCldlIHdpbGwgbG9vayBhdCBob3cgdG8gY2Fycnkgb3V0IHRoZXNlIHN0ZXBzIHdpdGggcmVwcm9kdWNpYmxlIGNvZGUuCgoKIyMgWWVhcmx5IFNub3dmYWxsIGluIElvd2EgQ2l0eQoKPGRpdiBpZD0iaWMtc25vd2ZhbGwtZXhhbXBsZSI+PC9kaXY+CgpIb3cgZGlkIHRoZSB3aW50ZXIgb2YgMjAxOC85IGNvbXBhcmUgdG8gb3RoZXIgeWVhcnM/CgpgYGB7ciBzbm93ZmFsbCwgZWNobyA9IEZBTFNFLCBtZXNzYWdlID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRSwgZmlnLmhlaWdodCA9IDYsIGZpZy53aWR0aCA9IDEwfQpgYGAKClRoZSBkYXRhIGFyZSBhdmFpbGFibGUgZnJvbSBhIFtOT0FBIHdlYiBzZXJpY2UKQVBJXShodHRwczovL3d3dy5uY2RjLm5vYWEuZ292L2Nkby13ZWIvd2Vic2VydmljZXMvdjIpIGFzIGEKW19DU1ZfXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9Db21tYS1zZXBhcmF0ZWRfdmFsdWVzKSBmaWxlLgoKYGBge3IsIGVjaG8gPSBGQUxTRX0KaGVhZChzZWxlY3QoaWNfZGF0YV9yYXcsIHllYXIsIG1vbnRoLCBlbGVtZW50LCBzdGFydHNfd2l0aCgiVkFMVUUiKSkpCmBgYAoKRGF0YSBwcmVwYXJhdGlvbiBzdGVwczoKCiogUmVhZCBpbiB0aGUgQ1NWIGZpbGUuCgoqIF9SZXNoYXBlXyB0aGUgZGF0YSB0byBoYXZlIGNvbHVtbnMgYGRhdGVgLCBgVE1BWGAsIGBUTUlOYCwgYFNOT1dgIGFuZCBgUFJDUGAuCgoqIF9GaWx0ZXJfIG91dCBib2d1cyBkYXRlcyBjcmVhdGVkIGJ5IHRoZSBvcmlnaW5hbCBmb3JtYXQuCgoqIF9Db252ZXJ0XyB1bml0cyB0byBtb3JlIHN0YW5kYXJkIChBbWVyaWNhbikgb25lcyAoZS5nLiBtaWxpbWV0ZXJzIHRvIGluY2hlcykuCgoqIC4uLgoKQ29kZSBpcyBhdmFpbGFibGUgW2hlcmVdKCNpYy1zbm93ZmFsbC1leGFtcGxlLWNvZGUpLgoKCiMjIEludGVybmV0IEFkb3B0aW9uIEFjcm9zcyB0aGUgV29ybGQKCjxkaXYgaWQ9ImludGVybmV0LWV4YW1wbGUiPjwvZGl2PgoKQW4gZXhhbXBsZSBmcm9tIFtXaWxrZSAoMjAxOSldKGh0dHBzOi8vY2xhdXN3aWxrZS5jb20vZGF0YXZpei8pIHdpdGgKW1dvcmxkIEJhbmsKZGF0YV0oaHR0cHM6Ly9kYXRhLndvcmxkYmFuay5vcmcvaW5kaWNhdG9yL0lULk5FVC5VU0VSLlpTKS4KCmBgYHtyIGludGVybmV0LWhlYXRtYXAsIGVjaG8gPSBGQUxTRSwgZmlnLmhlaWdodCA9IDYsIGZpZy53aWR0aCA9IDEwfQpgYGAKClRoZSBkYXRhIGFyZSBhdmFpbGFibGUgaW4gc2V2ZXJhbCBmb3JtYXRzIChDU1YsIFhNTCwgRXhjZWwpLgoKYGBge3IsIGVjaG8gPSBGQUxTRX0KaGVhZChhc190aWJibGUoaW50ZXJuZXRfcmF3KSkKYGBgCgpEYXRhIHByZXBhcmF0aW9uOgoKKiBSZWFkIGluIHRoZSBkYXRhLgoKKiBfRmlsdGVyXyBkb3duIHRvIHRoZSBjb3VudHJpZXMgd2Ugd2FudC4KCiogX1Jlc2hhcGVfIHRvIGhhdmUgY29sdW1ucyBgY291bnRyeWAsIGB5ZWFyYCwgYW5kIGB1c2Vyc2AuIAoKKiAuLi4KCkNvZGUgaXMgYXZhaWxhYmxlIFtoZXJlXSgjaW50ZXJuZXQtZXhhbXBsZS1jb2RlKS4KCgojIyBJb3dhIFdpbmQgVHVyYmluZXMKCkRhdGEgaXMgYXZhaWxhYmxlIGZyb20gdGhlIFtVLlMuIFdpbmQgVHVyYmluZQpEYXRhYmFzZV0oaHR0cHM6Ly9lZXJzY21hcC51c2dzLmdvdi91c3d0ZGIvKS4KCjwhLS0KRm9yIFdpbGtlJ3MgZXhhbXBsZSB1c2UgZGF0YSh3aW5kX3R1cmJpbmVzLCBwYWNrYWdlID0gImR2aXouc3VwcCIpLgpUaGlzIHVzZXMgbW9yZSByZWNlbnQgZGF0YSBmcm9tIFVTR1MuCi0tPgoKYGBge3IsIGZpZy5oZWlnaHQgPSA2LCBmaWcud2lkdGggPSAxMCwgbWVzc2FnZSA9IEZBTFNFLCBjbGFzcy5zb3VyY2UgPSAiZm9sZC1oaWRlIn0KbGlicmFyeShzZikKZGF0YShVU19jb3VudGllc19nZW9tcywgcGFja2FnZSA9ICJkdml6LnN1cHAiKQp3dGZpbGUgPC0gInVzd3RkYl92Nl8xXzIwMjMxMTI4LmNzdi5neiIKaWYgKCEgZmlsZS5leGlzdHMod3RmaWxlKSkKICAgIGRvd25sb2FkLmZpbGUocGFzdGUwKCJodHRwczovL3N0YXQudWlvd2EuZWR1L35sdWtlL2RhdGEvIiwgd3RmaWxlKSwgd3RmaWxlKQp3aW5kX3R1cmJpbmVzIDwtIHJlYWQuY3N2KHd0ZmlsZSkKCnNmX3d0IDwtIHN0X2FzX3NmKHdpbmRfdHVyYmluZXMsIGNvb3JkcyA9IGMoInhsb25nIiwgInlsYXQiKSwgY3JzID0gNDMyNikKc2Zfd3RfSUEgPC0gZmlsdGVyKHNmX3d0LCB0X3N0YXRlID09ICJJQSIpCnNmX3d0X0lBIDwtIG11dGF0ZShzZl93dF9JQSwKICAgICAgICAgICAgICAgICAgIHBfeWVhciA9IGlmZWxzZShwX3llYXIgPiAwLCBwX3llYXIsIE5BKSwKICAgICAgICAgICAgICAgICAgIHllYXIgPSBjdXQocF95ZWFyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBjKDAsIDIwMDUsIDIwMTAsIDIwMTUsIDIwMjAsIEluZiksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVscyA9IGMoImJlZm9yZSAyMDA1IiwgIjIwMDUtMjAwOSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIjIwMTAtMjAxNCIsICIyMDE1LTIwMjAiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICIyMDIxIGFuZCBsYXRlciIpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICByaWdodCA9IEZBTFNFKSkKCmdncGxvdChmaWx0ZXIoVVNfY291bnRpZXNfZ2VvbXMkbG93ZXI0OCwgU1RBVEVGUCA9PSAxOSkpICsKICAgIGdlb21fc2YoKSArCiAgICBnZW9tX3NmKGRhdGEgPSBzZl93dF9JQSwgYWVzKGZpbGwgPSB5ZWFyKSwKICAgICAgICAgICAgc2hhcGUgPSAyMSwgY29sb3IgPSAiYmxhY2siLCBzdHJva2UgPSAwLjI1KSArCiAgICBzY2FsZV9maWxsX3ZpcmlkaXNfZChkaXJlY3Rpb24gPSAtMSwgbmEudmFsdWUgPSAicmVkIikgKwogICAgZ2d0aGVtZXM6OnRoZW1lX21hcCgpICsKICAgIGdndGl0bGUoIkEgTWFwIikgKwogICAgdGhlbWUobGVnZW5kLmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbCA9ICJ0cmFuc3BhcmVudCIpLAogICAgICAgICAgcGxvdC50aXRsZSA9IGVsZW1lbnRfdGV4dChzaXplID0gMTYpKQpgYGAKClRoZXJlIGFyZSB0d28gZGF0YSBzZXRzOgoKKiBfU2hhcGVfIGluZm9ybWF0aW9uIGZvciBkcmF3aW5nIHRoZSBtYXAuCgoqIERhdGEgb24gaW5kaXZpZHVhbCB3aW5kIHR1cmJpbmVzLgoKRGF0YSBwcmVwYXJhdGlvbjoKCiogUmVhZCBpbiB0aGUgZGF0YS4KCiogTWF0Y2ggdXAgdGhlIF9wcm9qZWN0aW9uXyB1c2VkIGZvciB0aGUgbWFwIGFuZCBsb2NhdGlvbiBkYXRhLgoKKiAuLi4KCgojIyBJb3dhIFBvcHVsYXRpb24gaW4gMjAxMAoKYGBge3IsIGluY2x1ZGUgPSBGQUxTRX0KIyMgVGhpcyBpcyB3b3VsZCBiZSBuZWVkZWQgaWYgd2UgZGlkbid0IGhhdmUgYSBsaWJyYXJ5KHNmKSBlYXJsaWVyIHNvCiMjIHRoYXQgdGhlIGZpbHRlciBvcGVyYXRpb24gZG9lcyB0aGUgcmlnaHQgdGhpbmcgSSB0aGluawojIyBsb2FkTmFtZXNwYWNlKCJzZiIpCmBgYAoKYGBge3IsIGZpZy5oZWlnaHQgPSA2LCBmaWcud2lkdGggPSAxMCwgY2xhc3Muc291cmNlID0gImZvbGQtaGlkZSJ9CmRhdGEoVVNfY2Vuc3VzLCBwYWNrYWdlID0gImR2aXouc3VwcCIpCnVzY291bnRpZXMgPC0gbXV0YXRlKFVTX2NvdW50aWVzX2dlb21zJGxvd2VyNDgsCiAgICAgICAgICAgICAgICAgICAgIEZJUFMgPSBhcy5udW1lcmljKHBhc3RlMChTVEFURUZQLCBDT1VOVFlGUCkpKQp1c2NvdW50aWVzIDwtIGxlZnRfam9pbih1c2NvdW50aWVzLAogICAgICAgICAgICAgICAgICAgICAgICBzZWxlY3QoVVNfY2Vuc3VzLCBGSVBTLCBwb3AyMDEwKSwKICAgICAgICAgICAgICAgICAgICAgICAgIkZJUFMiKQpnZ3Bsb3QoZmlsdGVyKHVzY291bnRpZXMsIFNUQVRFRlAgPT0gMTkpKSArCiAgICBnZW9tX3NmKGFlcyhmaWxsID0gcG9wMjAxMCkpICsKICAgIHNjYWxlX2ZpbGxfdmlyaWRpc19jKGRpcmVjdGlvbiA9IC0xLCB0cmFucyA9ICJsb2cxMCIsCiAgICAgICAgICAgICAgICAgICAgICAgICBsYWJlbHMgPSBzY2FsZXM6OmNvbW1hKSArCiAgICBnZ3RpdGxlKCJBIENob3JvcGxldGggTWFwIikgKwogICAgZ2d0aGVtZXM6OnRoZW1lX21hcCgpICsKICAgIHRoZW1lKGxlZ2VuZC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGwgPSAidHJhbnNwYXJlbnQiKSwKICAgICAgICAgIHBsb3QudGl0bGUgPSBlbGVtZW50X3RleHQoc2l6ZSA9IDE2KSkKYGBgCgpBZ2FpbiB0aGVyZSBhcmUgdHdvIGRhdGEgc2V0czoKCiogU2hhcGUgZGF0YSBmb3IgZHJhd2luZyB0aGUgbWFwLgoKKiBDb3VudHkgcG9wdWxhdGlvbiBkYXRhIGZyb20gdGhlIDIwMTAgY2Vuc3VzLgoKRGF0YSBwcmVwYXJhdGlvbjoKCiogLi4uCgoqIF9NZXJnZV8gb3IgX2pvaW5fIHRoZSBwb3B1bGF0aW9uIGRhdGEgd2l0aCB0aGUgc2hhcGUgZGF0YS4KCiogLi4uCgoKIyBSZXByb2R1Y2liaWxpdHkKCgojIyBSZXByb2R1Y2libGUgUmVwb3J0cyBhbmQgQW5hbHlzZXMKClByZXBhcmluZyBhIHJlcG9ydCBvbiBhIGRhdGEgYW5hbHlzaXMgcHJvamVjdCB1c3VhbGx5IGludm9sdmVzCgoqIHJlYWRpbmcgdGhlIGRhdGEKCiogX3dyYW5nbGluZ18gdGhlIGRhdGEgaW50byB1c2FibGUgZm9ybQoKKiB2aXN1YWxpemluZywgc3VtbWFyaXppbmcsIGFuZCBtb2RlbGluZwoKKiB3cml0aW5nIGEgcmVwb3J0IHRoYXQgaW5jbHVkZXMgeW91ciByZXN1bHRzCgpUbyBtYWtlIHlvdXIgd29yayByZXByb2R1Y2libGUgZm9yIHNvbWVvbmUgZWxzZSwgb3IgZm9yIHlvdSB3aGVuIHRoZQpkYXRhIGNoYW5nZXMsIGl0IGlzIGJlc3QgdG8gdXNlIGNvZGUgZm9yIHRoZSBlbnRpcmUgd29ya2Zsb3cuCgpbX1IgTWFya2Rvd25fXShodHRwczovL3JtYXJrZG93bi5yc3R1ZGlvLmNvbS8pIGlzIG9uZSB0ZWNobm9sb2d5IHRoYXQKc3VwcG9ydHMgdGhpcy4KCgojIyBUb29scyBmb3IgUmVwcm9kdWNpYmlsaXR5CgpSIE1hcmtkb3duIGZpbGVzIGNvbnRhaW4gcmVwb3J0IHRleHQgYWxvbmcgd2l0aCBjb2RlIHRvIHByb2R1Y2UKbnVtZXJpY2FsIGFuZCBncmFwaGljYWwgcmVzdWx0cy4KClRvb2xzIGFyZSBhdmFpbGFibGUgdG8KCiogY29udmVydCBhbiBSIE1hcmtkb3duIGZpbGUgaW50byBhIFBERiBvciBIVE1MIHJlcG9ydDsKCiogZXh0cmFjdCB0aGUgY29kZSB1c2VkIHRvIHByb2R1Y2UgdGhlIGNvbXB1dGF0aW9uYWwgYW5kIGdyYXBoaWNhbCByZXN1bHRzLgoKYGBge3IsIGluY2x1ZGUgPSBGQUxTRX0KZG9jd29yZHMgPC0gaWYgKHVzaW5nX3hhcmluZ2FuKSAiVGhlc2Ugc2xpZGVzIHdlcmUiIGVsc2UgIlRoaXMgcGFnZSB3YXMiCmBgYApgciBkb2N3b3Jkc2AgZ2VuZXJhdGVkIGZyb20gdGhlIFIgTWFya2Rvd24gZmlsZSBbaW50cm8uUm1kXShpbnRyby5SbWQpLgoKWW91IHdpbGwgYmUgY3JlYXRpbmcgUiBNYXJrZG93biBmaWxlcyBsaWtlIHRoaXMgZm9yIHlvdXIgaG9tZXdvcmsgYW5kCnByb2plY3QuCgpTb21lIFIgTWFya2Rvd24gdHV0b3JpYWxzOgoKKiBbUiBNYXJrZG93bjogVGhlIERlZmluaXRpdmUgR3VpZGVdKGh0dHBzOi8vYm9va2Rvd24ub3JnL3lpaHVpL3JtYXJrZG93bi8pCiAgYnkgWWlodWkgWGllIGlzIGEgYm9vay1sZW5ndGggcHJlc2VudGF0aW9uLgoKKiBUaGUgW1IgTWFya2Rvd24gSG9tZSBQYWdlXShodHRwczovL3JtYXJrZG93bi5yc3R1ZGlvLmNvbSkgaGFzIGEgbGluawogIHRvIGEgW3R1dG9yaWFsXShodHRwczovL3JtYXJrZG93bi5yc3R1ZGlvLmNvbS9sZXNzb24tMS5odG1sKS4KCgojIENvZGUKCgojIyBDb2RlIGZvciB0aGUgW0lvd2EgQ2l0eSBTbm93ZmFsbCBFeGFtcGxlXSgjaWMtc25vd2ZhbGwtZXhhbXBsZSkKCjxkaXYgaWQ9ImljLXNub3dmYWxsLWV4YW1wbGUtY29kZSI+PC9kaXY+CgpSZWFkIHRoZSBkYXRhOgoKYGBge3Igc25vd2ZhbGwtcmVhZCwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KHRpZHl2ZXJzZSkKbGlicmFyeShsdWJyaWRhdGUpCmlmICghIGZpbGUuZXhpc3RzKGhlcmUoImljX25vYWEuY3N2Lmd6IikpKQogICAgZG93bmxvYWQuZmlsZSgiaHR0cDovL3d3dy5zdGF0LnVpb3dhLmVkdS9+bHVrZS9kYXRhL2ljX25vYWEuY3N2Lmd6IiwKICAgICAgICAgICAgICAgICAgaGVyZSgiaWNfbm9hYS5jc3YuZ3oiKSkKaWNfZGF0YV9yYXcgPC0gYXNfdGliYmxlKHJlYWQuY3N2KGhlcmUoImljX25vYWEuY3N2Lmd6IiksIGhlYWQgPSBUUlVFKSkKYGBgCgpSZXNoYXBlIGZyb20gKHZlcnkpIHdpZGUgdG8gKHRvbykgbG9uZzoKCmBgYHtyIHNub3dmYWxsLXdpZGUtdG8tbG9uZywgZXZhbCA9IEZBTFNFfQppY19kYXRhIDwtIHNlbGVjdChpY19kYXRhX3JhdywgeWVhciwgbW9udGgsIGVsZW1lbnQsIHN0YXJ0c193aXRoKCJWQUxVRSIpKQppY19kYXRhIDwtIHBpdm90X2xvbmdlcihpY19kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICBuYW1lc190byA9ICJkYXkiLAogICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAidmFsdWUiLCBjKFZBTFVFMSA6IFZBTFVFMzEpKQpgYGAKCkV4dHJhY3QgdGhlIGRheSBhcyBhIG51bWJlcjoKCmBgYHtyIHNub3dmYWxsLWV4dHJhY3QtZGF5LCBldmFsID0gRkFMU0V9CmljX2RhdGEgPC0gbXV0YXRlKGljX2RhdGEsIGRheSA9IGFzLmludGVnZXIoc3ViKCJWQUxVRSIsICIiLCBkYXkpKSkKYGBgCgpSZXNoYXBlIGZyb20gdG9vIGxvbmcgdG8gdGlkeSB3aXRoIG9uZSByb3cgcGVyIGRheSwga2VlcGluZyBvbmx5IHRoZQpwcmltYXJ5IHZhcmlhYmxlczoKCmBgYHtyIHNub3dmYWxsLWxvbmctdG8tdGlkeSwgZXZhbCA9IEZBTFNFfQpjb3JldmFycyA8LSBjKCJUTUFYIiwgIlRNSU4iLCAiUFJDUCIsICJTTk9XIiwgIlNOV0QiKQppY19kYXRhIDwtIGZpbHRlcihpY19kYXRhLCBlbGVtZW50ICVpbiUgY29yZXZhcnMpCmljX2RhdGEgPC0gcGl2b3Rfd2lkZXIoaWNfZGF0YSwgbmFtZXNfZnJvbSA9ICJlbGVtZW50IiwgdmFsdWVzX2Zyb20gPSAidmFsdWUiKQpgYGAKCkFkZCBhIGRhdGUgdmFyaWFibGUgZm9yIHBsb3R0aW5nIGFuZCB0byBoZWxwIGdldCByaWQgb2YgYm9ndXMgZGF5czoKCmBgYHtyIHNub3dmYWxsLWFkZC1kYXRlLCBldmFsID0gRkFMU0V9CmljX2RhdGEgPC0gbXV0YXRlKGljX2RhdGEsIGRhdGUgPSBsdWJyaWRhdGU6Om1ha2VfZGF0ZSh5ZWFyLCBtb250aCwgZGF5KSkKaWNfZGF0YSA8LSBmaWx0ZXIoaWNfZGF0YSwgISBpcy5uYShkYXRlKSkKYGBgCgpNYWtlIHVuaXRzIG1vcmUgc3RhbmRhcmQgKEFtZXJpY2FuKToKCmBgYHtyIHNub3dmYWxsLWZpeC11bml0cywgZXZhbCA9IEZBTFNFfQptbTJpbiA8LSBmdW5jdGlvbih4KSB4IC8gMjUuNApDMkYgPC0gZnVuY3Rpb24oeCkgMzIgKyAxLjggKiB4CmljX2RhdGEgPC0gdHJhbnNtdXRlKGljX2RhdGEsIHllYXIsIG1vbnRoLCBkYXksIGRhdGUsCiAgICAgICAgICAgICAgICAgICAgIFBSQ1AgPSBtbTJpbihQUkNQIC8gMTApLAogICAgICAgICAgICAgICAgICAgICBTTk9XID0gbW0yaW4oU05PVyksCiAgICAgICAgICAgICAgICAgICAgIFNOV0QgPSBtbTJpbihTTldEKSwKICAgICAgICAgICAgICAgICAgICAgVE1JTiA9IEMyRihUTUlOIC8gMTApLAogICAgICAgICAgICAgICAgICAgICBUTUFYID0gQzJGKFRNQVggLyAxMCkpCmBgYAoKQWRkIGEgTW9udGggZmFjdG9yIHdpdGggYWJicmV2aWF0ZWQgbGV2ZWxzOgoKYGBge3Igc25vd2ZhbGwtYWRkLW1vbnRoLCBldmFsID0gRkFMU0V9CmljX2RhdGEgPC0gbXV0YXRlKGljX2RhdGEsCiAgICAgICAgICAgICAgICAgIE1vbnRoID0gbHVicmlkYXRlOjptb250aChtb250aCwgbGFiZWwgPSBUUlVFLCBhYmJyID0gVFJVRSkpCmBgYAoKQXNzb2NpYXRlIEphbnVhcnkgdGhyb3VnaCBKdW5lIHdpdGggdGhlIHdpbnRlciBzdGFydGluZyBpbiB0aGUgcHJldmlvdXMgeWVhcjoKCmBgYHtyIHNub3dmYWxsLXd5ZWFyLCBldmFsID0gRkFMU0V9CmljX2RhdGEgPC0gbXV0YXRlKGljX2RhdGEsIHd5ZWFyID0gaWZlbHNlKG1vbnRoIDw9IDYsIHllYXIgLSAxLCB5ZWFyKSkKYGBgCgpDb21wdXRlIHRoZSB3aW50ZXIgdG90YWxzIGFuZCB0aGUgdG90YWwgZm9yIHRoZSAyMDE4Lzkgd2ludGVyOgoKYGBge3Igc25vd2ZhbGwtd2ludGVyLXRvdGFscywgZXZhbCA9IEZBTFNFfQppY19zbm93IDwtIGdyb3VwX2J5KGljX2RhdGEsIHd5ZWFyKSB8PgogICAgc3VtbWFyaXplKHNub3cgPSBzdW0oU05PVywgbmEucm0gPSBUUlVFKSkKaWNfc25vd18yMDE4IDwtIGZpbHRlcihpY19zbm93LCB3eWVhciA9PSAyMDE4KSRzbm93CmBgYAoKQ3JlYXRlIHRoZSBoaXN0b2dyYW0gYW5kIHNob3cgdGhlIDIwMTgvOSB0b3RhbDoKCmBgYHtyIHNub3dmYWxsLWhpc3RvZ3JhbSwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoaWNfc25vdykgKwogICAgZ2VvbV9oaXN0b2dyYW0oYWVzKHggPSBzbm93KSwgYmlucyA9IDE1LCBmaWxsID0gImdyZXkiLCBjb2xvciA9ICJibGFjayIpICsKICAgIGdlb21fdmxpbmUoeGludGVyY2VwdCA9IGljX3Nub3dfMjAxOCwgY29sb3IgPSAicmVkIiwgc2l6ZSA9IDIpICsKICAgIHNjYWxlX3hfY29udGludW91cyhuYW1lID0gIlRvdGFsIEFubnVhbCBTbm93ZmFsbCAoaW5jaGVzKSIsCiAgICAgICAgICAgICAgICAgICAgICAgc2VjLmF4aXMgPSBkdXBfYXhpcyhuYW1lID0gTlVMTCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrcyA9IGljX3Nub3dfMjAxOCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVscyA9ICIyMDE4LzkiKSkgKwogICAgc2NhbGVfeV9jb250aW51b3VzKGV4cGFuZCA9IGV4cGFuc2lvbigwLCAwKSkgKwogICAgZ2d0aXRsZSgiQSBIaXN0b2dyYW0iKSArCiAgICB0aGVtZV9taW5pbWFsKCkgKwogICAgdGhlbWUodGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTYpKQpgYGAKCjwhLS0gIyMgbm9saW50IHN0YXJ0IC0tPgpgYGB7ciBzbm93ZmFsbCwgZXZhbCA9IEZBTFNFLCBlY2hvID0gRkFMU0V9CmxpYnJhcnkobHVicmlkYXRlKQo8PHNub3dmYWxsLXJlYWQ+PgoKPDxzbm93ZmFsbC13aWRlLXRvLWxvbmc+PgoKPDxzbm93ZmFsbC1leHRyYWN0LWRheT4+Cgo8PHNub3dmYWxsLWxvbmctdG8tdGlkeT4+Cgo8PHNub3dmYWxsLWFkZC1kYXRlPj4KCjw8c25vd2ZhbGwtZml4LXVuaXRzPj4KCjw8c25vd2ZhbGwtYWRkLW1vbnRoPj4KCjw8c25vd2ZhbGwtd3llYXI+PgoKPDxzbm93ZmFsbC13aW50ZXItdG90YWxzPj4KCjw8c25vd2ZhbGwtaGlzdG9ncmFtPj4KYGBgCjwhLS0gIyMgbm9saW50IGVuZCAtLT4KCgojIyBDb2RlIGZvciB0aGUgW0ludGVybmV0IEV4YW1wbGVdKCNpbnRlcm5ldC1leGFtcGxlKQoKPGRpdiBpZD0iaW50ZXJuZXQtZXhhbXBsZS1jb2RlIj48L2Rpdj4KCkZyb20KW2NvZGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9jbGF1c3dpbGtlL2RhdGF2aXovYmxvYi9tYXN0ZXIvdmlzdWFsaXppbmdfYW1vdW50cy5SbWQpCmZvciB0aGUgW19WaXN1YWxpemluZyBhbW91bnRzXwpjaGFwdGVyXShodHRwczovL2NsYXVzd2lsa2UuY29tL2RhdGF2aXovdmlzdWFsaXppbmctYW1vdW50cy5odG1sKSBpbgpDbGF1cyBXaWxrZSdzIFtGdW5kYW1lbnRhbHMgb2YgRGF0YQpWaXN1YWxpemF0aW9uXShodHRwczovL2NsYXVzd2lsa2UuY29tL2RhdGF2aXovaW5kZXguaHRtbCkuCgpSZWFkaW5nIGluIHRoZSBkYXRhOgoKYGBge3IgaW50ZXJuZXQtaGVhdG1hcC1yZWFkaW5nLWRhdGEsIGV2YWwgPSBGQUxTRX0KYmFzZV91cmwgPC0gImh0dHA6Ly9hcGkud29ybGRiYW5rLm9yZy92Mi9lbi9pbmRpY2F0b3IvIgppZiAoZmlsZS5leGlzdHMoImludGVybmV0LnhscyIpKSB7CiAgICAjIyBpZiB3ZSBoYXZlIHRvIGdvIHdpdGggRXhjZWwKICAgICMjIHhsc191cmwgPC0gcGFzdGUwKGJhc2VfdXJsLCAiSVQuTkVULlVTRVIuWlM/ZG93bmxvYWRmb3JtYXQ9ZXhjZWwiKQogICAgIyMgZG93bmxvYWQuZmlsZSh4bHNfdXJsLCAiaW50ZXJuZXQueGxzIikKICAgIGludGVybmV0X3JhdyA8LSByZWFkeGw6OnJlYWRfeGxzKCJpbnRlcm5ldC54bHMiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2tpcCA9IDIsIC5uYW1lX3JlcGFpciA9ICJ1bml2ZXJzYWwiKQogICAgbmFtZXMoaW50ZXJuZXRfcmF3KSA8LSBzdWIoIlxcLlxcLlxcLiIsICJYIiwgbmFtZXMoaW50ZXJuZXRfcmF3KSkKfSBlbHNlIHsKICAgIGNzdmZpbGUgPC0gaGVyZSgiaW50ZXJuZXQuY3N2IikKICAgIGlmICghIGZpbGUuZXhpc3RzKGNzdmZpbGUpKSB7CiAgICAgICAgY3N2X3VybCA8LSBwYXN0ZTAoYmFzZV91cmwsICJJVC5ORVQuVVNFUi5aUz9kb3dubG9hZGZvcm1hdD1jc3YiKQogICAgICAgIGRvd25sb2FkLmZpbGUoY3N2X3VybCwgImludGVybmV0LnppcCIpCiAgICAgICAgdW56aXAoImludGVybmV0LnppcCIpCiAgICAgICAgZmlsZS5yZW5hbWUoIkFQSV9JVC5ORVQuVVNFUi5aU19EUzJfZW5fY3N2X3YyXzIyNTUwMDcuY3N2IiwKICAgICAgICAgICAgICAgICAgICBjc3ZmaWxlKQogICAgfQoKICAgIGludGVybmV0X3JhdyA8LSByZWFkLmNzdihjc3ZmaWxlLCBza2lwID0gNCkKfQpgYGAKClJlc2hhcGUgdG8gbG9uZ2VyIGZvcm1hdDoKCmBgYHtyIGludGVybmV0LWhlYXRtYXAtcmVzaGFwZSwgZXZhbCA9IEZBTFNFfQppbnRlcm5ldCA8LSBzZWxlY3QoaW50ZXJuZXRfcmF3LCBjb3VudHJ5ID0gQ291bnRyeS5OYW1lLCBtYXRjaGVzKCJYLisiKSkKaW50ZXJuZXQgPC0gcGl2b3RfbG9uZ2VyKGludGVybmV0LCAtY291bnRyeSwKICAgICAgICAgICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInllYXIiLCB2YWx1ZXNfdG8gPSAidXNlcnMiKQppbnRlcm5ldCA8LSBtdXRhdGUoaW50ZXJuZXQsIHllYXIgPSBhcy5pbnRlZ2VyKHN1YigiWCIsICIiLCB5ZWFyKSkpCmBgYAoKU2VsZWN0IHNvbWUgY291bnRyaWVzIHRvIGluY2x1ZGUgaW4gdGhlIHBsb3Q6CgpgYGB7ciBpbnRlcm5ldC1oZWF0bWFwLXNlbGVjdC1jb3VudHJpZXMsIGV2YWwgPSBGQUxTRX0KY291bnRyeV9saXN0IDwtIGMoIlVuaXRlZCBTdGF0ZXMiLCAiQ2hpbmEiLCAiSW5kaWEiLCAiSmFwYW4iLCAiQWxnZXJpYSIsCiAgICAgICAgICAgICAgICAgICJCcmF6aWwiLCAiR2VybWFueSIsICJGcmFuY2UiLCAiVW5pdGVkIEtpbmdkb20iLCAiSXRhbHkiLAogICAgICAgICAgICAgICAgICAiTmV3IFplYWxhbmQiLCAiQ2FuYWRhIiwgIk1leGljbyIsICJDaGlsZSIsICJBcmdlbnRpbmEiLAogICAgICAgICAgICAgICAgICAiTm9yd2F5IiwgIlNvdXRoIEFmcmljYSIsICJLZW55YSIsICJJc3JhZWwiLCAiSWNlbGFuZCIpCgppbnRlcm5ldF9zaG9ydCA8LSBmaWx0ZXIoaW50ZXJuZXQsIGNvdW50cnkgJWluJSBjb3VudHJ5X2xpc3QpCiMjIGludGVybmV0X3Nob3J0IDwtIG11dGF0ZShpbnRlcm5ldF9zaG9ydCwKIyMgICAgICAgICAgICAgICAgICAgICAgICAgIHVzZXJzID0gaWZlbHNlKGlzLm5hKHVzZXJzKSwgMCwgdXNlcnMpKQpgYGAKCkdldCBvcmRlcmluZyBieSAyMDE3IGxldmVsczoKCmBgYHtyIGludGVybmV0LWhlYXRtYXAtMjAxNy1sZXZlbHMsIGV2YWwgPSBGQUxTRX0KaW50cjE3IDwtIGZpbHRlcihpbnRlcm5ldF9zaG9ydCwgeWVhciA9PSAyMDE3KQpsZXZzIDwtIGFycmFuZ2UoaW50cjE3LCB1c2VycykkY291bnRyeQpgYGAKClRoZSBiYXNpYyBwbG90OgoKYGBge3IsIGludGVybmV0LWhlYXRtYXAtYmFzaWMtcGxvdCwgZXZhbCA9IEZBTFNFfQpwX2luZXQgPC0gZ2dwbG90KGZpbHRlcihpbnRlcm5ldF9zaG9ydCwgeWVhciA+IDE5OTMpLAogICAgICAgICAgICAgICAgIGFlcyh4ID0geWVhciwKICAgICAgICAgICAgICAgICAgICAgeSA9IGZhY3Rvcihjb3VudHJ5LCBsZXZzKSwKICAgICAgICAgICAgICAgICAgICAgZmlsbCA9IHVzZXJzKSkgKwogICAgZ2VvbV90aWxlKGNvbG9yID0gIndoaXRlIiwgc2l6ZSA9IDAuMjUpCmBgYAoKQWRqdXN0IGNvbG9yIHBhbGV0dGUgYW5kIGd1aWRlOgoKYGBge3IgaW50ZXJuZXQtaGVhdG1hcC1maWxsLWNvbG9yLCBldmFsID0gRkFMU0V9CnBfaW5ldCA8LSBwX2luZXQgKwogICAgc2NhbGVfZmlsbF92aXJpZGlzX2MoCiAgICAgICAgb3B0aW9uID0gIkEiLCBiZWdpbiA9IDAuMDUsIGVuZCA9IDAuOTgsCiAgICAgICAgbGltaXRzID0gYygwLCAxMDApLAogICAgICAgIG5hbWUgPSAiaW50ZXJuZXQgdXNlcnMgLyAxMDAgcGVvcGxlIiwKICAgICAgICBndWlkZSA9IGd1aWRlX2NvbG9yYmFyKAogICAgICAgICAgICBkaXJlY3Rpb24gPSAiaG9yaXpvbnRhbCIsCiAgICAgICAgICAgIGxhYmVsLnBvc2l0aW9uID0gImJvdHRvbSIsCiAgICAgICAgICAgIHRpdGxlLnBvc2l0aW9uID0gInRvcCIsCiAgICAgICAgICAgIHRpY2tzID0gRkFMU0UsCiAgICAgICAgICAgIGJhcndpZHRoID0gZ3JpZDo6dW5pdCgzLjUsICJpbiIpLAogICAgICAgICAgICBiYXJoZWlnaHQgPSBncmlkOjp1bml0KDAuMiwgImluIiksCiAgICAgICAgICAgIG9yZGVyID0gMSkpCmBgYAoKQWRqdXN0IGB4YCBhbmQgYHlgIHNjYWxlczoKCmBgYHtyIGludGVybmV0LWhlYXRtYXAteC15LXNjYWxlcywgZXZhbCA9IEZBTFNFfQpwX2luZXQgPC0gcF9pbmV0ICsKICAgIHNjYWxlX3hfY29udGludW91cyhleHBhbmQgPSBjKDAsIDApLCBuYW1lID0gTlVMTCkgKwogICAgc2NhbGVfeV9kaXNjcmV0ZShuYW1lID0gTlVMTCwgcG9zaXRpb24gPSAicmlnaHQiKQpgYGAKCkFkZCBsYXllciBmb3IgYE5BYCB2YWx1ZXM6CgpgYGB7ciBpbnRlcm5ldC1oZWF0bWFwLU5BLWxheWVyLCBldmFsID0gRkFMU0V9CnBfaW5ldCA8LSBwX2luZXQgKwogICAgZ2duZXdzY2FsZTo6bmV3X3NjYWxlX2ZpbGwoKSArCiAgICBnZW9tX3RpbGUoZGF0YSA9IGZpbHRlcihpbnRlcm5ldF9zaG9ydCwgeWVhciA+IDE5OTMsIGlzLm5hKHVzZXJzKSksCiAgICAgICAgICAgICAgYWVzKGZpbGwgPSAiTkEiKSwgY29sb3IgPSAid2hpdGUiKSArCiAgICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSAiZ3JleTUwIikgKwogICAgZ3VpZGVzKGZpbGwgPSBndWlkZV9sZWdlbmQodGl0bGUgPSAiIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVsLnBvc2l0aW9uID0gImJvdHRvbSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aXRsZS5wb3NpdGlvbiA9ICJ0b3AiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAga2V5aGVpZ2h0ID0gZ3JpZDo6dW5pdCgwLjIsICJpbiIpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAga2V5d2lkdGggPSBncmlkOjp1bml0KDAuMiwgImluIiksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvcmRlciA9IDIpKQpgYGAKCkZpbmFsIHBsb3Qgd2l0aCB0aXRsZSBhbmQgdGhlbWUgYWRqdXN0bWVudHM6CgpgYGB7ciBpbnRlcm5ldC1oZWF0bWFwLWZpbmFscGxvdCwgZXZhbCA9IEZBTFNFfQpwX2luZXQgKwogICAgZ2d0aXRsZSgiQSBIZWF0IE1hcCIsCiAgICAgICAgICAgICJDb3VudHJpZXMgb3JkZXJlZCBieSBwZXJjZW50IGludGVybmV0IHVzZXJzIGluIDIwMTcuIikgKwogICAgdGhlbWUodGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTYpKSArCiAgICB0aGVtZShheGlzLmxpbmUgPSBlbGVtZW50X2JsYW5rKCksCiAgICAgICAgICBheGlzLnRpY2tzID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgICAgYXhpcy50aWNrcy5sZW5ndGggPSBncmlkOjp1bml0KDEsICJwdCIpLAogICAgICAgICAgbGVnZW5kLnBvc2l0aW9uID0gInRvcCIsCiAgICAgICAgICBsZWdlbmQuanVzdGlmaWNhdGlvbiA9ICJsZWZ0IiwKICAgICAgICAgIGxlZ2VuZC50aXRsZS5hbGlnbiA9IDAuNSwKICAgICAgICAgIGxlZ2VuZC50aXRsZSA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIgKiAxMiAvIDE0KQogICAgICAgICAgKQpgYGAKCmBgYHtyIGludGVybmV0LWhlYXRtYXAsIGV2YWwgPSBGQUxTRSwgZWNobyA9IEZBTFNFfQo8PGludGVybmV0LWhlYXRtYXAtcmVhZGluZy1kYXRhPj4KCjw8aW50ZXJuZXQtaGVhdG1hcC1yZXNoYXBlPj4KCjw8aW50ZXJuZXQtaGVhdG1hcC1zZWxlY3QtY291bnRyaWVzPj4KCjw8aW50ZXJuZXQtaGVhdG1hcC0yMDE3LWxldmVscz4+Cgo8PGludGVybmV0LWhlYXRtYXAtYmFzaWMtcGxvdD4+Cgo8PGludGVybmV0LWhlYXRtYXAtZmlsbC1jb2xvcj4+Cgo8PGludGVybmV0LWhlYXRtYXAteC15LXNjYWxlcz4+Cgo8PGludGVybmV0LWhlYXRtYXAtTkEtbGF5ZXI+PgoKPDxpbnRlcm5ldC1oZWF0bWFwLWZpbmFscGxvdD4+CmBgYAo=