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
a data set;
one or more geometrical representation (geoms );
mappings of values to aesthetic features of the geom;
a stat to produce values to be mapped;
position adjustments;
a coordinate system;
a scale specification;
a faceting scheme.
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:
Too many colors;
shapes are too small;
interference between shapes and colors.
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.
For geom_point
the default stat is stat_identity
.
For geom_bar
the default stat is stat_count
.
For geom_histogram
the default stat is stat_bin
.
Stats can provide computed variables that can be mapped to aesthetic features.
For stat_bin
some of the computed variables are
count
: number of points in bin
density
: density of points in bin, scaled to integrate to 1
The density
variable can be accessed as after_stat(dentity)
.
Older approaches that also work but are now discouraged:
stat(dentity)
..density..
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
axis labels and marks;
color or shape legends.
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:
theme()
can be used to adjust individual elements in a plot.
theme_set()
adjusts default settings for a session;
pre-defined theme functions allow consistent style changes.
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:
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
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
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?
p + geom_text(aes(label = continent))
p + geom_label(aes(label = continent))
p + geom_label(label = continent)
p + geom_text(text = continent)
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:
Changes the kernel used to construct the estimate.
Changes the smoothing bandwidth to make the result more or less smooth.
Changes the stat
used to stat_bw
.
Has no effect on the retult.
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 ?
p + coord_fixed(0.5)
p + coord_fixed(0.75)
p + coord_fixed(1.35)
p + coord_fixed(1.95)
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?
The x
axis labels are identical in both plots.
The x
axis labels in p2
are in dollars; the labels in p1
are in log dollars.
The x
axis labels in p1
are in dollars; the labels in p2
are in log dollars.
There are no labels on the x
axis in p2
.
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?
p
p + theme_grey()
p + theme_classic()
p + ggthemes::theme_economist()
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 ?
p + labs(x = "Displacement")
p + scale_x_continuous("Displacement")
p + xlab("Displacement")
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=