Background

The Grammar of Graphics is a language proposed by Leland Wilkinson for describing statistical graphs.

Wilkinson, L. (2005), The Grammar of Graphics, 2nd ed., Springer.

The grammar of graphics has served as the foundation for the graphics frameworks in SPSS, Vega-Lite and several other systems.

ggplot2 represents an implementation and extension of the grammar of graphics for R.

Wickham, H. (2016), ggplot2: Elegant Graphics for Data Analysis, 2nd ed., Springer. 3rd ed. in progress.

On line documentation: https://ggplot2.tidyverse.org/reference/index.html.

Hadley Wickham, Mine Çetinkaya-Rundel, and Garrett Grolemund (2023), R for Data Science (2nd Edition), O’Reilly.

Data visualization cheatsheet

Winston Chang (2018), R Graphics Cookbook, 2nd edition, O’Reilly. (Book source on GitHub)

The idea is that any basic plot can be built out of a combination of

ggplot2 provides tools for specifying these components and adjusting their features.

Many components and features are provided by default and do not need to be specified explicitly unless the defaults are to be changed.

A Basic Template

The simplest graph needs a data set, a geom, and a mapping:

ggplot(data = <DATA>) + <GEOM>(mapping = aes(<MAPPINGS>))

The appearance of geom objects is controlled by aesthetic features.

Each geom has some required and some optional aesthetics.

For geom_point the required aesthetics are

Optional aesthetics include

geom_point is used to produce a scatter plot.

Scatter Plots Using geom_point

The mpg data set included in the ggpllot2 package includes EPA fuel economy data from 1999 to 2008 for 38 popular models of cars.

mpg
## # A tibble: 234 × 11
##    manufacturer model      displ  year   cyl trans drv     cty   hwy fl    class
##    <chr>        <chr>      <dbl> <int> <int> <chr> <chr> <int> <int> <chr> <chr>
##  1 audi         a4           1.8  1999     4 auto… f        18    29 p     comp…
##  2 audi         a4           1.8  1999     4 manu… f        21    29 p     comp…
##  3 audi         a4           2    2008     4 manu… f        20    31 p     comp…
##  4 audi         a4           2    2008     4 auto… f        21    30 p     comp…
##  5 audi         a4           2.8  1999     6 auto… f        16    26 p     comp…
##  6 audi         a4           2.8  1999     6 manu… f        18    26 p     comp…
##  7 audi         a4           3.1  2008     6 auto… f        18    27 p     comp…
##  8 audi         a4 quattro   1.8  1999     4 manu… 4        18    26 p     comp…
##  9 audi         a4 quattro   1.8  1999     4 auto… 4        16    25 p     comp…
## 10 audi         a4 quattro   2    2008     4 manu… 4        20    28 p     comp…
## # ℹ 224 more rows

A simple scatter plot:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy))

Map color to vehicle class:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy,
                   color = class))

And map shape to number of cylinders:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy,
                   color = class,
                   shape = factor(cyl)))

Perception:

Aesthetics can be mapped to a variable or set to a fixed common value.

This can be used to override default settings:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy),
               color = "blue",
               shape = 1)

Changing the size aesthetics makes shapes easier to recognize:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy,
                   color = class,
                   shape = factor(cyl)),
               size = 3)

Perception: Still too many colors; still have interference.

Available point shapes are specified by number:

Shapes 1-20 have their color set by the color aesthetic and ignore the fill aesthetic.

For shapes 21-25 the color aesthetic specifies the border color and fill specifies the interior color.

Using shape 21 with cyl mapped to the fill aesthetic:

ggplot(mutate(mpg, cyl = factor(cyl))) +
    geom_point(aes(x = displ,
                   y = hwy,
                   fill = cyl),
               shape = 21,
               size = 4)

Perception: Borders, larger symbols, fewer colors help.

Specifying a new default is very different from specifying a constant value as an aesthetic.

Constant aesthetic: Rarely what you want:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy,
                   color = "blue"))

Default: Probably what you want:

ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy),
               color = "blue")

Geometric Objects

ggplot2 provides a number of geoms:

geom_abline             geom_area               geom_bar                geom_bin_2d
geom_bin2d              geom_blank              geom_boxplot            geom_col
geom_contour            geom_contour_filled     geom_count              geom_crossbar
geom_curve              geom_density            geom_density_2d         geom_density_2d_filled
geom_density2d          geom_density2d_filled   geom_dotplot            geom_errorbar
geom_errorbarh          geom_freqpoly           geom_function           geom_hex
geom_histogram          geom_hline              geom_jitter             geom_label
geom_line               geom_linerange          geom_map                geom_path
geom_point              geom_pointrange         geom_polygon            geom_qq
geom_qq_line            geom_quantile           geom_raster             geom_rect
geom_ribbon             geom_rug                geom_segment            geom_sf
geom_sf_label           geom_sf_text            geom_smooth             geom_spoke
geom_step               geom_text               geom_tile               geom_violin
geom_vline                                                              

Additional geoms are available in packages like ggforce, ggridges, and others described on the ggplot2 extensions site.

Geoms can be added as layers to a plot.

Mappings common to all, or most, geoms can be specified in the ggplot call:

ggplot(mpg,
       aes(x = displ,
           y = hwy)) +
    geom_smooth() +
    geom_point()

Geoms can also use different data sets.

One way to highlight Europe in a plot of life expectancy against log income for 2007 is to start with a plot of the full data:

library(dplyr)
library(gapminder)
gm_2007 <- filter(gapminder, year == 2007)

(p <- ggplot(gm_2007, aes(x = gdpPercap,
                          y = lifeExp)) +
     geom_point() +
     scale_x_log10())

Then add a layer showing only Europe:

gm_2007_eu <- filter(gm_2007, continent == "Europe")

p + geom_point(data = gm_2007_eu,
               color = "red",
               size = 3)

Statistical Transformations

All geoms use a statistical transformation (stat) to convert raw data to the values to be mapped to the object’s features.

The available stats are

stat_align              stat_bin                stat_bin_2d
stat_bin_hex            stat_bin2d              stat_binhex
stat_boxplot            stat_contour            stat_contour_filled
stat_count              stat_density            stat_density_2d
stat_density_2d_filled  stat_density2d          stat_density2d_filled
stat_ecdf               stat_ellipse            stat_function
stat_identity           stat_qq                 stat_qq_line
stat_quantile           stat_sf                 stat_sf_coordinates
stat_smooth             stat_spoke              stat_sum
stat_summary            stat_summary_2d         stat_summary_bin
stat_summary_hex        stat_summary2d          stat_unique
stat_ydensity                                   

Each geom has a default stat, and each stat has a default geom.

Stats can provide computed variables that can be mapped to aesthetic features.

For stat_bin some of the computed variables are

The density variable can be accessed as after_stat(dentity).

Older approaches that also work but are now discouraged:

By default, geom_histogram uses y = after_stat(count).

ggplot(faithful) +
    geom_histogram(aes(x = eruptions),
                   binwidth = 0.25,
                   fill = "grey",
                   color = "black")

Explicitly specifying y = after_stat(count) produces the same plot:

ggplot(faithful) +
    geom_histogram(aes(x = eruptions,
                       y = after_stat(count)),
                   binwidth = 0.25,
                   fill = "grey",
                   color = "black")

Using y = after_stat(density) produces a density scaled axis.

(p <- ggplot(faithful) +
     geom_histogram(aes(x = eruptions,
                        y = after_stat(density)),
                    binwidth = 0.25,
                    fill = "grey",
                    color = "black"))

stat_function can be used to add a density curve specified as a mixture of two normal densities:

(ms <- mutate(faithful,
              type = ifelse(eruptions < 3,
                            "short",
                            "long")) |>
     group_by(type) |>
     summarize(mean = mean(eruptions),
               sd = sd(eruptions),
               n = n()) |>
     mutate(p = n / sum(n)))
## # A tibble: 2 × 5
##   type   mean    sd     n     p
##   <chr> <dbl> <dbl> <int> <dbl>
## 1 long   4.29 0.411   175 0.643
## 2 short  2.04 0.267    97 0.357
f <- function(x)
    ms$p[1] * dnorm(x, ms$mean[1], ms$sd[1]) +
        ms$p[2] * dnorm(x, ms$mean[2], ms$sd[2])

p + stat_function(fun = f, color = "red")

Position Adjustments

The available position adjustments:

position_dodge        position_dodge2       position_fill
position_identity     position_jitter       position_jitterdodge
position_nudge        position_stack        

A bar chart showing the counts for the different cut categories in the diamonds data:

ggplot(diamonds, aes(x = cut)) +
    geom_bar()

Mapping clarity to fill shows the breakdown by both cut and clarity in a stacked bar chart:

ggplot(diamonds, aes(x = cut,
                     fill = clarity)) +
    geom_bar()

The default position for bar charts is position_stack:

ggplot(diamonds, aes(x = cut,
                     fill = clarity)) +
    geom_bar(position = "stack")

position_dodge produces side-by-side bar charts:

ggplot(diamonds, aes(x = cut,
                     fill = clarity)) +
    geom_bar(position = "dodge")

position_fill rescales all bars to be equal height to help compare proportions within bars.

ggplot(diamonds, aes(x = cut,
                     fill = clarity)) +
    geom_bar(position = "fill")

Using the counts to scale the widths would produce a spine plot, a variant of a mosaic plot.

This is easiest to do with the ggmosaic package.

position_jitter can be used with geom_point to avoid overplotting or break up rounding artifacts.

Another version of the Old Faithful data available as geyser in package MASS has some rounding in the duration variable:

data(geyser, package = "MASS")

## Adjust for different meaning of `waiting` variable
geyser2 <- na.omit(mutate(geyser,
                          duration = lag(duration)))

p <- ggplot(geyser2, aes(x = duration, y = waiting))
p + geom_point()

Jittering can help break up the distracting heaping of values on durations of 2 and 4 minutes.

The default amount of jittering isn’t quite enough in this case:

p + geom_point(position = "jitter")

To jitter only horizontally and by a larger amount you can use

p + geom_point(position =
                   position_jitter(height = 0,
                                   width = 0.1))

Coordinate Systems

Coordinate system functions include

coord_cartesian  coord_equal      coord_fixed      coord_flip
coord_map        coord_munch      coord_polar      coord_quickmap
coord_radial     coord_sf         coord_trans      

The default coordinate system is coord_cartesian.

Cartesian Coordinates

coord_cartesian can be used to zoom in on a particular regiion:

p + geom_point() +
    coord_cartesian(xlim = c(3, 4))

coord_fixed and coord_equal fix the aspect ratio for a cartesian coordinate system.

The aspect ratio is the ratio of the number physical display units per y unit to the number of physical display units per x unit.

The aspect ratio can be important for recognizing features and patterns.

river <- scan("https://www.stat.uiowa.edu/~luke/data/river.dat")
r <- data.frame(flow = river, month = seq_along(river))
ggplot(r, aes(x = month, y = flow)) +
    geom_point() +
    coord_fixed(ratio = 4)

Polar Coordinates

A filled bar chart

(p <- ggplot(diamonds) +
     geom_bar(aes(x = 1, fill = cut),
              position = "fill"))

is turned into a pie chart by changing to polar coordinates:

p + coord_polar(theta = "y")

Coordinate Systems for Maps

Coordinate systems are particularly important for maps.

Polygons for many political and geographic boundaries are available through the map_data function.

Boundaries for the lower 48 US states can be obtained as

usa <- map_data("state")

Polygon vertices are encoded by longitude and latitude.

Plotting these in the default cartesian coordinate system usually does not work well:

usa <- map_data("state")
m <- ggplot(usa, aes(x = long,
                     y = lat,
                     group = group)) +
    geom_polygon(fill = "white",
                 color = "black")
m

Using a fixed aspect ratio is better, but an aspect ratio of 1 does not work well:

m + coord_equal()

The problem is that away from the equator a one degree change in latitude corresponds to a larger distance than a one degree change in longitude.

The ratio of one degree longitude separation to one degree latitude separation for the latitude at the middle of Iowa of 41 degrees is

longlat <- cos(41 / 90 * pi / 2)
longlat
## [1] 0.7547096

A better map is obtained using the aspect ratio 1 / longlat:

m + coord_fixed(1 / longlat)

The best approach is to use a coordinate system designed specifically for maps.

There are many projections used in map making.

The default projection used by coord_map is the Mercator projection.

m + coord_map()

Proper map projections are non-linear; this is easier to see with an Albers projection:

m + coord_map("albers", 20, 50)

Scales

Scales are used for controlling the mapping of values to physical representations such as colors, shapes, and positions.

Scale functions are also responsible for producing guides for translating physical representations back to values, such as

There are currently 131 scale functions; some examples are

scale_color_gradient      scale_shape_manual     scale_x_log10
scale_color_manual        scale_size_area        scale_y_log10
scale_fill_gradient                              scale_x_sqrt
scale_fill_manual                                scale_y_sqrt

An experimental tool to help choosing scales has recently been introduced.

Start with a basic scatter plot:

(p <- ggplot(mpg, aes(x = displ,
                      y = hwy)) +
     geom_point())

Remove the x tick marks and labels (this can also be done with theme settings):

p + scale_x_continuous(labels = NULL,
                       breaks = NULL)

Change the tick locations and labels:

p + scale_x_continuous(labels =
                           paste(c(2, 4, 6), "ltr"),
                       breaks = c(2, 4, 6))

Use a logarithmic axis:

p + scale_x_log10(labels = paste(c(2, 4, 6), "ltr"),
                  breaks = c(2, 4, 6),
                  minor_breaks = c(3, 5, 7))

The Scales section in R for Data Science provides some more details.

Color assignment can also be controlled by scale functions.

For example, for some presidential approval ratings data

pr_appr
##         pres appr party year
## 1      Obama   79     D 2009
## 2     Carter   78     D 1977
## 3    Clinton   68     D 1993
## 4  G.W. Bush   65     R 2001
## 5     Reagan   58     R 1981
## 6 G.H.W Bush   56     R 1989
## 7      Trump   40     R 2017

the default color scale is not ideal:

ggplot(pr_appr,
       aes(x = appr, y = pres, fill = party)) +
    geom_col()

The common assignment of red for Republican and blue for Democrat can be obtained by

ggplot(pr_appr,
       aes(x = appr, y = pres, fill = party)) +
    geom_col() +
    scale_fill_manual(values
                      = c(R = "red", D = "blue"))

A better choice is to use a well-designed color palette:

ggplot(pr_appr,
       aes(x = appr, y = pres, fill = party)) +
    geom_col() +
    colorspace::scale_fill_discrete_diverging(
                    palette = "Blue-Red 2")

Facets

Faceting uses the small multiples approach to introduce additional variables.

For a single variable facet_wrap is usually used:

p <- ggplot(mpg) +
    geom_point(aes(x = displ,
                   y = hwy))
p + facet_wrap(~ class)

For two variables, each with a modest number of categories, facet_grid can be effective:

p + facet_grid(factor(cyl) ~ drv)

To show common data in all facets make sure the data does not contain the faceting variable.

This was used to show muted views of the full data in faceted plots.

A faceted plot of the gapminder data:

library(gapminder)

years_to_keep <- c(1977, 1987, 1997, 2007)
gd <- filter(gapminder,
             year %in% years_to_keep)

ggplot(gd,
       aes(x = gdpPercap,
           y = lifeExp,
           color = continent)) +
    geom_point(size = 2.5) +
    scale_x_log10() +
    facet_wrap(~ year)

Add a muted version of the full data in the background of each panel:

library(gapminder)

years_to_keep <- c(1977, 1987, 1997, 2007)
gd <- filter(gapminder,
             year %in% years_to_keep)
gd_no_year <- mutate(gd, year = NULL)

ggplot(gd,
       aes(x = gdpPercap,
           y = lifeExp,
           color = continent)) +
    geom_point(data = gd_no_year,
               color = "grey80") +
    geom_point(size = 2.5) +
    scale_x_log10() +
    facet_wrap(~ year)

Usually facets use common axis scales, but one or both can be allowed to vary.

A useful approach for showing time series data with a good aspect ratio can be to split the data into facets for non-overlapping portions of the time axis.

pd <- rep(paste(seq(1, by = 32, length.out = 4),
                seq(32, by = 32, length.out = 4),
                sep = " - "),
         each =  32)
rd <- data.frame(month = seq_along(river),
                 flow = river,
                 panel = pd)
ggplot(rd, aes(x = month,
               y = flow)) +
    geom_point() +
    facet_wrap(~ panel,
               scale = "free_x",
               ncol = 1)

Facet arrangement can also be used to convey other information, such as geographic location.

The geofacet package allows facets to be placed in approximate locations of different geographic regions.

An example for data from US states:

library(geofacet)
ggplot(state_unemp, aes(year, rate)) +
    geom_line() +
    facet_geo(~ state,
              grid = "us_state_grid2",
              label = "code") +
    scale_x_continuous(labels =
                           function(x) paste0("'", substr(x, 3, 4))) +
    labs(title = "Seasonally Adjusted US Unemployment Rate 2000-2016",
         caption = "Data Source: bls.gov",
         x = "Year",
         y = "Unemployment Rate (%)") +
    theme(strip.text.x = element_text(size = 6),
          axis.text = element_text(size = 5))

Arrangement according to a calendar can also be useful.

Themes

ggplot2 supports the notion of themes for adjusting non-data appearance aspects of a plot, such as

Theme elements can be customized in several ways:

The full documentation of the theme function lists many customizable elements.

One simple example:

ggplot(mutate(mpg, cyl = factor(cyl))) +
    geom_point(aes(x = displ,
                   y = hwy,
                   fill = cyl),
               shape = 21,
               size = 3) +
    theme(legend.position = "top",
          axis.text = element_text(size = 12),
          axis.title = element_text(size = 14,
                                    face = "bold"))

Another example:

gthm <-
    theme(plot.background =
              element_rect(fill = "lightblue",
                           color = NA),
          panel.background =
              element_rect(fill = "pink"))
p + gthm

Some alternate complete themes provided by ggplot2 are

theme_bw        theme_gray      theme_minimal   theme_void
theme_classic   theme_grey      theme_dark      theme_light

Some examples:

p_bw <- p + theme_bw() + ggtitle("BW")

p_classic <- p + theme_classic() + ggtitle("Classic")

p_min <- p + theme_minimal() + ggtitle("Minimal")

p_void <- p + theme_void() + ggtitle("Void")

library(patchwork)
(p_bw + p_classic) / (p_min + p_void)

The ggthemes package provides some additional themes.

Some examples:

library(ggthemes)

p_econ <- p + theme_economist() + ggtitle("Economist")

p_wsj <- p + theme_wsj() + ggtitle("WSJ")

p_tufte <- p + theme_tufte() + ggtitle("Tufte")

p_few <- p + theme_few() + ggtitle("Few")

(p_econ + p_wsj) / (p_tufte + p_few)

ggthemes also provides theme_map that removes unnecessary elements from maps:

m + coord_map() + theme_map()

The Themes section in R for Data Science provides some more details.

A More Complete Template

ggplot(data = <DATA>) +
    <GEOM>(mapping = aes(<MAPPINGS>),
           stat = <STAT>,
           position = <POSITION>) +
    < ... MORE GEOMS ... > +
    <COORDINATE_ADJUSTMENT> +
    <SCALE_ADJUSTMENT> +
    <FACETING> +
    <THEME_ADJUSTMENT>

Labels and Annotations

A basic plot:

p <- ggplot(mpg, aes(x = displ,
                     y = hwy))
p1 <- p + geom_point(aes(color = factor(cyl)),
                     size = 2.5)
p1

Axis labels are based on the expressions given to aes.

This is convenient for exploration but usually not ideal for a report.

The labs() function can be used to change axis and legend labels:

p1 + labs(x = "Displacement (Liters)",
          y = "Highway Miles Per Gallon",
          color = "Cylinders")

The labs() function can also add a title, subtitle, and caption:

p2 <- p1 +
    labs(x = "Displacement (Liters)",
         y = "Highway Miles Per Gallon",
         color = "Cylinders",
         title = "Gas Mileage and Displacement",
         subtitle = paste("For models which had a new release every year",
                           "between 1999 and 2008"),
         caption = "Data Source: https://fueleconomy.gov/")
p2

Annotations can be used to provide popout that draws a viewer’s attention to particular features.

The annotate() function is one option:

p2 +
    annotate("label", x = 2.8, y = 43,
             label = "Volkswagens") +
    annotate("rect",
             xmin = 1.7, xmax = 2.1,
             ymin = 40, ymax = 45,
             fill = NA, color = "black")

Often more convenient are some geom_mark objects provided by the ggforce package:

library(ggforce)
p2 +
    geom_mark_hull(aes(filter = class == "2seater"),
                   description =
                       paste("2-Seaters have high displacement",
                             "values, but also high fuel efficiency",
                             "for their displacement.")) +
    geom_mark_rect(aes(filter = hwy > 40),
                   description =
                       "These are Volkswagens") +
    geom_mark_circle(aes(filter = hwy == 12),
                     description =
                         "Three pickups and an SUV.")

These annotations can be customized in a number of ways.

Arranging Plots

There are several tools available for assembling ensemble plots.

The patchwork package is a good choice.

A simple example:

p1 <- ggplot(mpg, aes(x = displ,
                      y = hwy)) +
    geom_point()
p2 <- ggplot(mpg, aes(x = cyl,
                      y = hwy,
                      group = cyl)) +
    geom_boxplot()
p3 <- ggplot(mpg, aes(x = cyl)) +
    geom_bar()

library(patchwork)
(p1 + p2) / p3

Animation

The gganimate package can be used to add animation to a ggplot graph.

Start with a plot p for all years in the gapminder data, with year in the background:

p <- gapminder |>
    arrange(desc(pop)) |>
    ggplot(aes(x = gdpPercap, y = lifeExp)) +
    geom_text(aes(x = 5000, y = 55, label = as.character(year)),
              size = 50, color = "grey",
              hjust = "center", vjust = "center") +
    geom_point(aes(size = pop, fill = continent), shape = 21) +
    scale_x_log10(labels = scales::comma) +
    ylim(c(20, 85)) +
    scale_size_area(max_size = 20,
                    labels = scales::comma,
                    breaks = c(0.25 * 10 ^ 9, 0.5 * 10 ^ 9, 10 ^ 9)) +
    scale_fill_manual(values = c(Africa = "deepskyblue",
                                 Asia = "red",
                                 Americas = "green",
                                 Europe = "gold",
                                 Oceania = "brown")) +
    labs(x = "Income", y = "Life expectancy") +
    theme(text = element_text(size = 16)) +
    guides(fill = guide_legend(title = "Continent",
                               override.aes = list(size = 5),
                               order = 1),
           size = guide_legend(title = "Population",
                               label.hjust = 1,
                               order = 2)) +
    theme_minimal() +
        theme(panel.border = element_rect(fill = NA, color = "grey20"))

A GIF animation:

library(gganimate)
animate(p +
        transition_states(
            year,
            transition_length = 2,
            state_length = 0))

A movie:

animate(p +
        transition_states(
            year,
            transition_length = 2,
            state_length = 0,
            wrap = FALSE),
        renderer = ffmpeg_renderer())

Interaction

Plotly

The ggplotly function in the plotly package can be used to add some interactive features to a plot created with ggplot2.

  • In an R session a call to ggplotly() opens may open a browser window with the interactive plot.

  • In an RStudio session the plot appears in the graphics panel.

  • In an Rmarkdown document the interactive plot is embedded in the html file.

Another interactive plotting approach that can be used from R is described in an Infoworld article.

A simple example using ggplotly():

library(ggplot2)
library(plotly)
p <- ggplot(mutate(mpg, cyl = factor(cyl))) +
    geom_point(aes(x = displ,
                   y = hwy,
                   fill = cyl),
               shape = 21,
               size = 3)
ggplotly(p)

Adding a text aesthetic allows the tooltip display to be customized:

p <- ggplot(mutate(mpg, cyl = factor(cyl))) +
    geom_point(aes(x = displ,
                   y = hwy,
                   fill = cyl,
                   text = paste(year,
                                manufacturer,
                                model)),
               shape = 21,
               size = 3)
ggplotly(p, tooltip = "text") |>
    style(hoverlabel = list(bgcolor = "white"))

Ggiraph

The ggiraph package provides another approach.

library(ggplot2)
library(ggiraph)
p <- ggplot(mutate(mpg, cyl = factor(cyl))) +
    geom_point_interactive(
        aes(x = displ,
            y = hwy,
            fill = cyl,
            tooltip = paste(year,
                            manufacturer,
                            model)),
        shape = 21,
        size = 3)
girafe(ggobj = p)

Grammar of Interactive Graphics

There have been several efforts to develop a grammar of interactive graphics, including ggvis and animint; neither seems to be under active development at this time.

A promising approach is Vega-Lite, with a Python interface Altair and an R interface altair to the Python interface.

An example using the altair package:

rub <- read.csv(here::here("rubber.csv"))

library(altair)

chartTH <- alt$Chart(rub)$
    mark_point()$
    encode(x = alt$X("H:Q", scale = alt$Scale(domain = range(rub$H))),
           y = alt$Y("T:Q", scale = alt$Scale(domain = range(rub$T))))

brush <- alt$selection_interval()

chartTH_brush <- chartTH$add_selection(brush)

chartTH_selection <-
    chartTH_brush$encode(color = alt$condition(brush,
                                               "Origin:N",
                                               alt$value("lightgray")))

chartAT <- chartTH_selection$
    encode(x = alt$X("T:Q", scale = alt$Scale(domain = range(rub$T))),
           y = alt$Y("A:Q", scale = alt$Scale(domain = range(rub$A))))

chartAT | chartTH_selection

The resulting linked plots:

Notes

Reading

Chapters Data visualization and Graphics for communication in R for Data Science, O’Reilly.

Chapter Make a plot in Data Visualization.

Chapter ggplot2 in Introduction to Data Science: Data Analysis and Prediction Algorithms with R.

Interactive Tutorial

An interactive learnr tutorial for these notes is available.

You can run the tutorial with

STAT4580::runTutorial("ggplot")

You can install the current version of the STAT4580 package with

remotes::install_gitlab("luke-tierney/STAT4580")

You may need to install the remotes package from CRAN first.

Exercises

  1. In the following expression, which value of the shape aesthetic produces a plot with points represented as triangles outlined in black colored according to the number of cylinders?
```r
library(ggplot2)
ggplot(mpg, aes(x = displ, y = hwy, fill = factor(cyl))) +
    geom_point(size = 4, shape = ---)
```
a. 15
b. 17
c. 21
d. 24
  1. It can sometimes be useful to plot text labels in a scatterplot instead of points. Consider the plot set up as

    library(ggplot2)
    library(dplyr)
    data(gapminder, package = "gapminder")
    p <- filter(gapminder, year == 2007) |>
        group_by(continent) |>
        summarize(gdpPercap = mean(gdpPercap), lifeExp = mean(lifeExp)) |>
        ggplot(aes(x = gdpPercap, y = lifeExp))

    Which of the following produces a plot with continent names on white rectangles?

    1. p + geom_text(aes(label = continent))
    2. p + geom_label(aes(label = continent))
    3. p + geom_label(label = continent)
    4. p + geom_text(text = continent)
  2. The following code plots a kernel density estimate for the eruptions variable in the faithful data set:

    library(ggplot2)
    ggplot(faithful, aes(x = eruptions)) + geom_density(bw = 0.1)

    Look at the help page for geom_density. Which of the following best describes what specifying a value for bw does:

    1. Changes the kernel used to construct the estimate.
    2. Changes the smoothing bandwidth to make the result more or less smooth.
    3. Changes the stat used to stat_bw.
    4. Has no effect on the retult.
  3. This code creates a map of Iowa counties.

    library(ggplot2)
    p <- ggplot(map_data("county", "iowa"),
                aes(x = long, y = lat, group = group)) +
        geom_polygon(, fill = "White", color = "black")

    Which of these produces a plot with an aspect ratio that best matches the map on this page?

    1. p + coord_fixed(0.5)
    2. p + coord_fixed(0.75)
    3. p + coord_fixed(1.35)
    4. p + coord_fixed(1.95)
  4. Consider the two plots created by this code (print the values of p1 and p2 to see the plots):

    library(ggplot2)
    data(gapminder, package = "gapminder")
    p1 <- ggplot(gapminder, aes(x = log(gdpPercap), y = lifeExp)) +
        geom_point() +
        scale_x_continuous(name = "")
    p2 <- ggplot(gapminder, aes(x = gdpPercap, y = lifeExp)) +
        geom_point() +
        scale_x_log10(labels = scales::comma, name = "") 

    Which of these statements is true?

    1. The x axis labels are identical in both plots.
    2. The x axis labels in p2 are in dollars; the labels in p1 are in log dollars.
    3. The x axis labels in p1 are in dollars; the labels in p2 are in log dollars.
    4. There are no labels on the x axis in p2.
  5. Consider the plot created by

    library(ggplot2)
    data(gapminder, package = "gapminder")
    p <- ggplot(gapminder, aes(x = gdpPercap, y = lifeExp)) +
        geom_point() +
        scale_x_log10(labels = scales::comma) 

    Which of these expressions produces a plot with a white background?

    1. p
    2. p + theme_grey()
    3. p + theme_classic()
    4. p + ggthemes::theme_economist()
  6. There are many different ways to change the x axis label in ggplot. Consider the plot created by

    library(ggplot2)
    p <- ggplot(mpg, aes(x = displ, y = hwy)) +
        geom_point()

    Which of the following does not change the x axis label to Displacement?

    1. p + labs(x = "Displacement")
    2. p + scale_x_continuous("Displacement")
    3. p + xlab("Displacement")
    4. p + theme(axis.title.x = "Displacement")
LS0tCnRpdGxlOiAiVGhlIEdyYW1tYXIgb2YgR3JhcGhpY3MiCm91dHB1dDoKICBodG1sX2RvY3VtZW50OgogICAgdG9jOiB5ZXMKICAgIGNvZGVfZm9sZGluZzogc2hvdwogICAgY29kZV9kb3dubG9hZDogdHJ1ZQotLS0KCjxsaW5rIHJlbD0ic3R5bGVzaGVldCIgaHJlZj0ic3RhdDQ1ODAuY3NzIiB0eXBlPSJ0ZXh0L2NzcyIgLz4KCmBgYHtyIHNldHVwLCBpbmNsdWRlID0gRkFMU0V9CnNvdXJjZShoZXJlOjpoZXJlKCJzZXR1cC5SIikpCmtuaXRyOjpvcHRzX2NodW5rJHNldChjb2xsYXBzZSA9IFRSVUUsIG1lc3NhZ2UgPSBGQUxTRSwKICAgICAgICAgICAgICAgICAgICAgIGZpZy5oZWlnaHQgPSA1LCBmaWcud2lkdGggPSA2LCBmaWcuYWxpZ24gPSAiY2VudGVyIikKCmxpYnJhcnkoZHBseXIpCmxpYnJhcnkoZ2dwbG90MikKbGlicmFyeShsYXR0aWNlKQpsaWJyYXJ5KGdyaWRFeHRyYSkKc2V0LnNlZWQoMTIzNDUpCmBgYAoKCiMjIEJhY2tncm91bmQKClRoZSBfR3JhbW1hciBvZiBHcmFwaGljc18gaXMgYSBsYW5ndWFnZSBwcm9wb3NlZCBieSBMZWxhbmQgV2lsa2luc29uCmZvciBkZXNjcmliaW5nIHN0YXRpc3RpY2FsIGdyYXBocy4KCj4gV2lsa2luc29uLCBMLiAoMjAwNSksIF9UaGUgR3JhbW1hciBvZiBHcmFwaGljc18sIDJuZCBlZC4sIFNwcmluZ2VyLgoKVGhlIGdyYW1tYXIgb2YgZ3JhcGhpY3MgaGFzIHNlcnZlZCBhcyB0aGUgZm91bmRhdGlvbiBmb3IgdGhlIGdyYXBoaWNzCmZyYW1ld29ya3MgaW4gW1NQU1NdKGh0dHBzOi8vd3d3LmlibS5jb20vcHJvZHVjdHMvc3Bzcy1zdGF0aXN0aWNzKSwKW1ZlZ2EtTGl0ZV0oaHR0cHM6Ly92ZWdhLmdpdGh1Yi5pby92ZWdhLWxpdGUvKSBhbmQgc2V2ZXJhbCBvdGhlcgpzeXN0ZW1zLgoKYGdncGxvdDJgIHJlcHJlc2VudHMgYW4gaW1wbGVtZW50YXRpb24gYW5kIGV4dGVuc2lvbiBvZiB0aGUgZ3JhbW1hcgpvZiBncmFwaGljcyBmb3IgUi4KCj4gV2lja2hhbSwgSC4gKDIwMTYpLCBfZ2dwbG90MjogRWxlZ2FudCBHcmFwaGljcyBmb3IgRGF0YSBBbmFseXNpc18sCj4gMm5kIGVkLiwgU3ByaW5nZXIuIFszcmQgZWQuIGluIHByb2dyZXNzXShodHRwczovL2dncGxvdDItYm9vay5vcmcvKS4KCj4gT24gbGluZSBkb2N1bWVudGF0aW9uOiA8aHR0cHM6Ly9nZ3Bsb3QyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL2luZGV4Lmh0bWw+LgoKPiBIYWRsZXkgV2lja2hhbSwgTWluZSDDh2V0aW5rYXlhLVJ1bmRlbCwgYW5kIEdhcnJldHQgR3JvbGVtdW5kICgyMDIzKSwKPiBbX1IgZm9yIERhdGEgU2NpZW5jZSAoMm5kIEVkaXRpb24pX10oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei8pLAo+IE8nUmVpbGx5LgoKPiBbRGF0YSB2aXN1YWxpemF0aW9uIGNoZWF0c2hlZXRdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9yc3R1ZGlvL2NoZWF0c2hlZXRzL21haW4vZGF0YS12aXN1YWxpemF0aW9uLnBkZikKCj4gV2luc3RvbiBDaGFuZyAoMjAxOCksIFtfUiBHcmFwaGljcyBDb29rYm9va18sIDJuZAo+IGVkaXRpb25dKGh0dHBzOi8vci1ncmFwaGljcy5vcmcvKSwgT+KAmVJlaWxseS4gKFtCb29rIHNvdXJjZSBvbgo+IEdpdEh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL3djaC9yZ2Nvb2tib29rKSkKClRoZSBpZGVhIGlzIHRoYXQgYW55IGJhc2ljIHBsb3QgY2FuIGJlIGJ1aWx0IG91dCBvZiBhIGNvbWJpbmF0aW9uIG9mCgoqIGEgZGF0YSBzZXQ7CgoqIG9uZSBvciBtb3JlIGdlb21ldHJpY2FsIHJlcHJlc2VudGF0aW9uIChfZ2VvbXNfKTsKCiogbWFwcGluZ3Mgb2YgdmFsdWVzIHRvIF9hZXN0aGV0aWNfIGZlYXR1cmVzIG9mIHRoZSBnZW9tOwoKKiBhIF9zdGF0XyB0byBwcm9kdWNlIHZhbHVlcyB0byBiZSBtYXBwZWQ7CgoqIHBvc2l0aW9uIGFkanVzdG1lbnRzOwoKKiBhIGNvb3JkaW5hdGUgc3lzdGVtOwoKKiBhIHNjYWxlIHNwZWNpZmljYXRpb247CgoqIGEgZmFjZXRpbmcgc2NoZW1lLgoKYGdncGxvdDJgIHByb3ZpZGVzIHRvb2xzIGZvciBzcGVjaWZ5aW5nIHRoZXNlIGNvbXBvbmVudHMgYW5kIGFkanVzdGluZwp0aGVpciBmZWF0dXJlcy4KCk1hbnkgY29tcG9uZW50cyBhbmQgZmVhdHVyZXMgYXJlIHByb3ZpZGVkIGJ5IGRlZmF1bHQgYW5kIGRvIG5vdCBuZWVkCnRvIGJlIHNwZWNpZmllZCBleHBsaWNpdGx5IHVubGVzcyB0aGUgZGVmYXVsdHMgYXJlIHRvIGJlIGNoYW5nZWQuCgoKIyMgQSBCYXNpYyBUZW1wbGF0ZQoKVGhlIHNpbXBsZXN0IGdyYXBoIG5lZWRzIGEgZGF0YSBzZXQsIGEgZ2VvbSwgYW5kIGEgbWFwcGluZzoKCmBgYHIKZ2dwbG90KGRhdGEgPSA8REFUQT4pICsgPEdFT00+KG1hcHBpbmcgPSBhZXMoPE1BUFBJTkdTPikpCmBgYAoKVGhlIGFwcGVhcmFuY2Ugb2YgZ2VvbSBvYmplY3RzIGlzIGNvbnRyb2xsZWQgYnkgX2Flc3RoZXRpY18gZmVhdHVyZXMuCgpFYWNoIGdlb20gaGFzIHNvbWUgcmVxdWlyZWQgYW5kIHNvbWUgb3B0aW9uYWwgYWVzdGhldGljcy4KCkZvciBgZ2VvbV9wb2ludGAgdGhlIHJlcXVpcmVkIGFlc3RoZXRpY3MgYXJlCgoqIGB4YCBwb3NpdGlvbgoqIGB5YCBwb3NpdGlvbi4KCk9wdGlvbmFsIGFlc3RoZXRpY3MgaW5jbHVkZQoKKiBgY29sb3JgCiogYGZpbGxgCiogYHNoYXBlYAoqIGBzaXplYAoKYGdlb21fcG9pbnRgIGlzIHVzZWQgdG8gcHJvZHVjZSBhIF9zY2F0dGVyIHBsb3RfLgoKCiMjIFNjYXR0ZXIgUGxvdHMgVXNpbmcgYGdlb21fcG9pbnRgCgpUaGUgYG1wZ2AgZGF0YSBzZXQgaW5jbHVkZWQgaW4gdGhlIGBnZ3BsbG90MmAgcGFja2FnZSBpbmNsdWRlcyBFUEEKZnVlbCBlY29ub215IGRhdGEgZnJvbSAxOTk5IHRvIDIwMDggZm9yIDM4IHBvcHVsYXIgbW9kZWxzIG9mIGNhcnMuCgpgYGB7cn0KbXBnCmBgYAoKYGBge3IsIGluY2x1ZGUgPSBGQUxTRX0KZmlnX2FsaWduIDwtIGlmICh1c2luZ194YXJpbmdhbikgImxlZnQiIGVsc2UgImNlbnRlciIKYGBgCgpBIHNpbXBsZSBzY2F0dGVyIHBsb3Q6CgpgYGB7ciBtcGctcGxhaW4sIGV2YWwgPSBGQUxTRX0KZ2dwbG90KG1wZykgKwogICAgZ2VvbV9wb2ludChhZXMoeCA9IGRpc3BsLAogICAgICAgICAgICAgICAgICAgeSA9IGh3eSkpCmBgYAoKYGBge3IgbXBnLXBsYWluLCBlY2hvID0gRkFMU0UsIGZpZy53aWR0aCA9IDUuNzUsIGZpZy5hbGlnbiA9IGZpZ19hbGlnbn0KYGBgCgpNYXAgY29sb3IgdG8gdmVoaWNsZSBjbGFzczoKCmBgYHtyIG1wZy1jb2xvciwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QobXBnKSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICB5ID0gaHd5LAogICAgICAgICAgICAgICAgICAgY29sb3IgPSBjbGFzcykpCmBgYAoKYGBge3IgbXBnLWNvbG9yLCBlY2hvID0gRkFMU0UsIGZpZy53aWR0aCA9IDcsIGZpZy5hbGlnbiA9IGZpZ19hbGlnbn0KYGBgCgpBbmQgbWFwIHNoYXBlIHRvIG51bWJlciBvZiBjeWxpbmRlcnM6CgpgYGB7ciBtcGctY29sb3Itc2hhcGUsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KG1wZykgKwogICAgZ2VvbV9wb2ludChhZXMoeCA9IGRpc3BsLAogICAgICAgICAgICAgICAgICAgeSA9IGh3eSwKICAgICAgICAgICAgICAgICAgIGNvbG9yID0gY2xhc3MsCiAgICAgICAgICAgICAgICAgICBzaGFwZSA9IGZhY3RvcihjeWwpKSkKYGBgCgpgYGB7ciBtcGctY29sb3Itc2hhcGUsIGVjaG8gPSBGQUxTRSwgZmlnLndpZHRoID0gNywgZmlnLmFsaWduID0gZmlnX2FsaWdufQpgYGAKCjwhLS0gLS0+IFBlcmNlcHRpb246CgoqIFRvbyBtYW55IGNvbG9yczsKKiBzaGFwZXMgYXJlIHRvbyBzbWFsbDsKKiBpbnRlcmZlcmVuY2UgYmV0d2VlbiBzaGFwZXMgYW5kIGNvbG9ycy4KCkFlc3RoZXRpY3MgY2FuIGJlIG1hcHBlZCB0byBhIHZhcmlhYmxlIG9yIHNldCB0byBhIGZpeGVkIGNvbW1vbiB2YWx1ZS4KClRoaXMgY2FuIGJlIHVzZWQgdG8gb3ZlcnJpZGUgZGVmYXVsdCBzZXR0aW5nczoKCmBgYHtyIG1wZy1maXhlZCwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QobXBnKSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICB5ID0gaHd5KSwKICAgICAgICAgICAgICAgY29sb3IgPSAiYmx1ZSIsCiAgICAgICAgICAgICAgIHNoYXBlID0gMSkKYGBgCgpgYGB7ciBtcGctZml4ZWQsIGVjaG8gPSBGQUxTRSwgZmlnLndpZHRoID0gNywgZmlnLmFsaWduID0gZmlnX2FsaWdufQpgYGAKCkNoYW5naW5nIHRoZSBgc2l6ZWAgYWVzdGhldGljcyBtYWtlcyBzaGFwZXMgZWFzaWVyIHRvIHJlY29nbml6ZToKCmBgYHtyIG1wZy1jb2xvci1zaGFwZS1sYXJnZSwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QobXBnKSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICB5ID0gaHd5LAogICAgICAgICAgICAgICAgICAgY29sb3IgPSBjbGFzcywKICAgICAgICAgICAgICAgICAgIHNoYXBlID0gZmFjdG9yKGN5bCkpLAogICAgICAgICAgICAgICBzaXplID0gMykKYGBgCgpgYGB7ciBtcGctY29sb3Itc2hhcGUtbGFyZ2UsIGVjaG8gPSBGQUxTRSwgZmlnLndpZHRoID0gNywgZmlnLmFsaWduID0gZmlnX2FsaWdufQpgYGAKCjwhLS0gLS0+IFBlcmNlcHRpb246IFN0aWxsIHRvbyBtYW55IGNvbG9yczsgc3RpbGwgaGF2ZSBpbnRlcmZlcmVuY2UuCgpBdmFpbGFibGUgcG9pbnQgc2hhcGVzIGFyZSBzcGVjaWZpZWQgYnkgbnVtYmVyOgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgZXZhbCA9IEZBTFNFfQpnZW5lcmF0ZVJQb2ludFNoYXBlcyA8LSBmdW5jdGlvbigpIHsKICAgIG9sZFBhciA8LSBwYXIoKQogICAgcGFyKGZvbnQgPSAyLCBtYXIgPSBjKDAuNSwgMCwgMCwgMCkpCiAgICB5IDwtIHJldihjKHJlcCgxLCA2KSwgcmVwKDIsIDUpLCByZXAoMywgNSksIHJlcCg0LCA1KSwgcmVwKDUsIDUpKSkKICAgIHggPC0gYyhyZXAoMSA6IDUsIDUpLCA2KQogICAgcGxvdCh4LCB5LCBwY2ggPSAwIDogMjUsIGNleCA9IDEuNSwgeWxpbSA9IGMoMSwgNS41KSwgeGxpbSA9IGMoMSwgNi41KSwKICAgICAgICAgYXhlcyA9IEZBTFNFLCB4bGFiID0gIiIsIHlsYWIgPSAiIiwgYmcgPSAiYmx1ZSIpCiAgICB0ZXh0KHgsIHksIGxhYmVscyA9IDAgOiAyNSwgcG9zID0gMykKICAgIHBhcihtYXIgPSBvbGRQYXIkbWFyLCBmb250ID0gb2xkUGFyJGZvbnQpCn0KZ2VuZXJhdGVSUG9pbnRTaGFwZXMoKQpgYGAKYGBge3IsIGVjaG8gPSBGQUxTRX0KZ2dwbG90KE5VTEwsIGFlcyh4ID0gcmVwKDEgOiA1LCA1KSwgeSA9IHJldihyZXAoMSA6IDUsIGVhY2ggPSA1KSkpKSArCiAgICBnZW9tX3BvaW50KHNoYXBlID0gMSA6IDI1LCBzaXplID0gNSwgZmlsbCA9ICJibHVlIikgKwogICAgZ2VvbV90ZXh0KGFlcyhsYWJlbCA9IDEgOiAyNSksIG51ZGdlX3kgPSAwLjI1LCBzaXplID0gNikgKwogICAgdGhlbWVfdm9pZCgpCmBgYAoKU2hhcGVzIDEtMjAgaGF2ZSB0aGVpciBjb2xvciBzZXQgYnkgdGhlIGBjb2xvcmAgYWVzdGhldGljIGFuZCBpZ25vcmUKdGhlIGBmaWxsYCBhZXN0aGV0aWMuCgpGb3Igc2hhcGVzIDIxLTI1IHRoZSBgY29sb3JgIGFlc3RoZXRpYyBzcGVjaWZpZXMgdGhlIGJvcmRlciBjb2xvciBhbmQKYGZpbGxgIHNwZWNpZmllcyB0aGUgaW50ZXJpb3IgY29sb3IuCgpVc2luZyBgc2hhcGVgIDIxIHdpdGggYGN5bGAgbWFwcGVkIHRvIHRoZSBgZmlsbGAgYWVzdGhldGljOgoKYGBge3IgbXBnLWZpbGwtMjEsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KG11dGF0ZShtcGcsIGN5bCA9IGZhY3RvcihjeWwpKSkgKwogICAgZ2VvbV9wb2ludChhZXMoeCA9IGRpc3BsLAogICAgICAgICAgICAgICAgICAgeSA9IGh3eSwKICAgICAgICAgICAgICAgICAgIGZpbGwgPSBjeWwpLAogICAgICAgICAgICAgICBzaGFwZSA9IDIxLAogICAgICAgICAgICAgICBzaXplID0gNCkKYGBgCmBgYHtyIG1wZy1maWxsLTIxLCBlY2hvID0gRkFMU0V9CmBgYAoKPCEtLSAtLT4gUGVyY2VwdGlvbjogQm9yZGVycywgbGFyZ2VyIHN5bWJvbHMsIGZld2VyIGNvbG9ycyBoZWxwLgoKU3BlY2lmeWluZyBhIG5ldyBkZWZhdWx0IGlzIHZlcnkgZGlmZmVyZW50IGZyb20gc3BlY2lmeWluZyBhIGNvbnN0YW50CnZhbHVlIGFzIGFuIGFlc3RoZXRpYy4KCkNvbnN0YW50IGFlc3RoZXRpYzogUmFyZWx5IHdoYXQgeW91IHdhbnQ6CgpgYGB7ciBtcGctYmFkLWNvbG9yLCBldmFsID0gRkFMU0V9CmdncGxvdChtcGcpICsKICAgIGdlb21fcG9pbnQoYWVzKHggPSBkaXNwbCwKICAgICAgICAgICAgICAgICAgIHkgPSBod3ksCiAgICAgICAgICAgICAgICAgICBjb2xvciA9ICJibHVlIikpCmBgYAoKYGBge3IgbXBnLWJhZC1jb2xvciwgZWNobyA9IEZBTFNFLCBmaWcuaGVpZ2h0ID0gNC4yfQpgYGAKCkRlZmF1bHQ6IFByb2JhYmx5IHdoYXQgeW91IHdhbnQ6CmBgYHtyIG1wZy1nb29kLWNvbG9yLCBldmFsID0gRkFMU0V9CmdncGxvdChtcGcpICsKICAgIGdlb21fcG9pbnQoYWVzKHggPSBkaXNwbCwKICAgICAgICAgICAgICAgICAgIHkgPSBod3kpLAogICAgICAgICAgICAgICBjb2xvciA9ICJibHVlIikKYGBgCgpgYGB7ciBtcGctZ29vZC1jb2xvciwgZWNobyA9IEZBTFNFLCBmaWcuaGVpZ2h0ID0gNC4yfQpgYGAKCgojIyBHZW9tZXRyaWMgT2JqZWN0cwoKYGdncGxvdDJgIHByb3ZpZGVzIGEgbnVtYmVyIG9mIGdlb21zOgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgcmVzdWx0cyA9ICJhc2lzIn0Kc2hvd0xpc3QgPC0gZnVuY3Rpb24odiwgbmNvbCA9IDQsIHBhZCA9IDIpIHsKICAgIHcgPC0gbWF4KG5jaGFyKHYpKSArIHBhZAogICAgbnJvdyA8LSBjZWlsaW5nKGxlbmd0aCh2KSAvIG5jb2wpCiAgICB2IDwtIGModiwgY2hhcmFjdGVyKG5jb2wgKiBucm93IC0gbGVuZ3RoKHYpKSkKCiAgICBjYXQoImBgYHJcbiIpCiAgICBmb3IgKGkgaW4gc2VxX2xlbihucm93KSkgewogICAgICAgIGxpbmUgPC0gdltuY29sICogKGkgLSAxKSArICgxIDogbmNvbCldCiAgICAgICAgZm9yIChqIGluIDEgOiBuY29sKQogICAgICAgICAgICBpZiAoaiA8IG5jb2wpCiAgICAgICAgICAgICAgICBjYXQoc3ByaW50ZigiJS0qcyIsIHcsIGxpbmVbal0pKQogICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICBjYXQoc3ByaW50ZigiJXNcbiIsIGxpbmVbal0pKQogICAgICAgICMjIGNhdChzcHJpbnRmKCIlLSpzJS0qcyUtKnMlc1xuIiwKICAgICAgICAjIyAgICAgICAgICAgICB3LCBsaW5lWzFdLCB3LCBsaW5lWzJdLCB3LCBsaW5lWzNdLCBsaW5lWzRdKSkKICAgIH0KICAgIGNhdCgiYGBgXG4iKQp9CnNob3dMaXN0KGxzKCJwYWNrYWdlOmdncGxvdDIiLCBwYXQgPSAiXmdlb21fIikpCmBgYAoKQWRkaXRpb25hbCBnZW9tcyBhcmUgYXZhaWxhYmxlIGluIHBhY2thZ2VzIGxpa2UgYGdnZm9yY2VgLCBgZ2dyaWRnZXNgLAphbmQgb3RoZXJzIGRlc2NyaWJlZCBvbiB0aGUgW2BnZ3Bsb3QyYCBleHRlbnNpb25zCnNpdGVdKGh0dHBzOi8vZXh0cy5nZ3Bsb3QyLnRpZHl2ZXJzZS5vcmcvKS4KCkdlb21zIGNhbiBiZSBhZGRlZCBhcyBfbGF5ZXJzXyB0byBhIHBsb3QuCgpNYXBwaW5ncyBjb21tb24gdG8gYWxsLCBvciBtb3N0LCBnZW9tcyBjYW4gYmUgc3BlY2lmaWVkIGluIHRoZQpgZ2dwbG90YCBjYWxsOgoKYGBge3IgbXBnLXNtb290aCwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QobXBnLAogICAgICAgYWVzKHggPSBkaXNwbCwKICAgICAgICAgICB5ID0gaHd5KSkgKwogICAgZ2VvbV9zbW9vdGgoKSArCiAgICBnZW9tX3BvaW50KCkKYGBgCgpgYGB7ciBtcGctc21vb3RoLCBlY2hvID0gRkFMU0UsIG1lc3NhZ2UgPSBGQUxTRX0KYGBgCgpHZW9tcyBjYW4gYWxzbyB1c2UgZGlmZmVyZW50IGRhdGEgc2V0cy4KCk9uZSB3YXkgdG8gaGlnaGxpZ2h0IEV1cm9wZSBpbiBhIHBsb3Qgb2YgbGlmZSBleHBlY3RhbmN5IGFnYWluc3QgbG9nCmluY29tZSBmb3IgMjAwNyBpcyB0byBzdGFydCB3aXRoIGEgcGxvdCBvZiB0aGUgZnVsbCBkYXRhOgoKYGBge3IgZ21fMjAwNywgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGRwbHlyKQpsaWJyYXJ5KGdhcG1pbmRlcikKZ21fMjAwNyA8LSBmaWx0ZXIoZ2FwbWluZGVyLCB5ZWFyID09IDIwMDcpCgoocCA8LSBnZ3Bsb3QoZ21fMjAwNywgYWVzKHggPSBnZHBQZXJjYXAsCiAgICAgICAgICAgICAgICAgICAgICAgICAgeSA9IGxpZmVFeHApKSArCiAgICAgZ2VvbV9wb2ludCgpICsKICAgICBzY2FsZV94X2xvZzEwKCkpCmBgYAoKYGBge3IgZ21fMjAwNywgZWNobyA9IEZBTFNFfQpgYGAKClRoZW4gYWRkIGEgbGF5ZXIgc2hvd2luZyBvbmx5IEV1cm9wZToKCmBgYHtyIGdtXzIwMDdfZXUsIGV2YWwgPSBGQUxTRX0KZ21fMjAwN19ldSA8LSBmaWx0ZXIoZ21fMjAwNywgY29udGluZW50ID09ICJFdXJvcGUiKQoKcCArIGdlb21fcG9pbnQoZGF0YSA9IGdtXzIwMDdfZXUsCiAgICAgICAgICAgICAgIGNvbG9yID0gInJlZCIsCiAgICAgICAgICAgICAgIHNpemUgPSAzKQpgYGAKCmBgYHtyIGdtXzIwMDdfZXUsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgU3RhdGlzdGljYWwgVHJhbnNmb3JtYXRpb25zCgpBbGwgZ2VvbXMgdXNlIGEgc3RhdGlzdGljYWwgdHJhbnNmb3JtYXRpb24gKF9zdGF0XykgdG8gY29udmVydCByYXcKZGF0YSB0byB0aGUgdmFsdWVzIHRvIGJlIG1hcHBlZCB0byB0aGUgb2JqZWN0J3MgZmVhdHVyZXMuCgpUaGUgYXZhaWxhYmxlIHN0YXRzIGFyZQoKYGBge3IsIGVjaG8gPSBGQUxTRSwgcmVzdWx0cyA9ICJhc2lzIn0Kc2hvd0xpc3QobHMoInBhY2thZ2U6Z2dwbG90MiIsIHBhdCA9ICJec3RhdF8iKSwgbmNvbCA9IDMpCmBgYAoKRWFjaCBnZW9tIGhhcyBhIGRlZmF1bHQgc3RhdCwgYW5kIGVhY2ggc3RhdCBoYXMgYSBkZWZhdWx0IGdlb20uCgoqIEZvciBgZ2VvbV9wb2ludGAgdGhlIGRlZmF1bHQgc3RhdCBpcyBgc3RhdF9pZGVudGl0eWAuCgoqIEZvciBgZ2VvbV9iYXJgIHRoZSBkZWZhdWx0IHN0YXQgaXMgYHN0YXRfY291bnRgLgoKKiBGb3IgYGdlb21faGlzdG9ncmFtYCB0aGUgZGVmYXVsdCBzdGF0IGlzIGBzdGF0X2JpbmAuCgpTdGF0cyBjYW4gcHJvdmlkZSBfY29tcHV0ZWQgdmFyaWFibGVzXyB0aGF0IGNhbiBiZSBtYXBwZWQgdG8gYWVzdGhldGljCmZlYXR1cmVzLgoKRm9yIGBzdGF0X2JpbmAgc29tZSBvZiB0aGUgY29tcHV0ZWQgdmFyaWFibGVzIGFyZQoKKiBgY291bnRgOiBudW1iZXIgb2YgcG9pbnRzIGluIGJpbgoqIGBkZW5zaXR5YDogZGVuc2l0eSBvZiBwb2ludHMgaW4gYmluLCBzY2FsZWQgdG8gaW50ZWdyYXRlIHRvIDEKClRoZSBgZGVuc2l0eWAgdmFyaWFibGUgY2FuIGJlIGFjY2Vzc2VkIGFzIGBhZnRlcl9zdGF0KGRlbnRpdHkpYC4KCk9sZGVyIGFwcHJvYWNoZXMgdGhhdCBhbHNvIHdvcmsgYnV0IGFyZSBub3cgZGlzY291cmFnZWQ6CgoqIGBzdGF0KGRlbnRpdHkpYAoqIGAuLmRlbnNpdHkuLmAKCkJ5IGRlZmF1bHQsIGBnZW9tX2hpc3RvZ3JhbWAgdXNlcyBgeSA9IGFmdGVyX3N0YXQoY291bnQpYC4KCmBgYHtyIGdleXNlci1jb3VudCwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZmFpdGhmdWwpICsKICAgIGdlb21faGlzdG9ncmFtKGFlcyh4ID0gZXJ1cHRpb25zKSwKICAgICAgICAgICAgICAgICAgIGJpbndpZHRoID0gMC4yNSwKICAgICAgICAgICAgICAgICAgIGZpbGwgPSAiZ3JleSIsCiAgICAgICAgICAgICAgICAgICBjb2xvciA9ICJibGFjayIpCmBgYApgYGB7ciBnZXlzZXItY291bnQsIGVjaG8gPSBGQUxTRX0KYGBgCgpFeHBsaWNpdGx5IHNwZWNpZnlpbmcgYHkgPSBhZnRlcl9zdGF0KGNvdW50KWAgcHJvZHVjZXMgdGhlIHNhbWUgcGxvdDoKCmBgYHtyIGdleXNlci1jb3VudC1leHAsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KGZhaXRoZnVsKSArCiAgICBnZW9tX2hpc3RvZ3JhbShhZXMoeCA9IGVydXB0aW9ucywKICAgICAgICAgICAgICAgICAgICAgICB5ID0gYWZ0ZXJfc3RhdChjb3VudCkpLAogICAgICAgICAgICAgICAgICAgYmlud2lkdGggPSAwLjI1LAogICAgICAgICAgICAgICAgICAgZmlsbCA9ICJncmV5IiwKICAgICAgICAgICAgICAgICAgIGNvbG9yID0gImJsYWNrIikKYGBgCmBgYHtyIGdleXNlci1jb3VudC1leHAsIGVjaG8gPSBGQUxTRX0KYGBgCgpVc2luZyBgeSA9IGFmdGVyX3N0YXQoZGVuc2l0eSlgIHByb2R1Y2VzIGEgZGVuc2l0eSBzY2FsZWQgYXhpcy4KCmBgYHtyIGdleXNlci1kZW50aXR5LCBldmFsID0gRkFMU0V9CihwIDwtIGdncGxvdChmYWl0aGZ1bCkgKwogICAgIGdlb21faGlzdG9ncmFtKGFlcyh4ID0gZXJ1cHRpb25zLAogICAgICAgICAgICAgICAgICAgICAgICB5ID0gYWZ0ZXJfc3RhdChkZW5zaXR5KSksCiAgICAgICAgICAgICAgICAgICAgYmlud2lkdGggPSAwLjI1LAogICAgICAgICAgICAgICAgICAgIGZpbGwgPSAiZ3JleSIsCiAgICAgICAgICAgICAgICAgICAgY29sb3IgPSAiYmxhY2siKSkKYGBgCmBgYHtyIGdleXNlci1kZW50aXR5LCBlY2hvID0gRkFMU0V9CmBgYAoKYHN0YXRfZnVuY3Rpb25gIGNhbiBiZSB1c2VkIHRvIGFkZCBhIGRlbnNpdHkgY3VydmUgc3BlY2lmaWVkIGFzIGEKbWl4dHVyZSBvZiB0d28gbm9ybWFsIGRlbnNpdGllczoKCmBgYHtyfQoobXMgPC0gbXV0YXRlKGZhaXRoZnVsLAogICAgICAgICAgICAgIHR5cGUgPSBpZmVsc2UoZXJ1cHRpb25zIDwgMywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICJzaG9ydCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAibG9uZyIpKSB8PgogICAgIGdyb3VwX2J5KHR5cGUpIHw+CiAgICAgc3VtbWFyaXplKG1lYW4gPSBtZWFuKGVydXB0aW9ucyksCiAgICAgICAgICAgICAgIHNkID0gc2QoZXJ1cHRpb25zKSwKICAgICAgICAgICAgICAgbiA9IG4oKSkgfD4KICAgICBtdXRhdGUocCA9IG4gLyBzdW0obikpKQpgYGAKCmBgYHtyIGdleXNlci1oaXN0LWRlbnMsIGV2YWwgPSBGQUxTRX0KZiA8LSBmdW5jdGlvbih4KQogICAgbXMkcFsxXSAqIGRub3JtKHgsIG1zJG1lYW5bMV0sIG1zJHNkWzFdKSArCiAgICAgICAgbXMkcFsyXSAqIGRub3JtKHgsIG1zJG1lYW5bMl0sIG1zJHNkWzJdKQoKcCArIHN0YXRfZnVuY3Rpb24oZnVuID0gZiwgY29sb3IgPSAicmVkIikKYGBgCmBgYHtyIGdleXNlci1oaXN0LWRlbnMsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgUG9zaXRpb24gQWRqdXN0bWVudHMKClRoZSBhdmFpbGFibGUgcG9zaXRpb24gYWRqdXN0bWVudHM6CgpgYGB7ciwgZWNobyA9IEZBTFNFLCByZXN1bHRzID0gImFzaXMifQpzaG93TGlzdChscygicGFja2FnZTpnZ3Bsb3QyIiwgcGF0ID0gIl5wb3NpdGlvbl8iKSwgbmNvbCA9IDMpCmBgYAoKQSBiYXIgY2hhcnQgc2hvd2luZyB0aGUgY291bnRzIGZvciB0aGUgZGlmZmVyZW50IGBjdXRgIGNhdGVnb3JpZXMgaW4KdGhlIGBkaWFtb25kc2AgZGF0YToKCmBgYHtyIGRpYW1vbmRzLWN1dCwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZGlhbW9uZHMsIGFlcyh4ID0gY3V0KSkgKwogICAgZ2VvbV9iYXIoKQpgYGAKYGBge3IgZGlhbW9uZHMtY3V0LCBlY2hvID0gRkFMU0V9CmBgYAoKTWFwcGluZyBgY2xhcml0eWAgdG8gYGZpbGxgIHNob3dzIHRoZSBicmVha2Rvd24gYnkgYm90aCBgY3V0YCBhbmQKYGNsYXJpdHlgIGluIGEgX3N0YWNrZWQgYmFyIGNoYXJ0XzoKCmBgYHtyIGRpYW1vbmRzLXN0YWNrMSwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZGlhbW9uZHMsIGFlcyh4ID0gY3V0LAogICAgICAgICAgICAgICAgICAgICBmaWxsID0gY2xhcml0eSkpICsKICAgIGdlb21fYmFyKCkKYGBgCmBgYHtyIGRpYW1vbmRzLXN0YWNrMSwgZWNobyA9IEZBTFNFfQpgYGAKClRoZSBkZWZhdWx0IGBwb3NpdGlvbmAgZm9yIGJhciBjaGFydHMgaXMgYHBvc2l0aW9uX3N0YWNrYDoKCmBgYHtyIGRpYW1vbmRzLXN0YWNrMiwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZGlhbW9uZHMsIGFlcyh4ID0gY3V0LAogICAgICAgICAgICAgICAgICAgICBmaWxsID0gY2xhcml0eSkpICsKICAgIGdlb21fYmFyKHBvc2l0aW9uID0gInN0YWNrIikKYGBgCmBgYHtyIGRpYW1vbmRzLXN0YWNrMiwgZWNobyA9IEZBTFNFfQpgYGAKCmBwb3NpdGlvbl9kb2RnZWAgcHJvZHVjZXMgX3NpZGUtYnktc2lkZSBiYXIgY2hhcnRzXzoKCmBgYHtyIGRpYW1vbmRzLWRvZGdlLCBldmFsID0gRkFMU0V9CmdncGxvdChkaWFtb25kcywgYWVzKHggPSBjdXQsCiAgICAgICAgICAgICAgICAgICAgIGZpbGwgPSBjbGFyaXR5KSkgKwogICAgZ2VvbV9iYXIocG9zaXRpb24gPSAiZG9kZ2UiKQpgYGAKYGBge3IgZGlhbW9uZHMtZG9kZ2UsIGVjaG8gPSBGQUxTRX0KYGBgCgpgcG9zaXRpb25fZmlsbGAgcmVzY2FsZXMgYWxsIGJhcnMgdG8gYmUgZXF1YWwgaGVpZ2h0IHRvIGhlbHAgY29tcGFyZQpwcm9wb3J0aW9ucyB3aXRoaW4gYmFycy4KCmBgYHtyIGRpYW1vbmRzLWZpbGwsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KGRpYW1vbmRzLCBhZXMoeCA9IGN1dCwKICAgICAgICAgICAgICAgICAgICAgZmlsbCA9IGNsYXJpdHkpKSArCiAgICBnZW9tX2Jhcihwb3NpdGlvbiA9ICJmaWxsIikKYGBgCmBgYHtyIGRpYW1vbmRzLWZpbGwsIGVjaG8gPSBGQUxTRX0KYGBgCgpVc2luZyB0aGUgY291bnRzIHRvIHNjYWxlIHRoZSB3aWR0aHMgd291bGQgcHJvZHVjZSBhIF9zcGluZSBwbG90XywgYQp2YXJpYW50IG9mIGEgX21vc2FpYyBwbG90Xy4KClRoaXMgaXMgZWFzaWVzdCB0byBkbyB3aXRoIHRoZSBgZ2dtb3NhaWNgIHBhY2thZ2UuCgpgcG9zaXRpb25faml0dGVyYCBjYW4gYmUgdXNlZCB3aXRoIGBnZW9tX3BvaW50YCB0byBhdm9pZCBvdmVycGxvdHRpbmcKb3IgYnJlYWsgdXAgcm91bmRpbmcgYXJ0aWZhY3RzLgoKQW5vdGhlciB2ZXJzaW9uIG9mIHRoZSBPbGQgRmFpdGhmdWwgZGF0YSBhdmFpbGFibGUgYXMgYGdleXNlcmAgaW4KcGFja2FnZSBgTUFTU2AgaGFzIHNvbWUgcm91bmRpbmcgaW4gdGhlIGBkdXJhdGlvbmAgdmFyaWFibGU6CgpgYGB7ciBnZXlzZXIyLCBldmFsID0gRkFMU0V9CmRhdGEoZ2V5c2VyLCBwYWNrYWdlID0gIk1BU1MiKQoKIyMgQWRqdXN0IGZvciBkaWZmZXJlbnQgbWVhbmluZyBvZiBgd2FpdGluZ2AgdmFyaWFibGUKZ2V5c2VyMiA8LSBuYS5vbWl0KG11dGF0ZShnZXlzZXIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgZHVyYXRpb24gPSBsYWcoZHVyYXRpb24pKSkKCnAgPC0gZ2dwbG90KGdleXNlcjIsIGFlcyh4ID0gZHVyYXRpb24sIHkgPSB3YWl0aW5nKSkKcCArIGdlb21fcG9pbnQoKQpgYGAKCmBgYHtyIGdleXNlcjIsIGVjaG8gPSBGQUxTRX0KYGBgCgpfSml0dGVyaW5nXyBjYW4gaGVscCBicmVhayB1cCB0aGUgZGlzdHJhY3RpbmcgX2hlYXBpbmdfIG9mIHZhbHVlcyBvbgpkdXJhdGlvbnMgb2YgMiBhbmQgNCBtaW51dGVzLgoKVGhlIGRlZmF1bHQgYW1vdW50IG9mIGppdHRlcmluZyBpc24ndCBxdWl0ZSBlbm91Z2ggaW4gdGhpcyBjYXNlOgoKYGBge3IgZ2V5c2VyMi1qaXQsIGV2YWwgPSBGQUxTRX0KcCArIGdlb21fcG9pbnQocG9zaXRpb24gPSAiaml0dGVyIikKYGBgCmBgYHtyIGdleXNlcjItaml0LCBlY2hvID0gRkFMU0V9CmBgYAoKVG8gaml0dGVyIG9ubHkgaG9yaXpvbnRhbGx5IGFuZCBieSBhIGxhcmdlciBhbW91bnQgeW91IGNhbiB1c2UKCmBgYHtyIGdleXNlcjItaml0MiwgZXZhbCA9IEZBTFNFfQpwICsgZ2VvbV9wb2ludChwb3NpdGlvbiA9CiAgICAgICAgICAgICAgICAgICBwb3NpdGlvbl9qaXR0ZXIoaGVpZ2h0ID0gMCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB3aWR0aCA9IDAuMSkpCmBgYApgYGB7ciBnZXlzZXIyLWppdDIsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgQ29vcmRpbmF0ZSBTeXN0ZW1zCgpDb29yZGluYXRlIHN5c3RlbSBmdW5jdGlvbnMgaW5jbHVkZQoKYGBge3IsIGVjaG8gPSBGQUxTRSwgcmVzdWx0cyA9ICJhc2lzIn0Kc2hvd0xpc3QobHMoInBhY2thZ2U6Z2dwbG90MiIsIHBhdCA9ICJeY29vcmRfIikpCmBgYAoKVGhlIGRlZmF1bHQgY29vcmRpbmF0ZSBzeXN0ZW0gaXMgYGNvb3JkX2NhcnRlc2lhbmAuCgoKIyMjIENhcnRlc2lhbiBDb29yZGluYXRlcwoKYGNvb3JkX2NhcnRlc2lhbmAgY2FuIGJlIHVzZWQgdG8gX3pvb20gaW5fIG9uIGEgcGFydGljdWxhciByZWdpaW9uOgoKYGBge3IgZ2V5c2VyMi16b29tLCBldmFsID0gRkFMU0V9CnAgKyBnZW9tX3BvaW50KCkgKwogICAgY29vcmRfY2FydGVzaWFuKHhsaW0gPSBjKDMsIDQpKQpgYGAKYGBge3IgZ2V5c2VyMi16b29tLCBlY2hvID0gRkFMU0V9CmBgYAoKYGNvb3JkX2ZpeGVkYCBhbmQgYGNvb3JkX2VxdWFsYCBmaXggdGhlIF9hc3BlY3QgcmF0aW9fIGZvciBhIGNhcnRlc2lhbgpjb29yZGluYXRlIHN5c3RlbS4KClRoZSBhc3BlY3QgcmF0aW8gaXMgdGhlIHJhdGlvIG9mIHRoZSBudW1iZXIgcGh5c2ljYWwgZGlzcGxheSB1bml0cyBwZXIKYHlgIHVuaXQgdG8gdGhlIG51bWJlciBvZiBwaHlzaWNhbCBkaXNwbGF5IHVuaXRzIHBlciBgeGAgdW5pdC4KClRoZSBhc3BlY3QgcmF0aW8gY2FuIGJlIGltcG9ydGFudCBmb3IgcmVjb2duaXppbmcgZmVhdHVyZXMgYW5kIHBhdHRlcm5zLgoKYGBge3J9CnJpdmVyIDwtIHNjYW4oImh0dHBzOi8vd3d3LnN0YXQudWlvd2EuZWR1L35sdWtlL2RhdGEvcml2ZXIuZGF0IikKciA8LSBkYXRhLmZyYW1lKGZsb3cgPSByaXZlciwgbW9udGggPSBzZXFfYWxvbmcocml2ZXIpKQpgYGAKCmBgYHtyIHJpdmVyLWZsYXQsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KHIsIGFlcyh4ID0gbW9udGgsIHkgPSBmbG93KSkgKwogICAgZ2VvbV9wb2ludCgpICsKICAgIGNvb3JkX2ZpeGVkKHJhdGlvID0gNCkKYGBgCmBgYHtyIHJpdmVyLWZsYXQsIGVjaG8gPSBGQUxTRSwgZmlnLmhlaWdodCA9IDIsIGZpZy53aWR0aCA9IDh9CmBgYAoKCiMjIyBQb2xhciBDb29yZGluYXRlcwoKQSBmaWxsZWQgYmFyIGNoYXJ0CgpgYGB7ciBkaWFtb25kcy1maWxsLTEsIGV2YWwgPSBGQUxTRX0KKHAgPC0gZ2dwbG90KGRpYW1vbmRzKSArCiAgICAgZ2VvbV9iYXIoYWVzKHggPSAxLCBmaWxsID0gY3V0KSwKICAgICAgICAgICAgICBwb3NpdGlvbiA9ICJmaWxsIikpCmBgYApgYGB7ciBkaWFtb25kcy1maWxsLTEsIGVjaG8gPSBGQUxTRX0KYGBgCgppcyB0dXJuZWQgaW50byBhIHBpZSBjaGFydCBieSBjaGFuZ2luZyB0byBwb2xhciBjb29yZGluYXRlczoKCmBgYHtyIGRpYW1vbmRzLXBpZSwgZXZhbCA9IEZBTFNFfQpwICsgY29vcmRfcG9sYXIodGhldGEgPSAieSIpCmBgYApgYGB7ciBkaWFtb25kcy1waWUsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMjIENvb3JkaW5hdGUgU3lzdGVtcyBmb3IgTWFwcwoKQ29vcmRpbmF0ZSBzeXN0ZW1zIGFyZSBwYXJ0aWN1bGFybHkgaW1wb3J0YW50IGZvciBtYXBzLgoKUG9seWdvbnMgZm9yIG1hbnkgcG9saXRpY2FsIGFuZCBnZW9ncmFwaGljIGJvdW5kYXJpZXMgYXJlIGF2YWlsYWJsZQp0aHJvdWdoIHRoZSBgbWFwX2RhdGFgIGZ1bmN0aW9uLgoKQm91bmRhcmllcyBmb3IgdGhlIGxvd2VyIDQ4IFVTIHN0YXRlcyBjYW4gYmUgb2J0YWluZWQgYXMKCmBgYHtyfQp1c2EgPC0gbWFwX2RhdGEoInN0YXRlIikKYGBgCgpQb2x5Z29uIHZlcnRpY2VzIGFyZSBlbmNvZGVkIGJ5IGxvbmdpdHVkZSBhbmQgbGF0aXR1ZGUuCgpQbG90dGluZyB0aGVzZSBpbiB0aGUgZGVmYXVsdCBjYXJ0ZXNpYW4gY29vcmRpbmF0ZSBzeXN0ZW0gdXN1YWxseSBkb2VzCm5vdCB3b3JrIHdlbGw6CgpgYGB7ciB1c2EtY2FydCwgZXZhbCA9IEZBTFNFfQp1c2EgPC0gbWFwX2RhdGEoInN0YXRlIikKbSA8LSBnZ3Bsb3QodXNhLCBhZXMoeCA9IGxvbmcsCiAgICAgICAgICAgICAgICAgICAgIHkgPSBsYXQsCiAgICAgICAgICAgICAgICAgICAgIGdyb3VwID0gZ3JvdXApKSArCiAgICBnZW9tX3BvbHlnb24oZmlsbCA9ICJ3aGl0ZSIsCiAgICAgICAgICAgICAgICAgY29sb3IgPSAiYmxhY2siKQptCmBgYApgYGB7ciB1c2EtY2FydCwgZWNobyA9IEZBTFNFfQpgYGAKClVzaW5nIGEgZml4ZWQgYXNwZWN0IHJhdGlvIGlzIGJldHRlciwgYnV0IGFuIGFzcGVjdCByYXRpbyBvZiAxIGRvZXMKbm90IHdvcmsgd2VsbDoKCmBgYHtyfQptICsgY29vcmRfZXF1YWwoKQpgYGAKClRoZSBwcm9ibGVtIGlzIHRoYXQgYXdheSBmcm9tIHRoZSBlcXVhdG9yIGEgb25lIGRlZ3JlZSBjaGFuZ2UgaW4KbGF0aXR1ZGUgY29ycmVzcG9uZHMgdG8gYSBsYXJnZXIgZGlzdGFuY2UgdGhhbiBhIG9uZSBkZWdyZWUgY2hhbmdlIGluCmxvbmdpdHVkZS4KClRoZSByYXRpbyBvZiBvbmUgZGVncmVlIGxvbmdpdHVkZSBzZXBhcmF0aW9uIHRvIG9uZSBkZWdyZWUgbGF0aXR1ZGUKc2VwYXJhdGlvbiBmb3IgdGhlIGxhdGl0dWRlIGF0IHRoZSBtaWRkbGUgb2YgSW93YSBvZiA0MSBkZWdyZWVzIGlzCgpgYGB7cn0KbG9uZ2xhdCA8LSBjb3MoNDEgLyA5MCAqIHBpIC8gMikKbG9uZ2xhdApgYGAKCkEgYmV0dGVyIG1hcCBpcyBvYnRhaW5lZCB1c2luZyB0aGUgYXNwZWN0IHJhdGlvIGAxIC8gbG9uZ2xhdGA6CgpgYGB7ciB1c2EtZml4ZWQsIGV2YWwgPSBGQUxTRX0KbSArIGNvb3JkX2ZpeGVkKDEgLyBsb25nbGF0KQpgYGAKYGBge3IgdXNhLWZpeGVkLCBlY2hvID0gRkFMU0V9CmBgYAoKVGhlIGJlc3QgYXBwcm9hY2ggaXMgdG8gdXNlIGEgY29vcmRpbmF0ZSBzeXN0ZW0gZGVzaWduZWQgc3BlY2lmaWNhbGx5IGZvcgptYXBzLgoKVGhlcmUgYXJlIG1hbnkgX3Byb2plY3Rpb25zXyB1c2VkIGluIG1hcCBtYWtpbmcuCgpUaGUgZGVmYXVsdCBwcm9qZWN0aW9uIHVzZWQgYnkgYGNvb3JkX21hcGAgaXMgdGhlCltNZXJjYXRvcl0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvTWVyY2F0b3JfcHJvamVjdGlvbikKcHJvamVjdGlvbi4KCmBgYHtyIHVzYS1tZXJjYXRvciwgZXZhbCA9IEZBTFNFfQptICsgY29vcmRfbWFwKCkKYGBgCgpgYGB7ciB1c2EtbWVyY2F0b3IsIGVjaG8gPSBGQUxTRX0KYGBgCgpQcm9wZXIgbWFwIHByb2plY3Rpb25zIGFyZSBub24tbGluZWFyOyB0aGlzIGlzIGVhc2llciB0byBzZWUgd2l0aCBhbgpBbGJlcnMgcHJvamVjdGlvbjoKCmBgYHtyIHVzYS1hbGJlcnMsIGV2YWwgPSBGQUxTRX0KbSArIGNvb3JkX21hcCgiYWxiZXJzIiwgMjAsIDUwKQpgYGAKYGBge3IgdXNhLWFsYmVycywgZWNobyA9IEZBTFNFfQpgYGAKCgojIyBTY2FsZXMKClNjYWxlcyBhcmUgdXNlZCBmb3IgY29udHJvbGxpbmcgdGhlIG1hcHBpbmcgb2YgdmFsdWVzIHRvIHBoeXNpY2FsCnJlcHJlc2VudGF0aW9ucyBzdWNoIGFzIGNvbG9ycywgc2hhcGVzLCBhbmQgcG9zaXRpb25zLgoKU2NhbGUgZnVuY3Rpb25zIGFyZSBhbHNvIHJlc3BvbnNpYmxlIGZvciBwcm9kdWNpbmcgX2d1aWRlc18gZm9yCnRyYW5zbGF0aW5nIHBoeXNpY2FsIHJlcHJlc2VudGF0aW9ucyBiYWNrIHRvIHZhbHVlcywgc3VjaCBhcwoKKiBheGlzIGxhYmVscyBhbmQgbWFya3M7CgoqIGNvbG9yIG9yIHNoYXBlIGxlZ2VuZHMuCgpUaGVyZSBhcmUgY3VycmVudGx5IGByIGxlbmd0aChscygicGFja2FnZTpnZ3Bsb3QyIiwgcGF0ID0gInNjYWxlXyIpKWAKc2NhbGUgZnVuY3Rpb25zOyBzb21lIGV4YW1wbGVzIGFyZQoKYGBgcgpzY2FsZV9jb2xvcl9ncmFkaWVudCAgICAgIHNjYWxlX3NoYXBlX21hbnVhbCAgICAgc2NhbGVfeF9sb2cxMApzY2FsZV9jb2xvcl9tYW51YWwgICAgICAgIHNjYWxlX3NpemVfYXJlYSAgICAgICAgc2NhbGVfeV9sb2cxMApzY2FsZV9maWxsX2dyYWRpZW50ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2NhbGVfeF9zcXJ0CnNjYWxlX2ZpbGxfbWFudWFsICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzY2FsZV95X3NxcnQKCmBgYAoKQW4gW2V4cGVyaW1lbnRhbCB0b29sXShodHRwczovL2dncGxvdDJ0b3IuY29tL3NjYWxlcy8pIHRvIGhlbHAKY2hvb3Npbmcgc2NhbGVzIGhhcyByZWNlbnRseSBiZWVuIGludHJvZHVjZWQuCgpTdGFydCB3aXRoIGEgYmFzaWMgc2NhdHRlciBwbG90OgoKYGBge3IgbXBnLWJhc2ljLCBldmFsID0gRkFMU0V9CihwIDwtIGdncGxvdChtcGcsIGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICAgICB5ID0gaHd5KSkgKwogICAgIGdlb21fcG9pbnQoKSkKYGBgCmBgYHtyIG1wZy1iYXNpYywgZWNobyA9IEZBTFNFfQpgYGAKClJlbW92ZSB0aGUgYHhgIHRpY2sgbWFya3MgYW5kIGxhYmVscyAodGhpcyBjYW4gYWxzbyBiZSBkb25lIHdpdGggdGhlbWUKc2V0dGluZ3MpOgoKYGBge3IgbXBnLW5vLXRpY2tzLWxhYnMsIGV2YWwgPSBGQUxTRX0KcCArIHNjYWxlX3hfY29udGludW91cyhsYWJlbHMgPSBOVUxMLAogICAgICAgICAgICAgICAgICAgICAgIGJyZWFrcyA9IE5VTEwpCmBgYApgYGB7ciBtcGctbm8tdGlja3MtbGFicywgZWNobyA9IEZBTFNFfQpgYGAKCkNoYW5nZSB0aGUgdGljayBsb2NhdGlvbnMgYW5kIGxhYmVsczoKCmBgYHtyIG1wZy1uZXctdGlja3MtbGFicywgZXZhbCA9IEZBTFNFfQpwICsgc2NhbGVfeF9jb250aW51b3VzKGxhYmVscyA9CiAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhc3RlKGMoMiwgNCwgNiksICJsdHIiKSwKICAgICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBjKDIsIDQsIDYpKQpgYGAKYGBge3IgbXBnLW5ldy10aWNrcy1sYWJzLCBlY2hvID0gRkFMU0V9CmBgYAoKVXNlIGEgbG9nYXJpdGhtaWMgYXhpczoKCmBgYHtyIG1wZy1sb2cteCwgZXZhbCA9IEZBTFNFfQpwICsgc2NhbGVfeF9sb2cxMChsYWJlbHMgPSBwYXN0ZShjKDIsIDQsIDYpLCAibHRyIiksCiAgICAgICAgICAgICAgICAgIGJyZWFrcyA9IGMoMiwgNCwgNiksCiAgICAgICAgICAgICAgICAgIG1pbm9yX2JyZWFrcyA9IGMoMywgNSwgNykpCmBgYApgYGB7ciBtcGctbG9nLXgsIGVjaG8gPSBGQUxTRX0KYGBgCgpUaGUKW1NjYWxlc10oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei9jb21tdW5pY2F0aW9uLmh0bWwjc2NhbGVzKQpzZWN0aW9uIGluIFtSIGZvciBEYXRhIFNjaWVuY2VdKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovKSBwcm92aWRlcyBzb21lCm1vcmUgZGV0YWlscy4KCkNvbG9yIGFzc2lnbm1lbnQgY2FuIGFsc28gYmUgY29udHJvbGxlZCBieSBzY2FsZSBmdW5jdGlvbnMuCgpGb3IgZXhhbXBsZSwgZm9yIHNvbWUgcHJlc2lkZW50aWFsIGFwcHJvdmFsIHJhdGluZ3MgZGF0YQoKYGBge3IsIGluY2x1ZGUgPSBGQUxTRX0KcHJfYXBwciA8LSBkYXRhLmZyYW1lKHByZXMgPSBjKCJPYmFtYSIsICJDYXJ0ZXIiLCAiQ2xpbnRvbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiRy5XLiBCdXNoIiwgIlJlYWdhbiIsICJHLkguVyBCdXNoIiwgIlRydW1wIiksCiAgICAgICAgICAgICAgICAgICAgICBhcHByID0gYyg3OSwgNzgsIDY4LCA2NSwgNTgsIDU2LCA0MCksCiAgICAgICAgICAgICAgICAgICAgICBwYXJ0eSA9IGMoIkQiLCAiRCIsICJEIiwgIlIiLCAiUiIsICJSIiwgIlIiKSwKICAgICAgICAgICAgICAgICAgICAgIHllYXIgPSBjKDIwMDksIDE5NzcsIDE5OTMsIDIwMDEsIDE5ODEsIDE5ODksIDIwMTcpKQpwcl9hcHByIDwtIG11dGF0ZShwcl9hcHByLCBwcmVzID0gcmVvcmRlcihwcmVzLCBhcHByKSkKYGBgCmBgYHtyfQpwcl9hcHByCmBgYAoKdGhlIGRlZmF1bHQgY29sb3Igc2NhbGUgaXMgbm90IGlkZWFsOgoKYGBge3IgcHItYXBwcjAsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KHByX2FwcHIsCiAgICAgICBhZXMoeCA9IGFwcHIsIHkgPSBwcmVzLCBmaWxsID0gcGFydHkpKSArCiAgICBnZW9tX2NvbCgpCmBgYAoKYGBge3IgcHItYXBwcjAsIGVjaG8gPSBGQUxTRX0KYGBgCgpUaGUgY29tbW9uIGFzc2lnbm1lbnQgb2YgcmVkIGZvciBSZXB1YmxpY2FuIGFuZCBibHVlIGZvciBEZW1vY3JhdCBjYW4KYmUgb2J0YWluZWQgYnkKCmBgYHtyIHByLWFwcHIsIGV2YWwgPSBGQUxTRX0KZ2dwbG90KHByX2FwcHIsCiAgICAgICBhZXMoeCA9IGFwcHIsIHkgPSBwcmVzLCBmaWxsID0gcGFydHkpKSArCiAgICBnZW9tX2NvbCgpICsKICAgIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcwogICAgICAgICAgICAgICAgICAgICAgPSBjKFIgPSAicmVkIiwgRCA9ICJibHVlIikpCmBgYAoKYGBge3IgcHItYXBwciwgZWNobyA9IEZBTFNFfQpgYGAKCkEgYmV0dGVyIGNob2ljZSBpcyB0byB1c2UgYSB3ZWxsLWRlc2lnbmVkIFtjb2xvcgpwYWxldHRlXShodHRwczovL2hjbHdpemFyZC5vcmcvI2NvbG9yLXBhbGV0dGVzKToKCmBgYHtyIHByLWFwcHItMiwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QocHJfYXBwciwKICAgICAgIGFlcyh4ID0gYXBwciwgeSA9IHByZXMsIGZpbGwgPSBwYXJ0eSkpICsKICAgIGdlb21fY29sKCkgKwogICAgY29sb3JzcGFjZTo6c2NhbGVfZmlsbF9kaXNjcmV0ZV9kaXZlcmdpbmcoCiAgICAgICAgICAgICAgICAgICAgcGFsZXR0ZSA9ICJCbHVlLVJlZCAyIikKYGBgCgpgYGB7ciBwci1hcHByLTIsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgRmFjZXRzCgpGYWNldGluZyB1c2VzIHRoZSBfc21hbGwgbXVsdGlwbGVzXyBhcHByb2FjaCB0byBpbnRyb2R1Y2UgYWRkaXRpb25hbAp2YXJpYWJsZXMuCgpGb3IgYSBzaW5nbGUgdmFyaWFibGUgYGZhY2V0X3dyYXBgIGlzIHVzdWFsbHkgdXNlZDoKCmBgYHtyIG1wZy1mYWNldC13cmFwLCBldmFsID0gRkFMU0V9CnAgPC0gZ2dwbG90KG1wZykgKwogICAgZ2VvbV9wb2ludChhZXMoeCA9IGRpc3BsLAogICAgICAgICAgICAgICAgICAgeSA9IGh3eSkpCnAgKyBmYWNldF93cmFwKH4gY2xhc3MpCmBgYApgYGB7ciBtcGctZmFjZXQtd3JhcCwgZWNobyA9IEZBTFNFfQpgYGAKCkZvciB0d28gdmFyaWFibGVzLCBlYWNoIHdpdGggYSBtb2Rlc3QgbnVtYmVyIG9mIGNhdGVnb3JpZXMsCmBmYWNldF9ncmlkYCBjYW4gYmUgZWZmZWN0aXZlOgoKYGBge3IgbXBnLWZhY2V0LWdyaWQsIGV2YWwgPSBGQUxTRX0KcCArIGZhY2V0X2dyaWQoZmFjdG9yKGN5bCkgfiBkcnYpCmBgYApgYGB7ciBtcGctZmFjZXQtZ3JpZCwgZWNobyA9IEZBTFNFfQpgYGAKCjwhLS0KVXNpbmcgdGhlIHByZXZpb3VzIG1wZyBmYWNldCBwbG90IHdvdWxkIGJlIGJldHRlciBidXQgdGhpcyBpcyBvbmUgb2YKdGhlIGhvbWV3b3JrIHByb2JsZW1zIGluIEhXMwotLT4KClRvIHNob3cgY29tbW9uIGRhdGEgaW4gYWxsIGZhY2V0cyBtYWtlIHN1cmUgdGhlIGRhdGEgZG9lcyBub3QgY29udGFpbiB0aGUKZmFjZXRpbmcgdmFyaWFibGUuCgpUaGlzIHdhcyB1c2VkIHRvIHNob3cgbXV0ZWQgdmlld3Mgb2YgdGhlIGZ1bGwgZGF0YSBpbiBmYWNldGVkIHBsb3RzLgoKQSBmYWNldGVkIHBsb3Qgb2YgdGhlIGBnYXBtaW5kZXJgIGRhdGE6CgpgYGB7ciBnYXBtaW5kZXItbm90LW11dGVkLCBldmFsID0gRkFMU0V9CmxpYnJhcnkoZ2FwbWluZGVyKQoKeWVhcnNfdG9fa2VlcCA8LSBjKDE5NzcsIDE5ODcsIDE5OTcsIDIwMDcpCmdkIDwtIGZpbHRlcihnYXBtaW5kZXIsCiAgICAgICAgICAgICB5ZWFyICVpbiUgeWVhcnNfdG9fa2VlcCkKCmdncGxvdChnZCwKICAgICAgIGFlcyh4ID0gZ2RwUGVyY2FwLAogICAgICAgICAgIHkgPSBsaWZlRXhwLAogICAgICAgICAgIGNvbG9yID0gY29udGluZW50KSkgKwogICAgZ2VvbV9wb2ludChzaXplID0gMi41KSArCiAgICBzY2FsZV94X2xvZzEwKCkgKwogICAgZmFjZXRfd3JhcCh+IHllYXIpCmBgYAoKYGBge3IgZ2FwbWluZGVyLW5vdC1tdXRlZCwgZWNobyA9IEZBTFNFLCBmaWcud2lkdGggPSA4fQpgYGAKCkFkZCBhIG11dGVkIHZlcnNpb24gb2YgdGhlIGZ1bGwgZGF0YSBpbiB0aGUgYmFja2dyb3VuZCBvZiBlYWNoIHBhbmVsOgoKPCEtLSB2YXJpYW50IG9mIGNvZGUgaW4gcHJlcmNlcC5SbWQgLS0+CmBgYHtyIGdhcG1pbmRlci1tdXRlZCwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGdhcG1pbmRlcikKCnllYXJzX3RvX2tlZXAgPC0gYygxOTc3LCAxOTg3LCAxOTk3LCAyMDA3KQpnZCA8LSBmaWx0ZXIoZ2FwbWluZGVyLAogICAgICAgICAgICAgeWVhciAlaW4lIHllYXJzX3RvX2tlZXApCmdkX25vX3llYXIgPC0gbXV0YXRlKGdkLCB5ZWFyID0gTlVMTCkKCmdncGxvdChnZCwKICAgICAgIGFlcyh4ID0gZ2RwUGVyY2FwLAogICAgICAgICAgIHkgPSBsaWZlRXhwLAogICAgICAgICAgIGNvbG9yID0gY29udGluZW50KSkgKwogICAgZ2VvbV9wb2ludChkYXRhID0gZ2Rfbm9feWVhciwKICAgICAgICAgICAgICAgY29sb3IgPSAiZ3JleTgwIikgKwogICAgZ2VvbV9wb2ludChzaXplID0gMi41KSArCiAgICBzY2FsZV94X2xvZzEwKCkgKwogICAgZmFjZXRfd3JhcCh+IHllYXIpCmBgYAoKYGBge3IgZ2FwbWluZGVyLW11dGVkLCBlY2hvID0gRkFMU0UsIGZpZy53aWR0aCA9IDh9CmBgYAoKVXN1YWxseSBmYWNldHMgdXNlIGNvbW1vbiBheGlzIHNjYWxlcywgYnV0IG9uZSBvciBib3RoIGNhbiBiZSBhbGxvd2VkCnRvIHZhcnkuCgpBIHVzZWZ1bCBhcHByb2FjaCBmb3Igc2hvd2luZyB0aW1lIHNlcmllcyBkYXRhIHdpdGggYSBnb29kIGFzcGVjdApyYXRpbyBjYW4gYmUgdG8gc3BsaXQgdGhlIGRhdGEgaW50byBmYWNldHMgZm9yIG5vbi1vdmVybGFwcGluZwpwb3J0aW9ucyBvZiB0aGUgdGltZSBheGlzLgoKYGBge3Igcml2ZXItZmFjZXQsIGV2YWwgPSBGQUxTRX0KcGQgPC0gcmVwKHBhc3RlKHNlcSgxLCBieSA9IDMyLCBsZW5ndGgub3V0ID0gNCksCiAgICAgICAgICAgICAgICBzZXEoMzIsIGJ5ID0gMzIsIGxlbmd0aC5vdXQgPSA0KSwKICAgICAgICAgICAgICAgIHNlcCA9ICIgLSAiKSwKICAgICAgICAgZWFjaCA9ICAzMikKcmQgPC0gZGF0YS5mcmFtZShtb250aCA9IHNlcV9hbG9uZyhyaXZlciksCiAgICAgICAgICAgICAgICAgZmxvdyA9IHJpdmVyLAogICAgICAgICAgICAgICAgIHBhbmVsID0gcGQpCmdncGxvdChyZCwgYWVzKHggPSBtb250aCwKICAgICAgICAgICAgICAgeSA9IGZsb3cpKSArCiAgICBnZW9tX3BvaW50KCkgKwogICAgZmFjZXRfd3JhcCh+IHBhbmVsLAogICAgICAgICAgICAgICBzY2FsZSA9ICJmcmVlX3giLAogICAgICAgICAgICAgICBuY29sID0gMSkKYGBgCmBgYHtyIHJpdmVyLWZhY2V0LCBlY2hvID0gRkFMU0V9CmBgYAoKRmFjZXQgYXJyYW5nZW1lbnQgY2FuIGFsc28gYmUgdXNlZCB0byBjb252ZXkgb3RoZXIgaW5mb3JtYXRpb24sIHN1Y2gKYXMgZ2VvZ3JhcGhpYyBsb2NhdGlvbi4KClRoZSBbYGdlb2ZhY2V0YCBwYWNrYWdlXShodHRwczovL2hhZmVuLmdpdGh1Yi5pby9nZW9mYWNldC8pIGFsbG93cwpmYWNldHMgdG8gYmUgcGxhY2VkIGluIGFwcHJveGltYXRlIGxvY2F0aW9ucyBvZiBkaWZmZXJlbnQgZ2VvZ3JhcGhpYwpyZWdpb25zLgoKQW4gZXhhbXBsZSBmb3IgZGF0YSBmcm9tIFVTIHN0YXRlczoKCmBgYHtyIGdlb2ZhY2V0LCBldmFsID0gRkFMU0V9CmxpYnJhcnkoZ2VvZmFjZXQpCmdncGxvdChzdGF0ZV91bmVtcCwgYWVzKHllYXIsIHJhdGUpKSArCiAgICBnZW9tX2xpbmUoKSArCiAgICBmYWNldF9nZW8ofiBzdGF0ZSwKICAgICAgICAgICAgICBncmlkID0gInVzX3N0YXRlX2dyaWQyIiwKICAgICAgICAgICAgICBsYWJlbCA9ICJjb2RlIikgKwogICAgc2NhbGVfeF9jb250aW51b3VzKGxhYmVscyA9CiAgICAgICAgICAgICAgICAgICAgICAgICAgIGZ1bmN0aW9uKHgpIHBhc3RlMCgiJyIsIHN1YnN0cih4LCAzLCA0KSkpICsKICAgIGxhYnModGl0bGUgPSAiU2Vhc29uYWxseSBBZGp1c3RlZCBVUyBVbmVtcGxveW1lbnQgUmF0ZSAyMDAwLTIwMTYiLAogICAgICAgICBjYXB0aW9uID0gIkRhdGEgU291cmNlOiBibHMuZ292IiwKICAgICAgICAgeCA9ICJZZWFyIiwKICAgICAgICAgeSA9ICJVbmVtcGxveW1lbnQgUmF0ZSAoJSkiKSArCiAgICB0aGVtZShzdHJpcC50ZXh0LnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IDYpLAogICAgICAgICAgYXhpcy50ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSA1KSkKYGBgCmBgYHtyIGdlb2ZhY2V0LCBlY2hvID0gRkFMU0UsIG1lc3NhZ2UgPSBGQUxTRX0KYGBgCgpBcnJhbmdlbWVudCBhY2NvcmRpbmcgdG8gYSBjYWxlbmRhciBjYW4gYWxzbyBiZSB1c2VmdWwuCgoKIyMgVGhlbWVzCgpgZ2dwbG90MmAgc3VwcG9ydHMgdGhlIG5vdGlvbiBvZiBfdGhlbWVzXyBmb3IgYWRqdXN0aW5nIG5vbi1kYXRhCmFwcGVhcmFuY2UgYXNwZWN0cyBvZiBhIHBsb3QsIHN1Y2ggYXMKCiogcGxvdCB0aXRsZXMKCiogYXhpcyBhbmQgbGVnZW5kIHBsYWNlbWVudCBhbmQgdGl0bGVzCgoqIGJhY2tncm91bmQgY29sb3JzCgoqIGd1aWRlIGxpbmUgcGxhY2VtZW50CgpUaGVtZSBlbGVtZW50cyBjYW4gYmUgY3VzdG9taXplZCBpbiBzZXZlcmFsIHdheXM6CgoqIGB0aGVtZSgpYCBjYW4gYmUgdXNlZCB0byBhZGp1c3QgaW5kaXZpZHVhbCBlbGVtZW50cyBpbiBhIHBsb3QuCgoqIGB0aGVtZV9zZXQoKWAgYWRqdXN0cyBkZWZhdWx0IHNldHRpbmdzIGZvciBhIHNlc3Npb247CgoqIHByZS1kZWZpbmVkIHRoZW1lIGZ1bmN0aW9ucyBhbGxvdyBjb25zaXN0ZW50IHN0eWxlIGNoYW5nZXMuCgpUaGUKW2Z1bGwgZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9nZ3Bsb3QyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL3RoZW1lLmh0bWwpCm9mIHRoZSBgdGhlbWVgIGZ1bmN0aW9uIGxpc3RzIG1hbnkgY3VzdG9taXphYmxlIGVsZW1lbnRzLgoKT25lIHNpbXBsZSBleGFtcGxlOgoKYGBge3IgdGhlbWUtc2ltcGxlLCBldmFsID0gRkFMU0V9CmdncGxvdChtdXRhdGUobXBnLCBjeWwgPSBmYWN0b3IoY3lsKSkpICsKICAgIGdlb21fcG9pbnQoYWVzKHggPSBkaXNwbCwKICAgICAgICAgICAgICAgICAgIHkgPSBod3ksCiAgICAgICAgICAgICAgICAgICBmaWxsID0gY3lsKSwKICAgICAgICAgICAgICAgc2hhcGUgPSAyMSwKICAgICAgICAgICAgICAgc2l6ZSA9IDMpICsKICAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJ0b3AiLAogICAgICAgICAgYXhpcy50ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMiksCiAgICAgICAgICBheGlzLnRpdGxlID0gZWxlbWVudF90ZXh0KHNpemUgPSAxNCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmFjZSA9ICJib2xkIikpCmBgYApgYGB7ciB0aGVtZS1zaW1wbGUsIGVjaG8gPSBGQUxTRX0KYGBgCgpBbm90aGVyIGV4YW1wbGU6CgpgYGB7ciB0aGVtZS1zaW1wbGUtMiwgZXZhbCA9IEZBTFNFfQpndGhtIDwtCiAgICB0aGVtZShwbG90LmJhY2tncm91bmQgPQogICAgICAgICAgICAgIGVsZW1lbnRfcmVjdChmaWxsID0gImxpZ2h0Ymx1ZSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG9yID0gTkEpLAogICAgICAgICAgcGFuZWwuYmFja2dyb3VuZCA9CiAgICAgICAgICAgICAgZWxlbWVudF9yZWN0KGZpbGwgPSAicGluayIpKQpwICsgZ3RobQpgYGAKYGBge3IgdGhlbWUtc2ltcGxlLTIsIGVjaG8gPSBGQUxTRX0KYGBgCgpTb21lIGFsdGVybmF0ZSBjb21wbGV0ZSB0aGVtZXMgcHJvdmlkZWQgYnkgYGdncGxvdDJgIGFyZQoKYGBgcgp0aGVtZV9idyAgICAgICAgdGhlbWVfZ3JheSAgICAgIHRoZW1lX21pbmltYWwgICB0aGVtZV92b2lkCnRoZW1lX2NsYXNzaWMgICB0aGVtZV9ncmV5ICAgICAgdGhlbWVfZGFyayAgICAgIHRoZW1lX2xpZ2h0CmBgYAoKU29tZSBleGFtcGxlczoKCmBgYHtyIGFsdC10aGVtZXMsIGV2YWwgPSBGQUxTRX0KcF9idyA8LSBwICsgdGhlbWVfYncoKSArIGdndGl0bGUoIkJXIikKCnBfY2xhc3NpYyA8LSBwICsgdGhlbWVfY2xhc3NpYygpICsgZ2d0aXRsZSgiQ2xhc3NpYyIpCgpwX21pbiA8LSBwICsgdGhlbWVfbWluaW1hbCgpICsgZ2d0aXRsZSgiTWluaW1hbCIpCgpwX3ZvaWQgPC0gcCArIHRoZW1lX3ZvaWQoKSArIGdndGl0bGUoIlZvaWQiKQoKbGlicmFyeShwYXRjaHdvcmspCihwX2J3ICsgcF9jbGFzc2ljKSAvIChwX21pbiArIHBfdm9pZCkKYGBgCmBgYHtyIGFsdC10aGVtZXMsIGVjaG8gPSBGQUxTRX0KYGBgCgpUaGUKW2BnZ3RoZW1lc2BdKGh0dHA6Ly93d3cucnB1YnMuY29tL01lbnRvcnNfVWJpcXVtL2dndGhlbWVzXzEpCnBhY2thZ2UgcHJvdmlkZXMgc29tZSBhZGRpdGlvbmFsIHRoZW1lcy4KClNvbWUgZXhhbXBsZXM6CgpgYGB7ciBnZ3RoZW1lcy1leGFtcGxlcywgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGdndGhlbWVzKQoKcF9lY29uIDwtIHAgKyB0aGVtZV9lY29ub21pc3QoKSArIGdndGl0bGUoIkVjb25vbWlzdCIpCgpwX3dzaiA8LSBwICsgdGhlbWVfd3NqKCkgKyBnZ3RpdGxlKCJXU0oiKQoKcF90dWZ0ZSA8LSBwICsgdGhlbWVfdHVmdGUoKSArIGdndGl0bGUoIlR1ZnRlIikKCnBfZmV3IDwtIHAgKyB0aGVtZV9mZXcoKSArIGdndGl0bGUoIkZldyIpCgoocF9lY29uICsgcF93c2opIC8gKHBfdHVmdGUgKyBwX2ZldykKYGBgCmBgYHtyIGdndGhlbWVzLWV4YW1wbGVzLCBlY2hvID0gRkFMU0V9CmBgYAoKYGdndGhlbWVzYCBhbHNvIHByb3ZpZGVzIGB0aGVtZV9tYXBgIHRoYXQgcmVtb3ZlcyB1bm5lY2Vzc2FyeSBlbGVtZW50cwpmcm9tIG1hcHM6CgpgYGB7cn0KbSArIGNvb3JkX21hcCgpICsgdGhlbWVfbWFwKCkKYGBgCgpUaGUKW1RoZW1lc10oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei9jb21tdW5pY2F0aW9uLmh0bWwjc2VjLXRoZW1lcykKc2VjdGlvbiBpbiBbUiBmb3IgRGF0YSBTY2llbmNlXShodHRwczovL3I0ZHMuaGFkbGV5Lm56LykgcHJvdmlkZXMgc29tZQptb3JlIGRldGFpbHMuCgoKIyMgQSBNb3JlIENvbXBsZXRlIFRlbXBsYXRlCgpgYGByCmdncGxvdChkYXRhID0gPERBVEE+KSArCiAgICA8R0VPTT4obWFwcGluZyA9IGFlcyg8TUFQUElOR1M+KSwKICAgICAgICAgICBzdGF0ID0gPFNUQVQ+LAogICAgICAgICAgIHBvc2l0aW9uID0gPFBPU0lUSU9OPikgKwogICAgPCAuLi4gTU9SRSBHRU9NUyAuLi4gPiArCiAgICA8Q09PUkRJTkFURV9BREpVU1RNRU5UPiArCiAgICA8U0NBTEVfQURKVVNUTUVOVD4gKwogICAgPEZBQ0VUSU5HPiArCiAgICA8VEhFTUVfQURKVVNUTUVOVD4KYGBgCgoKIyMgTGFiZWxzIGFuZCBBbm5vdGF0aW9ucwoKQSBiYXNpYyBwbG90OgoKYGBge3IgbXBnLWFubiwgZXZhbCA9IEZBTFNFfQpwIDwtIGdncGxvdChtcGcsIGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICAgIHkgPSBod3kpKQpwMSA8LSBwICsgZ2VvbV9wb2ludChhZXMoY29sb3IgPSBmYWN0b3IoY3lsKSksCiAgICAgICAgICAgICAgICAgICAgIHNpemUgPSAyLjUpCnAxCmBgYApgYGB7ciBtcGctYW5uLCBlY2hvID0gRkFMU0V9CmBgYAoKQXhpcyBsYWJlbHMgYXJlIGJhc2VkIG9uIHRoZSBleHByZXNzaW9ucyBnaXZlbiB0byBgYWVzYC4KClRoaXMgaXMgY29udmVuaWVudCBmb3IgZXhwbG9yYXRpb24gYnV0IHVzdWFsbHkgbm90IGlkZWFsIGZvciBhIHJlcG9ydC4KClRoZSBgbGFicygpYCBmdW5jdGlvbiBjYW4gYmUgdXNlZCB0byBjaGFuZ2UgYXhpcyBhbmQgbGVnZW5kIGxhYmVsczoKCmBgYHtyIG1wZy1hbm4tbGFicywgZXZhbCA9IEZBTFNFfQpwMSArIGxhYnMoeCA9ICJEaXNwbGFjZW1lbnQgKExpdGVycykiLAogICAgICAgICAgeSA9ICJIaWdod2F5IE1pbGVzIFBlciBHYWxsb24iLAogICAgICAgICAgY29sb3IgPSAiQ3lsaW5kZXJzIikKYGBgCmBgYHtyIG1wZy1hbm4tbGFicywgZWNobyA9IEZBTFNFfQpgYGAKClRoZSBgbGFicygpYCBmdW5jdGlvbiBjYW4gYWxzbyBhZGQgYSB0aXRsZSwgc3VidGl0bGUsIGFuZCBjYXB0aW9uOgoKYGBge3IgbXBnLWFubi1sYWJzLTIsIGV2YWwgPSBGQUxTRX0KcDIgPC0gcDEgKwogICAgbGFicyh4ID0gIkRpc3BsYWNlbWVudCAoTGl0ZXJzKSIsCiAgICAgICAgIHkgPSAiSGlnaHdheSBNaWxlcyBQZXIgR2FsbG9uIiwKICAgICAgICAgY29sb3IgPSAiQ3lsaW5kZXJzIiwKICAgICAgICAgdGl0bGUgPSAiR2FzIE1pbGVhZ2UgYW5kIERpc3BsYWNlbWVudCIsCiAgICAgICAgIHN1YnRpdGxlID0gcGFzdGUoIkZvciBtb2RlbHMgd2hpY2ggaGFkIGEgbmV3IHJlbGVhc2UgZXZlcnkgeWVhciIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICJiZXR3ZWVuIDE5OTkgYW5kIDIwMDgiKSwKICAgICAgICAgY2FwdGlvbiA9ICJEYXRhIFNvdXJjZTogaHR0cHM6Ly9mdWVsZWNvbm9teS5nb3YvIikKcDIKYGBgCmBgYHtyIG1wZy1hbm4tbGFicy0yLCBlY2hvID0gRkFMU0V9CmBgYAoKQW5ub3RhdGlvbnMgY2FuIGJlIHVzZWQgdG8gcHJvdmlkZSBwb3BvdXQgdGhhdCBkcmF3cyBhIHZpZXdlcidzCmF0dGVudGlvbiB0byBwYXJ0aWN1bGFyIGZlYXR1cmVzLgoKVGhlIGBhbm5vdGF0ZSgpYCBmdW5jdGlvbiBpcyBvbmUgb3B0aW9uOgoKYGBge3IgbXBnLWFubi1wb3BvdXQsIGV2YWwgPSBGQUxTRX0KcDIgKwogICAgYW5ub3RhdGUoImxhYmVsIiwgeCA9IDIuOCwgeSA9IDQzLAogICAgICAgICAgICAgbGFiZWwgPSAiVm9sa3N3YWdlbnMiKSArCiAgICBhbm5vdGF0ZSgicmVjdCIsCiAgICAgICAgICAgICB4bWluID0gMS43LCB4bWF4ID0gMi4xLAogICAgICAgICAgICAgeW1pbiA9IDQwLCB5bWF4ID0gNDUsCiAgICAgICAgICAgICBmaWxsID0gTkEsIGNvbG9yID0gImJsYWNrIikKYGBgCmBgYHtyIG1wZy1hbm4tcG9wb3V0LCBlY2hvID0gRkFMU0V9CmBgYAoKT2Z0ZW4gbW9yZSBjb252ZW5pZW50IGFyZSBzb21lIGBnZW9tX21hcmtgIG9iamVjdHMgcHJvdmlkZWQgYnkgdGhlCmBnZ2ZvcmNlYCBwYWNrYWdlOgoKYGBge3IgbXBnLWFubi1wb3BvdXQtMiwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGdnZm9yY2UpCnAyICsKICAgIGdlb21fbWFya19odWxsKGFlcyhmaWx0ZXIgPSBjbGFzcyA9PSAiMnNlYXRlciIpLAogICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb24gPQogICAgICAgICAgICAgICAgICAgICAgIHBhc3RlKCIyLVNlYXRlcnMgaGF2ZSBoaWdoIGRpc3BsYWNlbWVudCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgInZhbHVlcywgYnV0IGFsc28gaGlnaCBmdWVsIGVmZmljaWVuY3kiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICJmb3IgdGhlaXIgZGlzcGxhY2VtZW50LiIpKSArCiAgICBnZW9tX21hcmtfcmVjdChhZXMoZmlsdGVyID0gaHd5ID4gNDApLAogICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb24gPQogICAgICAgICAgICAgICAgICAgICAgICJUaGVzZSBhcmUgVm9sa3N3YWdlbnMiKSArCiAgICBnZW9tX21hcmtfY2lyY2xlKGFlcyhmaWx0ZXIgPSBod3kgPT0gMTIpLAogICAgICAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbiA9CiAgICAgICAgICAgICAgICAgICAgICAgICAiVGhyZWUgcGlja3VwcyBhbmQgYW4gU1VWLiIpCmBgYApgYGB7ciBtcGctYW5uLXBvcG91dC0yLCBlY2hvID0gRkFMU0UsIGZpZy53aWR0aCA9IDcsIGZpZy5oZWlnaHQgPSA1LjV9CiN8IHdhcm5pbmc6IGZhbHNlCmBgYAoKVGhlc2UgYW5ub3RhdGlvbnMgY2FuIGJlIGN1c3RvbWl6ZWQgaW4gYSBudW1iZXIgb2Ygd2F5cy4KCgojIyBBcnJhbmdpbmcgUGxvdHMKClRoZXJlIGFyZSBzZXZlcmFsIHRvb2xzIGF2YWlsYWJsZSBmb3IgYXNzZW1ibGluZyBlbnNlbWJsZSBwbG90cy4KClRoZSBbYHBhdGNod29ya2BdKGh0dHBzOi8vcGF0Y2h3b3JrLmRhdGEtaW1hZ2luaXN0LmNvbS8pIHBhY2thZ2UgaXMgYQpnb29kIGNob2ljZS4KCkEgc2ltcGxlIGV4YW1wbGU6CgpgYGB7ciBtcGctcGF0Y2h3b3JrLCBldmFsID0gRkFMU0V9CnAxIDwtIGdncGxvdChtcGcsIGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICAgICB5ID0gaHd5KSkgKwogICAgZ2VvbV9wb2ludCgpCnAyIDwtIGdncGxvdChtcGcsIGFlcyh4ID0gY3lsLAogICAgICAgICAgICAgICAgICAgICAgeSA9IGh3eSwKICAgICAgICAgICAgICAgICAgICAgIGdyb3VwID0gY3lsKSkgKwogICAgZ2VvbV9ib3hwbG90KCkKcDMgPC0gZ2dwbG90KG1wZywgYWVzKHggPSBjeWwpKSArCiAgICBnZW9tX2JhcigpCgpsaWJyYXJ5KHBhdGNod29yaykKKHAxICsgcDIpIC8gcDMKYGBgCmBgYHtyIG1wZy1wYXRjaHdvcmssIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgQW5pbWF0aW9uCgpUaGUgW2BnZ2FuaW1hdGVgXShodHRwczovL2dpdGh1Yi5jb20vdGhvbWFzcDg1L2dnYW5pbWF0ZSkgcGFja2FnZQpjYW4gYmUgdXNlZCB0byBhZGQgYW5pbWF0aW9uIHRvIGEgYGdncGxvdGAgZ3JhcGguCgpTdGFydCB3aXRoIGEgcGxvdCBgcGAgZm9yIGFsbCB5ZWFycyBpbiB0aGUgYGdhcG1pbmRlcmAgZGF0YSwgd2l0aApgeWVhcmAgaW4gdGhlIGJhY2tncm91bmQ6CgpgYGB7cn0KcCA8LSBnYXBtaW5kZXIgfD4KICAgIGFycmFuZ2UoZGVzYyhwb3ApKSB8PgogICAgZ2dwbG90KGFlcyh4ID0gZ2RwUGVyY2FwLCB5ID0gbGlmZUV4cCkpICsKICAgIGdlb21fdGV4dChhZXMoeCA9IDUwMDAsIHkgPSA1NSwgbGFiZWwgPSBhcy5jaGFyYWN0ZXIoeWVhcikpLAogICAgICAgICAgICAgIHNpemUgPSA1MCwgY29sb3IgPSAiZ3JleSIsCiAgICAgICAgICAgICAgaGp1c3QgPSAiY2VudGVyIiwgdmp1c3QgPSAiY2VudGVyIikgKwogICAgZ2VvbV9wb2ludChhZXMoc2l6ZSA9IHBvcCwgZmlsbCA9IGNvbnRpbmVudCksIHNoYXBlID0gMjEpICsKICAgIHNjYWxlX3hfbG9nMTAobGFiZWxzID0gc2NhbGVzOjpjb21tYSkgKwogICAgeWxpbShjKDIwLCA4NSkpICsKICAgIHNjYWxlX3NpemVfYXJlYShtYXhfc2l6ZSA9IDIwLAogICAgICAgICAgICAgICAgICAgIGxhYmVscyA9IHNjYWxlczo6Y29tbWEsCiAgICAgICAgICAgICAgICAgICAgYnJlYWtzID0gYygwLjI1ICogMTAgXiA5LCAwLjUgKiAxMCBeIDksIDEwIF4gOSkpICsKICAgIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IGMoQWZyaWNhID0gImRlZXBza3libHVlIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQXNpYSA9ICJyZWQiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBBbWVyaWNhcyA9ICJncmVlbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEV1cm9wZSA9ICJnb2xkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgT2NlYW5pYSA9ICJicm93biIpKSArCiAgICBsYWJzKHggPSAiSW5jb21lIiwgeSA9ICJMaWZlIGV4cGVjdGFuY3kiKSArCiAgICB0aGVtZSh0ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxNikpICsKICAgIGd1aWRlcyhmaWxsID0gZ3VpZGVfbGVnZW5kKHRpdGxlID0gIkNvbnRpbmVudCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvdmVycmlkZS5hZXMgPSBsaXN0KHNpemUgPSA1KSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9yZGVyID0gMSksCiAgICAgICAgICAgc2l6ZSA9IGd1aWRlX2xlZ2VuZCh0aXRsZSA9ICJQb3B1bGF0aW9uIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVsLmhqdXN0ID0gMSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9yZGVyID0gMikpICsKICAgIHRoZW1lX21pbmltYWwoKSArCiAgICAgICAgdGhlbWUocGFuZWwuYm9yZGVyID0gZWxlbWVudF9yZWN0KGZpbGwgPSBOQSwgY29sb3IgPSAiZ3JleTIwIikpCmBgYAoKYGBge3IgZ2FwbWluZGVyLWZ1bGwsIGVjaG8gPSBGQUxTRSwgZmlnLmhlaWdodCA9IDYsIGZpZy53aWR0aCA9IDh9CnAKYGBgCgpBIFtHSUZdKGh0dHBzOi8vc2ltcGxlLndpa2lwZWRpYS5vcmcvd2lraS9HcmFwaGljc19JbnRlcmNoYW5nZV9Gb3JtYXQpCmFuaW1hdGlvbjoKCmBgYHtyIGdhcG1pbmRlci1hbmltLCBldmFsID0gRkFMU0V9CmxpYnJhcnkoZ2dhbmltYXRlKQphbmltYXRlKHAgKwogICAgICAgIHRyYW5zaXRpb25fc3RhdGVzKAogICAgICAgICAgICB5ZWFyLAogICAgICAgICAgICB0cmFuc2l0aW9uX2xlbmd0aCA9IDIsCiAgICAgICAgICAgIHN0YXRlX2xlbmd0aCA9IDApKQpgYGAKYGBge3IgZ2FwbWluZGVyLWFuaW0sIGVjaG8gPSBGQUxTRSwgZmlnLmhlaWdodCA9IDYsIGZpZy53aWR0aCA9IDh9CmBgYAoKQSBtb3ZpZToKCmBgYHtyIGdhcG1pbmRlci1hbmltLW1vdmllLCBldmFsID0gRkFMU0V9CmFuaW1hdGUocCArCiAgICAgICAgdHJhbnNpdGlvbl9zdGF0ZXMoCiAgICAgICAgICAgIHllYXIsCiAgICAgICAgICAgIHRyYW5zaXRpb25fbGVuZ3RoID0gMiwKICAgICAgICAgICAgc3RhdGVfbGVuZ3RoID0gMCwKICAgICAgICAgICAgd3JhcCA9IEZBTFNFKSwKICAgICAgICByZW5kZXJlciA9IGZmbXBlZ19yZW5kZXJlcigpKQpgYGAKPGNlbnRlcj4gPCEtLSB0aGVyZSBzaG91bGQvbWF5IGJlIGEgYmV0dGVyIHdheSAtLT4KYGBge3IgZ2FwbWluZGVyLWFuaW0tbW92aWUsIGVjaG8gPSBGQUxTRSwgZmlnLmhlaWdodCA9IDYsIGZpZy53aWR0aCA9IDgsIG91dC53aWR0aCA9ICIxMDAlIn0KYGBgCjwvY2VudGVyPgoKCiMjIEludGVyYWN0aW9uCgoKIyMjIFBsb3RseQoKVGhlIGBnZ3Bsb3RseWAgZnVuY3Rpb24gaW4gdGhlIFtgcGxvdGx5YCBwYWNrYWdlXShodHRwczovL3Bsb3RseS5jb20vci8pCmNhbiBiZSB1c2VkIHRvIGFkZCBzb21lIGludGVyYWN0aXZlIGZlYXR1cmVzIHRvIGEgcGxvdCBjcmVhdGVkIHdpdGgKYGdncGxvdDJgLgoKKiBJbiBhbiBSIHNlc3Npb24gYSBjYWxsIHRvIGBnZ3Bsb3RseSgpYCBvcGVucyBtYXkgb3BlbiBhIGJyb3dzZXIKICB3aW5kb3cgd2l0aCB0aGUgaW50ZXJhY3RpdmUgcGxvdC4KCiogSW4gYW4gUlN0dWRpbyBzZXNzaW9uIHRoZSBwbG90IGFwcGVhcnMgaW4gdGhlIGdyYXBoaWNzIHBhbmVsLgoKKiBJbiBhbiBSbWFya2Rvd24gZG9jdW1lbnQgdGhlIGludGVyYWN0aXZlIHBsb3QgaXMgZW1iZWRkZWQgaW4gdGhlCiAgYGh0bWxgIGZpbGUuCgpBbm90aGVyIGludGVyYWN0aXZlIHBsb3R0aW5nIGFwcHJvYWNoIHRoYXQgY2FuIGJlIHVzZWQgZnJvbSBSIGlzCmRlc2NyaWJlZCBpbiBhbiBbSW5mb3dvcmxkCmFydGljbGVdKGh0dHBzOi8vd3d3LmluZm93b3JsZC5jb20vYXJ0aWNsZS8zNjA3MDY4L3Bsb3QtaW4tci13aXRoLWVjaGFydHM0ci5odG1sKS4KCkEgc2ltcGxlIGV4YW1wbGUgdXNpbmcgYGdncGxvdGx5KClgOgoKYGBge3IgbXBnLXBsb3RseSwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGdncGxvdDIpCmxpYnJhcnkocGxvdGx5KQpwIDwtIGdncGxvdChtdXRhdGUobXBnLCBjeWwgPSBmYWN0b3IoY3lsKSkpICsKICAgIGdlb21fcG9pbnQoYWVzKHggPSBkaXNwbCwKICAgICAgICAgICAgICAgICAgIHkgPSBod3ksCiAgICAgICAgICAgICAgICAgICBmaWxsID0gY3lsKSwKICAgICAgICAgICAgICAgc2hhcGUgPSAyMSwKICAgICAgICAgICAgICAgc2l6ZSA9IDMpCmdncGxvdGx5KHApCmBgYApgYGB7ciBtcGctcGxvdGx5LCBlY2hvID0gRkFMU0UsIG1lc3NhZ2UgPSBGQUxTRX0KYGBgCgpBZGRpbmcgYSBgdGV4dGAgYWVzdGhldGljIGFsbG93cyB0aGUgdG9vbHRpcCBkaXNwbGF5IHRvIGJlIGN1c3RvbWl6ZWQ6CgpgYGB7ciBtcGctcGxvdGx5LTIsIGV2YWwgPSBGQUxTRX0KcCA8LSBnZ3Bsb3QobXV0YXRlKG1wZywgY3lsID0gZmFjdG9yKGN5bCkpKSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgICAgICAgICB5ID0gaHd5LAogICAgICAgICAgICAgICAgICAgZmlsbCA9IGN5bCwKICAgICAgICAgICAgICAgICAgIHRleHQgPSBwYXN0ZSh5ZWFyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1hbnVmYWN0dXJlciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtb2RlbCkpLAogICAgICAgICAgICAgICBzaGFwZSA9IDIxLAogICAgICAgICAgICAgICBzaXplID0gMykKZ2dwbG90bHkocCwgdG9vbHRpcCA9ICJ0ZXh0IikgfD4KICAgIHN0eWxlKGhvdmVybGFiZWwgPSBsaXN0KGJnY29sb3IgPSAid2hpdGUiKSkKYGBgCmBgYHtyIG1wZy1wbG90bHktMiwgZWNobyA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0UsIG1lc3NhZ2UgPSBGQUxTRX0KYGBgCgoKIyMjIEdnaXJhcGgKClRoZSBbYGdnaXJhcGhgIHBhY2thZ2VdKGh0dHBzOi8vZGF2aWRnb2hlbC5naXRodWIuaW8vZ2dpcmFwaC8pCnByb3ZpZGVzIGFub3RoZXIgYXBwcm9hY2guCgpgYGB7ciBtcGctZ2dpcmFwaCwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGdncGxvdDIpCmxpYnJhcnkoZ2dpcmFwaCkKcCA8LSBnZ3Bsb3QobXV0YXRlKG1wZywgY3lsID0gZmFjdG9yKGN5bCkpKSArCiAgICBnZW9tX3BvaW50X2ludGVyYWN0aXZlKAogICAgICAgIGFlcyh4ID0gZGlzcGwsCiAgICAgICAgICAgIHkgPSBod3ksCiAgICAgICAgICAgIGZpbGwgPSBjeWwsCiAgICAgICAgICAgIHRvb2x0aXAgPSBwYXN0ZSh5ZWFyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFudWZhY3R1cmVyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWwpKSwKICAgICAgICBzaGFwZSA9IDIxLAogICAgICAgIHNpemUgPSAzKQpnaXJhZmUoZ2dvYmogPSBwKQpgYGAKCmBgYHtyIG1wZy1nZ2lyYXBoLCBlY2hvID0gRkFMU0V9CmBgYAoKCiMjIyBHcmFtbWFyIG9mIEludGVyYWN0aXZlIEdyYXBoaWNzCgpUaGVyZSBoYXZlIGJlZW4gc2V2ZXJhbCBlZmZvcnRzIHRvIGRldmVsb3AgYSBncmFtbWFyIG9mIGludGVyYWN0aXZlCmdyYXBoaWNzLCBpbmNsdWRpbmcgW2BnZ3Zpc2BdKGh0dHBzOi8vZ2d2aXMucnN0dWRpby5jb20vKSBhbmQKW2BhbmltaW50YF0oaHR0cHM6Ly90ZGhvY2suZ2l0aHViLmlvL2FuaW1pbnQvKTsgbmVpdGhlciBzZWVtcyB0byBiZQp1bmRlciBhY3RpdmUgZGV2ZWxvcG1lbnQgYXQgdGhpcyB0aW1lLgoKQSBwcm9taXNpbmcgYXBwcm9hY2ggaXMKW1ZlZ2EtTGl0ZV0oaHR0cHM6Ly92ZWdhLmdpdGh1Yi5pby92ZWdhLWxpdGUvKSwgd2l0aCBhIFB5dGhvbgppbnRlcmZhY2UgW0FsdGFpcl0oaHR0cHM6Ly9hbHRhaXItdml6LmdpdGh1Yi5pby8pIGFuZCBhbiBSIGludGVyZmFjZQpbYWx0YWlyXShodHRwczovL3ZlZ2F3aWRnZXQuZ2l0aHViLmlvL2FsdGFpci8pIHRvIHRoZSBQeXRob24KaW50ZXJmYWNlLgoKQW4gZXhhbXBsZSB1c2luZyB0aGUgYGFsdGFpcmAgcGFja2FnZToKCmBgYHtyIHJ1YmJlci1hbHRhaXIsIGV2YWwgPSBGQUxTRX0KcnViIDwtIHJlYWQuY3N2KGhlcmU6OmhlcmUoInJ1YmJlci5jc3YiKSkKCmxpYnJhcnkoYWx0YWlyKQoKY2hhcnRUSCA8LSBhbHQkQ2hhcnQocnViKSQKICAgIG1hcmtfcG9pbnQoKSQKICAgIGVuY29kZSh4ID0gYWx0JFgoIkg6USIsIHNjYWxlID0gYWx0JFNjYWxlKGRvbWFpbiA9IHJhbmdlKHJ1YiRIKSkpLAogICAgICAgICAgIHkgPSBhbHQkWSgiVDpRIiwgc2NhbGUgPSBhbHQkU2NhbGUoZG9tYWluID0gcmFuZ2UocnViJFQpKSkpCgpicnVzaCA8LSBhbHQkc2VsZWN0aW9uX2ludGVydmFsKCkKCmNoYXJ0VEhfYnJ1c2ggPC0gY2hhcnRUSCRhZGRfc2VsZWN0aW9uKGJydXNoKQoKY2hhcnRUSF9zZWxlY3Rpb24gPC0KICAgIGNoYXJ0VEhfYnJ1c2gkZW5jb2RlKGNvbG9yID0gYWx0JGNvbmRpdGlvbihicnVzaCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiT3JpZ2luOk4iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFsdCR2YWx1ZSgibGlnaHRncmF5IikpKQoKY2hhcnRBVCA8LSBjaGFydFRIX3NlbGVjdGlvbiQKICAgIGVuY29kZSh4ID0gYWx0JFgoIlQ6USIsIHNjYWxlID0gYWx0JFNjYWxlKGRvbWFpbiA9IHJhbmdlKHJ1YiRUKSkpLAogICAgICAgICAgIHkgPSBhbHQkWSgiQTpRIiwgc2NhbGUgPSBhbHQkU2NhbGUoZG9tYWluID0gcmFuZ2UocnViJEEpKSkpCgpjaGFydEFUIHwgY2hhcnRUSF9zZWxlY3Rpb24KYGBgCgpUaGUgcmVzdWx0aW5nIGxpbmtlZCBwbG90czoKCmBgYHtyIHJ1YmJlci1hbHRhaXIsIGVjaG8gPSBGQUxTRSwgZXJyb3IgPSBUUlVFLCB3YXJuaW5nID0gRkFMU0V9CmBgYAoKCiMjIE5vdGVzCgoqIEEgbnVtYmVyIG9mIG90aGVyIFtgZ2dwbG90YAogIGV4dGVuc2lvbnNdKGh0dHBzOi8vZXh0cy5nZ3Bsb3QyLnRpZHl2ZXJzZS5vcmcvKSBhcmUgYXZhaWxhYmxlLgoKKiBBIFtibG9nCiAgcG9zdF0oaHR0cHM6Ly9tZWRpdW0uY29tL2JiYy12aXN1YWwtYW5kLWRhdGEtam91cm5hbGlzbS9ob3ctdGhlLWJiYy12aXN1YWwtYW5kLWRhdGEtam91cm5hbGlzbS10ZWFtLXdvcmtzLXdpdGgtZ3JhcGhpY3MtaW4tci1lZDBiMzU2OTM1MzUpCiAgZXhwbGFpbnMgaG93IHRoZSBbQkJDIFZpc3VhbCBhbmQgRGF0YQogIEpvdXJuYWxpc21dKGh0dHBzOi8vbWVkaXVtLmNvbS9iYmMtdmlzdWFsLWFuZC1kYXRhLWpvdXJuYWxpc20pIHRlYW0KICBjcmVhdGVzIHRoZWlyIGdyYXBoaWNzLiBNb3JlIGRldGFpbHMgYXJlIHByb3ZpZGVkIGluIGFuIFtfUiBjb29rCiAgYm9va19dKGh0dHBzOi8vYmJjLmdpdGh1Yi5pby9yY29va2Jvb2svKS4KCiogQSBbYmxvZwogIHBvc3RdKGh0dHBzOi8vYmxvZy5yZXZvbHV0aW9uYW5hbHl0aWNzLmNvbS8yMDE2LzA3L2RhdGEtam91cm5hbGlzbS13aXRoLXItYXQtNTM4Lmh0bWwpCiAgZGVzY3JpYmVzIHRoZSB1c2Ugb2YgUiBhbmQgYGdncGxvdGAgYnkKICBbRml2ZVRoaXJ0eUVpZ2h0XShodHRwczovL2ZpdmV0aGlydHllaWdodC5jb20vKS4gIFRoZSBgZ2d0aGVtZXNgCiAgcGFja2FnZXMgaW5jbHVkZXMgYHRoZW1lX2ZpdmV0aGlydHllaWdodGAgdG8gZW11bGF0ZSB0aGVpciBzdHlsZS4KCgojIyBSZWFkaW5nCgpDaGFwdGVycyBbX0RhdGEKdmlzdWFsaXphdGlvbl9dKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovZGF0YS12aXN1YWxpemUuaHRtbCkgYW5kCltfR3JhcGhpY3MgZm9yCmNvbW11bmljYXRpb25fXShodHRwczovL3I0ZHMuaGFkbGV5Lm56L2NvbW11bmljYXRpb24uaHRtbCkKaW4gW19SIGZvciBEYXRhIFNjaWVuY2VfXShodHRwczovL3I0ZHMuaGFkbGV5Lm56LyksIE8nUmVpbGx5LgoKQ2hhcHRlciBbX01ha2UgYSBwbG90X10oaHR0cHM6Ly9zb2N2aXouY28vbWFrZXBsb3QuaHRtbCkgaW4gW19EYXRhClZpc3VhbGl6YXRpb25fXShodHRwczovL3NvY3Zpei5jby8pLgoKQ2hhcHRlcgpbX2dncGxvdDJfXShodHRwczovL3JhZmFsYWIuZGZjaS5oYXJ2YXJkLmVkdS9kc2Jvb2stcGFydC0xL2RhdGF2aXovZ2dwbG90Mi5odG1sKQppbiBbX0ludHJvZHVjdGlvbiB0byBEYXRhIFNjaWVuY2U6IERhdGEgQW5hbHlzaXMgYW5kIFByZWRpY3Rpb24KQWxnb3JpdGhtcyB3aXRoIFJfXShodHRwczovL3JhZmFsYWIuZGZjaS5oYXJ2YXJkLmVkdS9kc2Jvb2stcGFydC0xLykuCgoKIyMgSW50ZXJhY3RpdmUgVHV0b3JpYWwKCkFuIGludGVyYWN0aXZlIFtgbGVhcm5yYF0oaHR0cHM6Ly9yc3R1ZGlvLmdpdGh1Yi5pby9sZWFybnIvKSB0dXRvcmlhbApmb3IgdGhlc2Ugbm90ZXMgaXMgW2F2YWlsYWJsZV0oYHIgV0xOSygidHV0b3JpYWxzL2dncGxvdC5SbWQiKWApLgoKWW91IGNhbiBydW4gdGhlIHR1dG9yaWFsIHdpdGgKCmBgYHtyLCBldmFsID0gRkFMU0V9ClNUQVQ0NTgwOjpydW5UdXRvcmlhbCgiZ2dwbG90IikKYGBgCgpZb3UgY2FuIGluc3RhbGwgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgYFNUQVQ0NTgwYCBwYWNrYWdlIHdpdGgKCmBgYHtyLCBldmFsID0gRkFMU0V9CnJlbW90ZXM6Omluc3RhbGxfZ2l0bGFiKCJsdWtlLXRpZXJuZXkvU1RBVDQ1ODAiKQpgYGAKCllvdSBtYXkgbmVlZCB0byBpbnN0YWxsIHRoZSBgcmVtb3Rlc2AgcGFja2FnZSBmcm9tIENSQU4gZmlyc3QuCgoKIyMgRXhlcmNpc2VzCgoxLiBJbiB0aGUgZm9sbG93aW5nIGV4cHJlc3Npb24sIHdoaWNoIHZhbHVlIG9mIHRoZSBgc2hhcGVgIGFlc3RoZXRpYwogICBwcm9kdWNlcyBhIHBsb3Qgd2l0aCBwb2ludHMgcmVwcmVzZW50ZWQgYXMgdHJpYW5nbGVzIG91dGxpbmVkIGluCiAgIGJsYWNrIGNvbG9yZWQgYWNjb3JkaW5nIHRvIHRoZSBudW1iZXIgb2YgY3lsaW5kZXJzPwoKPCEtLSAjIyBub2xpbnQgc3RhcnQgLS0+CiAgICBgYGByCiAgICBsaWJyYXJ5KGdncGxvdDIpCiAgICBnZ3Bsb3QobXBnLCBhZXMoeCA9IGRpc3BsLCB5ID0gaHd5LCBmaWxsID0gZmFjdG9yKGN5bCkpKSArCiAgICAgICAgZ2VvbV9wb2ludChzaXplID0gNCwgc2hhcGUgPSAtLS0pCiAgICBgYGAKPCEtLSAjIyBub2xpbnQgZW5kIC0tPgoKICAgIGEuIDE1CiAgICBiLiAxNwogICAgYy4gMjEKICAgIGQuIDI0CgoyLiBJdCBjYW4gc29tZXRpbWVzIGJlIHVzZWZ1bCB0byBwbG90IHRleHQgbGFiZWxzIGluIGEgc2NhdHRlcnBsb3QKICAgaW5zdGVhZCBvZiBwb2ludHMuIENvbnNpZGVyIHRoZSBwbG90IHNldCB1cCBhcwoKICAgIGBgYHIKICAgIGxpYnJhcnkoZ2dwbG90MikKICAgIGxpYnJhcnkoZHBseXIpCiAgICBkYXRhKGdhcG1pbmRlciwgcGFja2FnZSA9ICJnYXBtaW5kZXIiKQogICAgcCA8LSBmaWx0ZXIoZ2FwbWluZGVyLCB5ZWFyID09IDIwMDcpIHw+CiAgICAgICAgZ3JvdXBfYnkoY29udGluZW50KSB8PgogICAgICAgIHN1bW1hcml6ZShnZHBQZXJjYXAgPSBtZWFuKGdkcFBlcmNhcCksIGxpZmVFeHAgPSBtZWFuKGxpZmVFeHApKSB8PgogICAgICAgIGdncGxvdChhZXMoeCA9IGdkcFBlcmNhcCwgeSA9IGxpZmVFeHApKQogICAgYGBgCiAgICBXaGljaCBvZiB0aGUgZm9sbG93aW5nIHByb2R1Y2VzIGEgcGxvdCB3aXRoIGNvbnRpbmVudAogICAgbmFtZXMgb24gd2hpdGUgcmVjdGFuZ2xlcz8KCiAgICBhLiBgcCArIGdlb21fdGV4dChhZXMobGFiZWwgPSBjb250aW5lbnQpKWAKICAgIGIuIGBwICsgZ2VvbV9sYWJlbChhZXMobGFiZWwgPSBjb250aW5lbnQpKWAKICAgIGMuIGBwICsgZ2VvbV9sYWJlbChsYWJlbCA9IGNvbnRpbmVudClgCiAgICBkLiBgcCArIGdlb21fdGV4dCh0ZXh0ID0gY29udGluZW50KWAKCjMuIFRoZSBmb2xsb3dpbmcgY29kZSBwbG90cyBhIF9rZXJuZWwgZGVuc2l0eSBlc3RpbWF0ZV8gZm9yIHRoZQogICBgZXJ1cHRpb25zYCB2YXJpYWJsZSBpbiB0aGUgYGZhaXRoZnVsYCBkYXRhIHNldDoKCiAgICBgYGByCiAgICBsaWJyYXJ5KGdncGxvdDIpCiAgICBnZ3Bsb3QoZmFpdGhmdWwsIGFlcyh4ID0gZXJ1cHRpb25zKSkgKyBnZW9tX2RlbnNpdHkoYncgPSAwLjEpCiAgICBgYGAKICAgIExvb2sgYXQgdGhlIGhlbHAgcGFnZSBmb3IgYGdlb21fZGVuc2l0eWAuIFdoaWNoIG9mIHRoZSBmb2xsb3dpbmcgYmVzdAogICAgZGVzY3JpYmVzIHdoYXQgc3BlY2lmeWluZyBhIHZhbHVlIGZvciBgYndgIGRvZXM6CgogICAgYS4gQ2hhbmdlcyB0aGUgX2tlcm5lbF8gdXNlZCB0byBjb25zdHJ1Y3QgdGhlIGVzdGltYXRlLgogICAgYi4gQ2hhbmdlcyB0aGUgX3Ntb290aGluZyBiYW5kd2lkdGhfIHRvIG1ha2UgdGhlIHJlc3VsdCBtb3JlIG9yIGxlc3Mgc21vb3RoLgogICAgYy4gQ2hhbmdlcyB0aGUgYHN0YXRgIHVzZWQgdG8gYHN0YXRfYndgLgogICAgZC4gSGFzIG5vIGVmZmVjdCBvbiB0aGUgcmV0dWx0LgoKNC4gVGhpcyBjb2RlIGNyZWF0ZXMgYSBtYXAgb2YgSW93YSBjb3VudGllcy4KCiAgICBgYGByCiAgICBsaWJyYXJ5KGdncGxvdDIpCiAgICBwIDwtIGdncGxvdChtYXBfZGF0YSgiY291bnR5IiwgImlvd2EiKSwKICAgICAgICAgICAgICAgIGFlcyh4ID0gbG9uZywgeSA9IGxhdCwgZ3JvdXAgPSBncm91cCkpICsKICAgICAgICBnZW9tX3BvbHlnb24oLCBmaWxsID0gIldoaXRlIiwgY29sb3IgPSAiYmxhY2siKQogICAgYGBgCiAgIAogICAgV2hpY2ggb2YgdGhlc2UgcHJvZHVjZXMgYSBwbG90IHdpdGggYW4gYXNwZWN0IHJhdGlvIHRoYXQgYmVzdAogICAgbWF0Y2hlcyB0aGUgbWFwIG9uIFt0aGlzCiAgICBwYWdlXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvdy9pbmRleC5waHA/dGl0bGU9TGlzdF9vZl9jb3VudGllc19pbl9Jb3dhJm9sZGlkPTEwMDExNzEwODIpPwoKICAgIGEuIGBwICsgY29vcmRfZml4ZWQoMC41KWAgCiAgICBiLiBgcCArIGNvb3JkX2ZpeGVkKDAuNzUpYAogICAgYy4gYHAgKyBjb29yZF9maXhlZCgxLjM1KWAKICAgIGQuIGBwICsgY29vcmRfZml4ZWQoMS45NSlgCgo1LiBDb25zaWRlciB0aGUgdHdvIHBsb3RzIGNyZWF0ZWQgYnkgdGhpcyBjb2RlIChwcmludCB0aGUgdmFsdWVzIG9mCiAgIGBwMWAgYW5kIGBwMmAgdG8gc2VlIHRoZSBwbG90cyk6CgogICAgYGBgcgogICAgbGlicmFyeShnZ3Bsb3QyKQogICAgZGF0YShnYXBtaW5kZXIsIHBhY2thZ2UgPSAiZ2FwbWluZGVyIikKICAgIHAxIDwtIGdncGxvdChnYXBtaW5kZXIsIGFlcyh4ID0gbG9nKGdkcFBlcmNhcCksIHkgPSBsaWZlRXhwKSkgKwogICAgICAgIGdlb21fcG9pbnQoKSArCiAgICAgICAgc2NhbGVfeF9jb250aW51b3VzKG5hbWUgPSAiIikKICAgIHAyIDwtIGdncGxvdChnYXBtaW5kZXIsIGFlcyh4ID0gZ2RwUGVyY2FwLCB5ID0gbGlmZUV4cCkpICsKICAgICAgICBnZW9tX3BvaW50KCkgKwogICAgICAgIHNjYWxlX3hfbG9nMTAobGFiZWxzID0gc2NhbGVzOjpjb21tYSwgbmFtZSA9ICIiKSAKICAgIGBgYAoKICAgIFdoaWNoIG9mIHRoZXNlIHN0YXRlbWVudHMgaXMgdHJ1ZT8KCiAgICBhLiBUaGUgYHhgIGF4aXMgbGFiZWxzIGFyZSBpZGVudGljYWwgaW4gYm90aCBwbG90cy4KICAgIGIuIFRoZSBgeGAgYXhpcyBsYWJlbHMgaW4gYHAyYCBhcmUgaW4gZG9sbGFyczsgdGhlIGxhYmVscyBpbiBgcDFgCiAgICAgICBhcmUgaW4gbG9nIGRvbGxhcnMuIAogICAgYy4gVGhlIGB4YCBheGlzIGxhYmVscyBpbiBgcDFgIGFyZSBpbiBkb2xsYXJzOyB0aGUgbGFiZWxzIGluIGBwMmAKICAgICAgIGFyZSBpbiBsb2cgZG9sbGFycy4KICAgIGQuIFRoZXJlIGFyZSBubyBsYWJlbHMgb24gdGhlIGB4YCBheGlzIGluIGBwMmAuCgo2LiBDb25zaWRlciB0aGUgcGxvdCBjcmVhdGVkIGJ5CgogICAgYGBgcgogICAgbGlicmFyeShnZ3Bsb3QyKQogICAgZGF0YShnYXBtaW5kZXIsIHBhY2thZ2UgPSAiZ2FwbWluZGVyIikKICAgIHAgPC0gZ2dwbG90KGdhcG1pbmRlciwgYWVzKHggPSBnZHBQZXJjYXAsIHkgPSBsaWZlRXhwKSkgKwogICAgICAgIGdlb21fcG9pbnQoKSArCiAgICAgICAgc2NhbGVfeF9sb2cxMChsYWJlbHMgPSBzY2FsZXM6OmNvbW1hKSAKICAgIGBgYAoKICAgIFdoaWNoIG9mIHRoZXNlIGV4cHJlc3Npb25zIHByb2R1Y2VzIGEgcGxvdCB3aXRoIGEgd2hpdGUgYmFja2dyb3VuZD8KCiAgICBhLiBgcGAKICAgIGIuIGBwICsgdGhlbWVfZ3JleSgpYAogICAgYy4gYHAgKyB0aGVtZV9jbGFzc2ljKClgCiAgICBkLiBgcCArIGdndGhlbWVzOjp0aGVtZV9lY29ub21pc3QoKWAKCjcuIFRoZXJlIGFyZSBtYW55IGRpZmZlcmVudCB3YXlzIHRvIGNoYW5nZSB0aGUgYHhgIGF4aXMgbGFiZWwgaW4KICAgYGdncGxvdGAuICBDb25zaWRlciB0aGUgcGxvdCBjcmVhdGVkIGJ5CgogICAgYGBgcgogICAgbGlicmFyeShnZ3Bsb3QyKQogICAgcCA8LSBnZ3Bsb3QobXBnLCBhZXMoeCA9IGRpc3BsLCB5ID0gaHd5KSkgKwogICAgICAgIGdlb21fcG9pbnQoKQogICAgYGBgCgogICAgV2hpY2ggb2YgdGhlIGZvbGxvd2luZyBkb2VzICoqbm90KiogY2hhbmdlIHRoZSBgeGAgYXhpcyBsYWJlbCB0bwogICAgX0Rpc3BsYWNlbWVudF8/CgogICAgYS4gYHAgKyBsYWJzKHggPSAiRGlzcGxhY2VtZW50IilgCiAgICBiLiBgcCArIHNjYWxlX3hfY29udGludW91cygiRGlzcGxhY2VtZW50IilgCiAgICBjLiBgcCArIHhsYWIoIkRpc3BsYWNlbWVudCIpYAogICAgZC4gYHAgKyB0aGVtZShheGlzLnRpdGxlLnggPSAiRGlzcGxhY2VtZW50IilgCgo=