1 Goals for Lab 01

  • Get your feet wet!
  • Innoculate you against ggplot2 errors- we all get them!
  • Get exposed to the range of things you can do, before we go deeep
  • Develop your own personal preferences for data visualizations!
    • Do you like or hate gridlines?
    • What fonts do you find pleasant to read?
    • What kinds of colors do you like?
    • Are you team theme_gray or theme_bw (or theme_minimal)?

These are important questions, and I want you to develop (well-informed) opinions on these matters!

2 Nathan’s Hot Dog Eating Contest

This includes a reconstruction of Nathan Yau’s hot dog contest example, as interpreted by Jackie Wirz, ported into R and ggplot2 by Steven Bedrick for a workshop for the OHSU Data Science Institute, and finally adapted by Alison Hill for all you intrepid Data-Viz-onauts!

First, we load our packages:

library(tidyverse)
library(extrafont)
library(here)

3 Read in and wrangle data

Next, we load some data. In the RStudio Cloud project for this lab, you will see a data directory with the necessary files. We can read it in using read_csv, and along the way use col_factor to tell it how to handle the gender column.

hot_dogs <- read_csv(here::here("data", "hot_dog_contest.csv"), 
    col_types = cols(
      gender = col_factor(levels = NULL)
    ))

Note that if the data were not already downloaded locally, read_csv() can load CSV files from a URL:

hot_dogs <- read_csv("http://bit.ly/cs631-hotdog", 
    col_types = cols(
      gender = col_factor(levels = NULL)
    ))

Either way you do it, check it out once read in and make sure it looks like this!

glimpse(hot_dogs)
Observations: 49
Variables: 4
$ year      <dbl> 2017, 2017, 2016, 2016, 2015, 2015, 2014, 2014, 2013, 2013,…
$ gender    <fct> male, female, male, female, male, female, male, female, mal…
$ name      <chr> "Joey Chestnut", "Miki Sudo", "Joey Chestnut", "Miki Sudo",…
$ num_eaten <dbl> 72.000, 41.000, 70.000, 38.000, 62.000, 38.000, 61.000, 34.…
hot_dogs
# A tibble: 49 x 4
    year gender name           num_eaten
   <dbl> <fct>  <chr>              <dbl>
 1  2017 male   Joey Chestnut       72  
 2  2017 female Miki Sudo           41  
 3  2016 male   Joey Chestnut       70  
 4  2016 female Miki Sudo           38  
 5  2015 male   Matthew Stonie      62  
 6  2015 female Miki Sudo           38  
 7  2014 male   Joey Chestnut       61  
 8  2014 female Miki Sudo           34  
 9  2013 male   Joey Chestnut       69  
10  2013 female Sonya Thomas        36.8
# … with 39 more rows

We’ll be wanting to somehow include information about whether a given year was before or after the incorporation of the competitive eating league, so let’s add an indicator field to the data using mutate(). Also, the data’s a little sketchy pre-1981 and for our purposes today we’ll be focusing on males only, so let’s do some filtering too:

hot_dogs <- hot_dogs %>% 
  mutate(post_ifoce = year >= 1997) %>% 
  filter(year >= 1981 & gender == 'male')
hot_dogs
# A tibble: 37 x 5
    year gender name           num_eaten post_ifoce
   <dbl> <fct>  <chr>              <dbl> <lgl>     
 1  2017 male   Joey Chestnut         72 TRUE      
 2  2016 male   Joey Chestnut         70 TRUE      
 3  2015 male   Matthew Stonie        62 TRUE      
 4  2014 male   Joey Chestnut         61 TRUE      
 5  2013 male   Joey Chestnut         69 TRUE      
 6  2012 male   Joey Chestnut         68 TRUE      
 7  2011 male   Joey Chestnut         62 TRUE      
 8  2010 male   Joey Chestnut         54 TRUE      
 9  2009 male   Joey Chestnut         68 TRUE      
10  2008 male   Joey Chestnut         59 TRUE      
# … with 27 more rows

4 Plot The Data

Now let’s try making a first crack at a sketchy plot:

ggplot(hot_dogs, aes(x = year, y = num_eaten)) + 
  geom_col()

Note that our data is already in “counted” form, so we’re using geom_col() instead of geom_bar().

5 Add Axis Labels And Title

ggplot(hot_dogs, aes(x = year, y = num_eaten)) + 
  geom_col() +
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017")

6 Play With Colors

Challenge #1:

Make 3 versions of the last plot we just made:

  • In the first, make all the columns outlined in “white”.

  • In the second, make all the columns outlined in “white” and filled in “navyblue”.

  • In the third, make all the columns outlined in “white” and filled in according to whether or not post_ifoce is TRUE or FALSE (use default colors for now).

ggplot(hot_dogs, aes(x = year, y = num_eaten)) + 
  geom_col(colour = "white") + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017")

ggplot(hot_dogs, aes(x = year, y = num_eaten)) + 
  geom_col(colour = "white", fill = "navyblue") + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017")

ggplot(hot_dogs, aes(x = year, y = num_eaten)) + 
  geom_col(aes(fill = post_ifoce), colour = "white") + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017")

Challenge #2:

What if you want to change the legend in the last plot you made? Use google to figure out how to do the following:

  • Delete the legend title

  • Make the legend text either “Post-IFOCE” or “Pre-IFOCE”.

ggplot(hot_dogs, aes(x = year, y = num_eaten)) + 
  geom_col(aes(fill = post_ifoce), colour = "white") + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017") +
  scale_fill_discrete(name = "",
                      labels=c("Pre-IFOCE", "Post-IFOCE"))

7 Change The Dataset

Now, let’s change the question a little bit. This looks at the creation of the IFOCE. What about the affiliation of the contestants? We’ll need some different data for this. Through the Magic Of Data Science™, we have dug that information up and put it into an expanded version of our CSV file available at http://bit.ly/cs631-hotdog-affiliated.

Challenge #3:

Let’s work with this new dataset! Do the following:

  • Read in the “hot_dog_contest_with_affiliation.csv” data file, using col_types to read in affiliated and gender as factors.

  • Within a mutate, create a new variable called post_ifoce that is TRUE if year is greater than or equal to 1997.

  • Also filter the new data for only years 1981 and after, and only for male competitors.

hdm_affil <- read_csv(here::here("data", "hot_dog_contest_with_affiliation.csv"), 
    col_types = cols(
      affiliated = col_factor(levels = NULL), 
      gender = col_factor(levels = NULL)
      )) %>% 
  mutate(post_ifoce = year >= 1997) %>% 
  filter(year >= 1981 & gender == "male") 
hdm_affil <- read_csv(here::here("data", "hot_dog_contest_with_affiliation.csv"), 
    col_types = cols(
      affiliated = col_factor(levels = NULL), 
      gender = col_factor(levels = NULL)
      )) %>% 
  mutate(post_ifoce = year >= 1997) %>% 
  filter(year >= 1981 & gender == "male") 
glimpse(hdm_affil)
Observations: 37
Variables: 6
$ year       <dbl> 2017, 2016, 2015, 2014, 2013, 2012, 2011, 2010, 2009, 2008…
$ gender     <fct> male, male, male, male, male, male, male, male, male, male…
$ name       <chr> "Joey Chestnut", "Joey Chestnut", "Matthew Stonie", "Joey …
$ num_eaten  <dbl> 72.000, 70.000, 62.000, 61.000, 69.000, 68.000, 62.000, 54…
$ affiliated <fct> current, current, current, current, current, current, curr…
$ post_ifoce <lgl> TRUE, TRUE, TRUE, TRUE, TRUE, TRUE, TRUE, TRUE, TRUE, TRUE…
Challenge #4:

Let’s do some basic EDA with this new dataset! Do the following:

  • Use dplyr::distinct to figure out how many unique values there are of affiliated.

  • Use dplyr::count to count the number of rows for each unique value of affiliated; use ?count to figure out how to sort the counts in descending order.

hdm_affil %>% 
  distinct(affiliated)
# A tibble: 3 x 1
  affiliated    
  <fct>         
1 current       
2 former        
3 not affiliated
hdm_affil %>% 
  count(affiliated, sort = TRUE)
# A tibble: 3 x 2
  affiliated         n
  <fct>          <int>
1 not affiliated    20
2 current           11
3 former             6

Now let’s plot this new data, and fill the columns according to our new affiliated column.

ggplot(hdm_affil, aes(x = year, y = num_eaten)) + 
  geom_col(aes(fill = affiliated)) + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017")

Challenge #5:

Do the following updates to the last plot we just made:

  • Update the colors using hex colors: c('#E9602B','#2277A0','#CCB683').

  • Change the legend title to “IFOCE-affiliation”.

  • Save this plot object as “affil_plot”.

affil_plot <- ggplot(hdm_affil, aes(x = year, y = num_eaten)) + 
  geom_col(aes(fill = affiliated)) + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017") +
  scale_fill_manual(values = c('#E9602B','#2277A0','#CCB683'),
                    name = "IFOCE-affiliation")
affil_plot

8 Play With Scales & Coordinates

The spacing’s a little funky down near the origin of the plot. The documentation tells us that the defaults are c(0.05, 0) for continuous variables. The first number is multiplicative and the second is additive.

The default was that 1.8 ((2017-1981)*.05+0) was added to the right and left sides of the x-axis as padding, so the effective default limits were c(1979, 2019).

Let’s tighten that up with the expand property for the scale_y_continuous (we’ll also change the breaks for y-axis tick marks here) and scale_x_continuous settings:

affil_plot <- affil_plot + 
  scale_y_continuous(expand = c(0, 0),
                     breaks = seq(0, 70, 10)) +
  scale_x_continuous(expand = c(0, 0))
affil_plot

But now the plot looks like it is wearing tight pants.

Let’s loosen things up a bit by updating the plot coordinates.

Challenge #6:

Use coord_cartesian to:

  • Set the x-axis range to 1980-2018

  • Set the y-axis range to 0-80

Using coord_cartesian is the preferred layer here because “setting limits on the coordinate system will zoom the plot (like you’re looking at it with a magnifying glass), and will not change the underlying data like setting limits on a scale will.”

Lesson:
Don’t change limits unless you really know what you are doing! Most of the time, you want to change the coordinates instead.
affil_plot <- affil_plot + 
  coord_cartesian(xlim = c(1980, 2018), ylim = c(0, 80)) 
affil_plot

9 Play With Theme Settings

Let’s change some key theme settings:

affil_plot +
  theme(plot.title = element_text(hjust = 0.5)) +
  theme(axis.text = element_text(size = 12)) +
  theme(panel.background = element_blank()) +
  theme(axis.line.x = element_line(color = "gray80", size = 0.5)) +
  theme(axis.ticks = element_line(color = "gray80", size = 0.5))

Lesson:
You can change almost anything that your heart desires to change!

By default, plot titles in ggplot2 are left-aligned. For hjust:

  • 0 == left
  • 0.5 == centered
  • 1 == right

We could also save all these as a custom theme. We are not fans of the default font, so we are also going to change this. To do this, you need to install the (extrafont package)[https://github.com/wch/extrafont] and follow its setup instructions before doing this next step.

hot_diggity <- theme(plot.title = element_text(hjust = 0.5),
                     axis.text = element_text(size = 12),
                     panel.background = element_blank(),
                     axis.line.x = element_line(color = "gray80", size = 0.5),
                     axis.ticks = element_line(color = "gray80", size = 0.5),
                     text = element_text(family = "Lato") # need extrafont for this
                     )
affil_plot + hot_diggity 

We could also use someone else’s theme:

library(ggthemes)
affil_plot + theme_fivethirtyeight(base_family = "Lato")

affil_plot + theme_tufte( base_family = "Palatino")

The final thing we have to mess with is the x-axis ticks and labels. We’ll do this in two steps, then override our previous layer scale_x_continuous.

years_to_label <- seq(from = 1981, to = 2017, by = 4)
years_to_label
 [1] 1981 1985 1989 1993 1997 2001 2005 2009 2013 2017
hd_years <- hdm_affil %>%
  distinct(year) %>% 
  mutate(year_lab = ifelse(year %in% years_to_label, year, ""))
affil_plot + 
  hot_diggity +
  scale_x_continuous(expand = c(0, 0), 
                     breaks = hd_years$year,
                     labels = hd_years$year_lab)
Scale for 'x' is already present. Adding another scale for 'x', which will
replace the existing scale.

10 Final (final, final) version

Don’t name your files “final” :)

All together in one chunk, here is our final (for now) plot! I’m also adding some additional elements here to show you options:

nathan_plot <- ggplot(hdm_affil, aes(x = year, y = num_eaten)) + 
  geom_col(aes(fill = affiliated)) + 
  labs(x = "Year", y = "Hot Dogs and Buns Consumed") +
  ggtitle("Nathan's Hot Dog Eating Contest Results, 1981-2017") +
  scale_fill_manual(values = c('#E9602B','#2277A0','#CCB683'),
                    name = "IFOCE-affiliation") + 
  hot_diggity +
  scale_y_continuous(expand = c(0, 0),
                     breaks = seq(0, 70, 10)) +
  scale_x_continuous(expand = c(0, 0), 
                     breaks = hd_years$year,
                     labels = hd_years$year_lab) + 
  coord_cartesian(xlim = c(1980, 2018), ylim = c(0, 80)) 
nathan_plot

Adding some plot annotations rather than having a fill legend:

nathan_ann <- nathan_plot +
  guides(fill = FALSE) +
  coord_cartesian(xlim = c(1980, 2019), ylim = c(0, 85)) +
  annotate('segment', x=1980.75, xend=2000.25, y= 30, yend=30, size=0.5, color="#CCB683")+
  annotate('segment', x=1980.75, xend=1980.75, y= 30, yend=28, size=0.5, color="#CCB683") +
  annotate('segment', x=2000.25, xend=2000.25, y= 30, yend=28, size=0.5, color="#CCB683") +
  annotate('segment', x=1990, xend=1990, y= 33, yend=30, size=0.5, color="#CCB683") +
nathan_ann
Coordinate system already present. Adding new coordinate system, which will replace the existing one.
Error in eval(expr, envir, enclos): object 'nathan_ann' not found
    annotate('text', x=1990, y=36, label="No MLE/IFOCE Affiliation", color="#CCB683", family="Lato", hjust=0.5, size = 3) +



  annotate('segment', x=2000.75, xend=2006.25, y= 58, yend=58, size=0.5, color="#2277A0") +
  annotate('segment', x=2000.75, xend=2000.75, y= 58, yend=56, size=0.5, color="#2277A0") +
  annotate('segment', x=2006.25, xend=2006.25, y= 58, yend=56, size=0.5, color="#2277A0") +
  annotate('segment', x=2003.5, xend=2003.5, y= 61, yend=58, size=0.5, color="#2277A0") +
  annotate('text', x=2003.5, y=65, label="MLE/IFOCE\nFormer Member", color="#2277A0", family="Lato", hjust=0.5, size = 3) +


  annotate('segment', x=2006.75, xend=2017.25, y= 76, yend=76, size=0.5, color="#E9602B") +
  annotate('segment', x=2006.75, xend=2006.75, y= 76, yend=74, size=0.5, color="#E9602B") +
  annotate('segment', x=2017.25, xend=2017.25, y= 76, yend=74, size=0.5, color="#E9602B") +
  annotate('segment', x=2012, xend=2012, y= 79, yend=76, size=0.5, color="#E9602B") +
  annotate('text', x=2012, y=82, label="MLE/IFOCE Current Member", color="#E9602B", family="Lato", hjust=0.5, size = 3)
Error: Cannot add ggproto objects together. Did you forget to add this object to a ggplot object?
nathan_ann
Error in eval(expr, envir, enclos): object 'nathan_ann' not found

Finally, adding in another layer of data from female contestants:

hdm_females <- read_csv(here::here("data", "hot_dog_contest_with_affiliation.csv"), 
    col_types = cols(
      affiliated = col_factor(levels = NULL), 
      gender = col_factor(levels = NULL)
      )) %>% 
  mutate(post_ifoce = year >= 1997) %>% 
  filter(year >= 1981 & gender == "female") 
glimpse(hdm_females)
Observations: 7
Variables: 6
$ year       <dbl> 2017, 2016, 2015, 2014, 2013, 2012, 2011
$ gender     <fct> female, female, female, female, female, female, female
$ name       <chr> "Miki Sudo", "Miki Sudo", "Miki Sudo", "Miki Sudo", "Sonya…
$ num_eaten  <dbl> 41.00, 38.00, 38.00, 34.00, 36.75, 45.00, 40.00
$ affiliated <fct> current, current, current, current, current, current, curr…
$ post_ifoce <lgl> TRUE, TRUE, TRUE, TRUE, TRUE, TRUE, TRUE
nathan_w_females <- nathan_ann +
  # add in the female data, and manually set a fill color
  geom_col(data = hdm_females, 
           width = 0.75, 
           fill = "#F68A39") 
Error in eval(expr, envir, enclos): object 'nathan_ann' not found
nathan_w_females
Error in eval(expr, envir, enclos): object 'nathan_w_females' not found

And adding a final caption:

caption <- paste(strwrap("* From 2011 on, separate Men's and Women's prizes have been awarded. All female champions to date have been MLE/IFOCE-affiliated.", 70), collapse="\n")

nathan_w_females +
  # now an asterisk to set off the female scores, and a caption
  annotate('text', x = 2018.5, y = 39, label="*", family = "Lato", size = 8) +
  labs(caption = caption) +
  theme(plot.caption = element_text(family = "Lato", size=8, hjust=0, margin=margin(t=15)))
Error in eval(expr, envir, enclos): object 'nathan_w_females' not found
LS0tCnRpdGxlOiAiTGFiIDAxOiBOYXRoYW4ncyBIb3QtRG9nIEVhdGluZyBDb250ZXN0IgpzdWJ0aXRsZTogIkNTNjMxIgphdXRob3I6ICJBbGlzb24gSGlsbCAod2l0aCBtaW5vciBtb2RpZmljYXRpb25zIGJ5IFN0ZXZlbiBCZWRyaWNrKSIKb3V0cHV0OgogIGh0bWxfZG9jdW1lbnQ6CiAgICB0aGVtZTogZmxhdGx5CiAgICB0b2M6IFRSVUUKICAgIHRvY19mbG9hdDogVFJVRQogICAgdG9jX2RlcHRoOiAyCiAgICBudW1iZXJfc2VjdGlvbnM6IFRSVUUKICAgIGNvZGVfZm9sZGluZzogaGlkZQogICAgY29kZV9kb3dubG9hZDogdHJ1ZQotLS0KCmBgYHtyIHNldHVwLCBpbmNsdWRlID0gRkFMU0UsIGNhY2hlID0gRkFMU0V9CmtuaXRyOjpvcHRzX2NodW5rJHNldChlcnJvciA9IFRSVUUsIGNvbW1lbnQgPSBOQSwgd2FybmluZ3MgPSBGQUxTRSwgZXJyb3JzID0gRkFMU0UsIG1lc3NhZ2VzID0gRkFMU0UsIHRpZHkgPSBGQUxTRSwgY2FjaGUgPSBUUlVFKQpgYGAKCmBgYHtyIGxvYWQtcGFja2FnZXMsIGluY2x1ZGUgPSBGQUxTRX0KbGlicmFyeSh0aWR5dmVyc2UpCmxpYnJhcnkoZXh0cmFmb250KQpgYGAKCiMgR29hbHMgZm9yIExhYiAwMQoKLSBHZXQgeW91ciBmZWV0IHdldCEKLSBJbm5vY3VsYXRlIHlvdSBhZ2FpbnN0IGBnZ3Bsb3QyYCBlcnJvcnMtIHdlIGFsbCBnZXQgdGhlbSEKLSBHZXQgZXhwb3NlZCB0byB0aGUgKnJhbmdlKiBvZiB0aGluZ3MgeW91IGNhbiBkbywgYmVmb3JlIHdlIGdvICoqZGVlZXAqKi4uLgotIERldmVsb3AgeW91ciAqb3duKiAqKnBlcnNvbmFsKiogcHJlZmVyZW5jZXMgZm9yIGRhdGEgdmlzdWFsaXphdGlvbnMhCiAgICAtIERvIHlvdSBsaWtlIG9yIGhhdGUgZ3JpZGxpbmVzPwogICAgLSBXaGF0IGZvbnRzIGRvIHlvdSBmaW5kIHBsZWFzYW50IHRvIHJlYWQ/CiAgICAtIFdoYXQga2luZHMgb2YgY29sb3JzIGRvIHlvdSBsaWtlPwogICAgLSBBcmUgeW91IHRlYW0gYHRoZW1lX2dyYXlgIG9yIGB0aGVtZV9id2AgKG9yIGB0aGVtZV9taW5pbWFsYCk/CiAgICAKVGhlc2UgYXJlIGltcG9ydGFudCBxdWVzdGlvbnMsIGFuZCBJIHdhbnQgeW91IHRvIGRldmVsb3AgKHdlbGwtaW5mb3JtZWQpIG9waW5pb25zIG9uIHRoZXNlIG1hdHRlcnMhCiFbXShpbWFnZXMvdGhlbWUtdGVhbS10d2VldHMucG5nKQoKCiMgTmF0aGFuJ3MgSG90IERvZyBFYXRpbmcgQ29udGVzdAoKIVtdKGh0dHBzOi8vaTAud3AuY29tL2Zsb3dpbmdkYXRhLmNvbS93cC1jb250ZW50L3VwbG9hZHMvMjAwOS8wNi9ob3QtZG9nczEuZ2lmP3pvb209MiZmaXQ9OTAwJTJDNDIzKQoKVGhpcyBpbmNsdWRlcyBhIHJlY29uc3RydWN0aW9uIG9mIFtOYXRoYW4gWWF1J3MgaG90IGRvZyBjb250ZXN0IGV4YW1wbGVdKGh0dHA6Ly9mbG93aW5nZGF0YS5jb20vMjAwOS8wNy8wMi93aG9zLWdvaW5nLXRvLXdpbi1uYXRoYW5zLWhvdC1kb2ctZWF0aW5nLWNvbnRlc3QvaG90LWRvZ3MtMi8pLCBhcyBpbnRlcnByZXRlZCBieSBKYWNraWUgV2lyeiwgcG9ydGVkIGludG8gUiBhbmQgYGdncGxvdDJgIGJ5IFN0ZXZlbiBCZWRyaWNrIGZvciBhIHdvcmtzaG9wIGZvciB0aGUgW09IU1UgRGF0YSBTY2llbmNlIEluc3RpdHV0ZV0oaHR0cHM6Ly9vaHN1bGlicmFyeS1kYXRhc2NpZW5jZWluc3RpdHV0ZS5naXRodWIuaW8pLCBhbmQgZmluYWxseSBhZGFwdGVkIGJ5IEFsaXNvbiBIaWxsIGZvciBhbGwgeW91IGludHJlcGlkIERhdGEtVml6LW9uYXV0cyEKCgpGaXJzdCwgd2UgbG9hZCBvdXIgcGFja2FnZXM6IAoKCmBgYHtyIGV2YWw9RkFMU0UsIG1lc3NhZ2UgPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFfQpsaWJyYXJ5KHRpZHl2ZXJzZSkKbGlicmFyeShleHRyYWZvbnQpCmxpYnJhcnkoaGVyZSkKYGBgCgoKIyBSZWFkIGluIGFuZCB3cmFuZ2xlIGRhdGEKCk5leHQsIHdlIGxvYWQgc29tZSBkYXRhLiBJbiB0aGUgUlN0dWRpbyBDbG91ZCBwcm9qZWN0IGZvciB0aGlzIGxhYiwgeW91IHdpbGwgc2VlIGEgYGRhdGFgIGRpcmVjdG9yeSB3aXRoIHRoZSBuZWNlc3NhcnkgZmlsZXMuIFdlIGNhbiByZWFkIGl0IGluIHVzaW5nIGByZWFkX2NzdmAsIGFuZCBhbG9uZyB0aGUgd2F5IHVzZSBgY29sX2ZhY3RvcmAgdG8gdGVsbCBpdCBob3cgdG8gaGFuZGxlIHRoZSBnZW5kZXIgY29sdW1uLgoKYGBge3J9CmhvdF9kb2dzIDwtIHJlYWRfY3N2KGhlcmU6OmhlcmUoImRhdGEiLCAiaG90X2RvZ19jb250ZXN0LmNzdiIpLCAKICAgIGNvbF90eXBlcyA9IGNvbHMoCiAgICAgIGdlbmRlciA9IGNvbF9mYWN0b3IobGV2ZWxzID0gTlVMTCkKICAgICkpCmBgYAoKTm90ZSB0aGF0IGlmIHRoZSBkYXRhIHdlcmUgX25vdF8gYWxyZWFkeSBkb3dubG9hZGVkIGxvY2FsbHksIGByZWFkX2NzdigpYCBjYW4gbG9hZCBDU1YgZmlsZXMgZnJvbSBhIFVSTDoKCmBgYHtyIGV2YWwgPSBGQUxTRX0KaG90X2RvZ3MgPC0gcmVhZF9jc3YoImh0dHA6Ly9iaXQubHkvY3M2MzEtaG90ZG9nIiwgCiAgICBjb2xfdHlwZXMgPSBjb2xzKAogICAgICBnZW5kZXIgPSBjb2xfZmFjdG9yKGxldmVscyA9IE5VTEwpCiAgICApKQpgYGAKCkVpdGhlciB3YXkgeW91IGRvIGl0LCBjaGVjayBpdCBvdXQgb25jZSByZWFkIGluIGFuZCBtYWtlIHN1cmUgaXQgbG9va3MgbGlrZSB0aGlzIQoKYGBge3J9CmdsaW1wc2UoaG90X2RvZ3MpCmhvdF9kb2dzCmBgYAoKV2UnbGwgYmUgd2FudGluZyB0byBzb21laG93IGluY2x1ZGUgaW5mb3JtYXRpb24gYWJvdXQgd2hldGhlciBhIGdpdmVuIHllYXIgd2FzIGJlZm9yZSBvciBhZnRlciB0aGUgaW5jb3Jwb3JhdGlvbiBvZiB0aGUgY29tcGV0aXRpdmUgZWF0aW5nIGxlYWd1ZSwgc28gbGV0J3MgYWRkIGFuIGluZGljYXRvciBmaWVsZCB0byB0aGUgZGF0YSB1c2luZyBgbXV0YXRlKClgLiBBbHNvLCB0aGUgZGF0YSdzIGEgbGl0dGxlIHNrZXRjaHkgcHJlLTE5ODEgYW5kIGZvciBvdXIgcHVycG9zZXMgdG9kYXkgd2UnbGwgYmUgZm9jdXNpbmcgb24gbWFsZXMgb25seSwgc28gbGV0J3MgZG8gc29tZSBgZmlsdGVyYGluZyB0b286CgpgYGB7cn0KaG90X2RvZ3MgPC0gaG90X2RvZ3MgJT4lIAogIG11dGF0ZShwb3N0X2lmb2NlID0geWVhciA+PSAxOTk3KSAlPiUgCiAgZmlsdGVyKHllYXIgPj0gMTk4MSAmIGdlbmRlciA9PSAnbWFsZScpCmhvdF9kb2dzCmBgYAoKCgojIFBsb3QgVGhlIERhdGEKCk5vdyBsZXQncyB0cnkgbWFraW5nIGEgZmlyc3QgY3JhY2sgYXQgYSBza2V0Y2h5IHBsb3Q6CgpgYGB7cn0KZ2dwbG90KGhvdF9kb2dzLCBhZXMoeCA9IHllYXIsIHkgPSBudW1fZWF0ZW4pKSArIAogIGdlb21fY29sKCkKYGBgCgpOb3RlIHRoYXQgb3VyIGRhdGEgaXMgYWxyZWFkeSBpbiAiY291bnRlZCIgZm9ybSwgc28gd2UncmUgdXNpbmcgYGdlb21fY29sKClgIGluc3RlYWQgb2YgYGdlb21fYmFyKClgLgoKCiMgQWRkIEF4aXMgTGFiZWxzIEFuZCBUaXRsZQoKCmBgYHtyfQpnZ3Bsb3QoaG90X2RvZ3MsIGFlcyh4ID0geWVhciwgeSA9IG51bV9lYXRlbikpICsgCiAgZ2VvbV9jb2woKSArCiAgbGFicyh4ID0gIlllYXIiLCB5ID0gIkhvdCBEb2dzIGFuZCBCdW5zIENvbnN1bWVkIikgKwogIGdndGl0bGUoIk5hdGhhbidzIEhvdCBEb2cgRWF0aW5nIENvbnRlc3QgUmVzdWx0cywgMTk4MS0yMDE3IikKYGBgCgojIFBsYXkgV2l0aCBDb2xvcnMKCjxkaXYgY2xhc3M9InBhbmVsIHBhbmVsLXN1Y2Nlc3MiPgogIDxkaXYgY2xhc3M9InBhbmVsLWhlYWRpbmciPkNoYWxsZW5nZSAjMTo8L2Rpdj4KICA8ZGl2IGNsYXNzPSJwYW5lbC1ib2R5Ij4KTWFrZSAzIHZlcnNpb25zIG9mIHRoZSBsYXN0IHBsb3Qgd2UganVzdCBtYWRlOgoKKiBfX0luIHRoZSBmaXJzdCxfXyBtYWtlIGFsbCB0aGUgY29sdW1ucyBvdXRsaW5lZCBpbiAid2hpdGUiLgoqIF9fSW4gdGhlIHNlY29uZCxfXyBtYWtlIGFsbCB0aGUgY29sdW1ucyBvdXRsaW5lZCBpbiAid2hpdGUiIGFuZCBmaWxsZWQgaW4gIm5hdnlibHVlIi4KKiBfX0luIHRoZSB0aGlyZCxfXyBtYWtlIGFsbCB0aGUgY29sdW1ucyBvdXRsaW5lZCBpbiAid2hpdGUiIGFuZCBmaWxsZWQgaW4gYWNjb3JkaW5nIHRvIHdoZXRoZXIgb3Igbm90IGBwb3N0X2lmb2NlYCBpcyBUUlVFIG9yIEZBTFNFICh1c2UgZGVmYXVsdCBjb2xvcnMgZm9yIG5vdykuCiAgPC9kaXY+CjwvZGl2PgoKCgpgYGB7cn0KZ2dwbG90KGhvdF9kb2dzLCBhZXMoeCA9IHllYXIsIHkgPSBudW1fZWF0ZW4pKSArIAogIGdlb21fY29sKGNvbG91ciA9ICJ3aGl0ZSIpICsgCiAgbGFicyh4ID0gIlllYXIiLCB5ID0gIkhvdCBEb2dzIGFuZCBCdW5zIENvbnN1bWVkIikgKwogIGdndGl0bGUoIk5hdGhhbidzIEhvdCBEb2cgRWF0aW5nIENvbnRlc3QgUmVzdWx0cywgMTk4MS0yMDE3IikKYGBgCgpgYGB7cn0KZ2dwbG90KGhvdF9kb2dzLCBhZXMoeCA9IHllYXIsIHkgPSBudW1fZWF0ZW4pKSArIAogIGdlb21fY29sKGNvbG91ciA9ICJ3aGl0ZSIsIGZpbGwgPSAibmF2eWJsdWUiKSArIAogIGxhYnMoeCA9ICJZZWFyIiwgeSA9ICJIb3QgRG9ncyBhbmQgQnVucyBDb25zdW1lZCIpICsKICBnZ3RpdGxlKCJOYXRoYW4ncyBIb3QgRG9nIEVhdGluZyBDb250ZXN0IFJlc3VsdHMsIDE5ODEtMjAxNyIpCmBgYAoKYGBge3J9CmdncGxvdChob3RfZG9ncywgYWVzKHggPSB5ZWFyLCB5ID0gbnVtX2VhdGVuKSkgKyAKICBnZW9tX2NvbChhZXMoZmlsbCA9IHBvc3RfaWZvY2UpLCBjb2xvdXIgPSAid2hpdGUiKSArIAogIGxhYnMoeCA9ICJZZWFyIiwgeSA9ICJIb3QgRG9ncyBhbmQgQnVucyBDb25zdW1lZCIpICsKICBnZ3RpdGxlKCJOYXRoYW4ncyBIb3QgRG9nIEVhdGluZyBDb250ZXN0IFJlc3VsdHMsIDE5ODEtMjAxNyIpCmBgYAoKCgo8ZGl2IGNsYXNzPSJwYW5lbCBwYW5lbC1zdWNjZXNzIj4KICA8ZGl2IGNsYXNzPSJwYW5lbC1oZWFkaW5nIj5DaGFsbGVuZ2UgIzI6PC9kaXY+CiAgPGRpdiBjbGFzcz0icGFuZWwtYm9keSI+CldoYXQgaWYgeW91IHdhbnQgdG8gY2hhbmdlIHRoZSBsZWdlbmQgaW4gdGhlIGxhc3QgcGxvdCB5b3UgbWFkZT8gVXNlIGdvb2dsZSB0byBmaWd1cmUgb3V0IGhvdyB0byBkbyB0aGUgZm9sbG93aW5nOgoKKiBEZWxldGUgdGhlIGxlZ2VuZCB0aXRsZQoqIE1ha2UgdGhlIGxlZ2VuZCB0ZXh0IGVpdGhlciAiUG9zdC1JRk9DRSIgb3IgIlByZS1JRk9DRSIuCiAgPC9kaXY+CjwvZGl2PgoKYGBge3J9CmdncGxvdChob3RfZG9ncywgYWVzKHggPSB5ZWFyLCB5ID0gbnVtX2VhdGVuKSkgKyAKICBnZW9tX2NvbChhZXMoZmlsbCA9IHBvc3RfaWZvY2UpLCBjb2xvdXIgPSAid2hpdGUiKSArIAogIGxhYnMoeCA9ICJZZWFyIiwgeSA9ICJIb3QgRG9ncyBhbmQgQnVucyBDb25zdW1lZCIpICsKICBnZ3RpdGxlKCJOYXRoYW4ncyBIb3QgRG9nIEVhdGluZyBDb250ZXN0IFJlc3VsdHMsIDE5ODEtMjAxNyIpICsKICBzY2FsZV9maWxsX2Rpc2NyZXRlKG5hbWUgPSAiIiwKICAgICAgICAgICAgICAgICAgICAgIGxhYmVscz1jKCJQcmUtSUZPQ0UiLCAiUG9zdC1JRk9DRSIpKQpgYGAKCiMgQ2hhbmdlIFRoZSBEYXRhc2V0CgpOb3csIGxldCdzIGNoYW5nZSB0aGUgcXVlc3Rpb24gYSBsaXR0bGUgYml0LiBUaGlzIGxvb2tzIGF0IHRoZSBfY3JlYXRpb25fIG9mIHRoZSBJRk9DRS4gV2hhdCBhYm91dCB0aGUgX2FmZmlsaWF0aW9uXyBvZiB0aGUgY29udGVzdGFudHM/IFdlJ2xsIG5lZWQgc29tZSBkaWZmZXJlbnQgZGF0YSBmb3IgdGhpcy4gVGhyb3VnaCB0aGUgX01hZ2ljIE9mIERhdGEgU2NpZW5jZeKEol8sIHdlIGhhdmUgZHVnIHRoYXQgaW5mb3JtYXRpb24gdXAgYW5kIHB1dCBpdCBpbnRvIGFuIGV4cGFuZGVkIHZlcnNpb24gb2Ygb3VyIENTViBmaWxlIGF2YWlsYWJsZSBhdCBbaHR0cDovL2JpdC5seS9jczYzMS1ob3Rkb2ctYWZmaWxpYXRlZF0oaHR0cDovL2JpdC5seS9jczYzMS1ob3Rkb2ctYWZmaWxpYXRlZCkuCgoKPGRpdiBjbGFzcz0icGFuZWwgcGFuZWwtc3VjY2VzcyI+CiAgPGRpdiBjbGFzcz0icGFuZWwtaGVhZGluZyI+Q2hhbGxlbmdlICMzOjwvZGl2PgogIDxkaXYgY2xhc3M9InBhbmVsLWJvZHkiPgpMZXQncyB3b3JrIHdpdGggdGhpcyBuZXcgZGF0YXNldCEgRG8gdGhlIGZvbGxvd2luZzoKCiogUmVhZCBpbiB0aGUgImhvdF9kb2dfY29udGVzdF93aXRoX2FmZmlsaWF0aW9uLmNzdiIgZGF0YSBmaWxlLCB1c2luZyBgY29sX3R5cGVzYCB0byByZWFkIGluIGBhZmZpbGlhdGVkYCBhbmQgYGdlbmRlcmAgYXMgZmFjdG9ycy4gCiogV2l0aGluIGEgYG11dGF0ZWAsIGNyZWF0ZSBhIG5ldyB2YXJpYWJsZSBjYWxsZWQgYHBvc3RfaWZvY2VgIHRoYXQgaXMgVFJVRSBpZiBgeWVhcmAgaXMgZ3JlYXRlciB0aGFuIG9yIGVxdWFsIHRvIDE5OTcuIAoqIEFsc28gYGZpbHRlcmAgdGhlIG5ldyBkYXRhIGZvciBvbmx5IHllYXJzIDE5ODEgYW5kIGFmdGVyLCBhbmQgb25seSBmb3IgbWFsZSBjb21wZXRpdG9ycy4KICA8L2Rpdj4KPC9kaXY+CgpgYGB7ciBldmFsID0gRkFMU0V9CmhkbV9hZmZpbCA8LSByZWFkX2NzdihoZXJlOjpoZXJlKCJkYXRhIiwgImhvdF9kb2dfY29udGVzdF93aXRoX2FmZmlsaWF0aW9uLmNzdiIpLCAKICAgIGNvbF90eXBlcyA9IGNvbHMoCiAgICAgIGFmZmlsaWF0ZWQgPSBjb2xfZmFjdG9yKGxldmVscyA9IE5VTEwpLCAKICAgICAgZ2VuZGVyID0gY29sX2ZhY3RvcihsZXZlbHMgPSBOVUxMKQogICAgICApKSAlPiUgCiAgbXV0YXRlKHBvc3RfaWZvY2UgPSB5ZWFyID49IDE5OTcpICU+JSAKICBmaWx0ZXIoeWVhciA+PSAxOTgxICYgZ2VuZGVyID09ICJtYWxlIikgCmBgYAoKCmBgYHtyfQpoZG1fYWZmaWwgPC0gcmVhZF9jc3YoaGVyZTo6aGVyZSgiZGF0YSIsICJob3RfZG9nX2NvbnRlc3Rfd2l0aF9hZmZpbGlhdGlvbi5jc3YiKSwgCiAgICBjb2xfdHlwZXMgPSBjb2xzKAogICAgICBhZmZpbGlhdGVkID0gY29sX2ZhY3RvcihsZXZlbHMgPSBOVUxMKSwgCiAgICAgIGdlbmRlciA9IGNvbF9mYWN0b3IobGV2ZWxzID0gTlVMTCkKICAgICAgKSkgJT4lIAogIG11dGF0ZShwb3N0X2lmb2NlID0geWVhciA+PSAxOTk3KSAlPiUgCiAgZmlsdGVyKHllYXIgPj0gMTk4MSAmIGdlbmRlciA9PSAibWFsZSIpIApnbGltcHNlKGhkbV9hZmZpbCkKYGBgCgo8ZGl2IGNsYXNzPSJwYW5lbCBwYW5lbC1zdWNjZXNzIj4KICA8ZGl2IGNsYXNzPSJwYW5lbC1oZWFkaW5nIj5DaGFsbGVuZ2UgIzQ6PC9kaXY+CiAgPGRpdiBjbGFzcz0icGFuZWwtYm9keSI+CkxldCdzIGRvIHNvbWUgYmFzaWMgRURBIHdpdGggdGhpcyBuZXcgZGF0YXNldCEgRG8gdGhlIGZvbGxvd2luZzoKCiogVXNlIGBkcGx5cjo6ZGlzdGluY3RgIHRvIGZpZ3VyZSBvdXQgaG93IG1hbnkgdW5pcXVlIHZhbHVlcyB0aGVyZSBhcmUgb2YgYGFmZmlsaWF0ZWRgLgoqIFVzZSBgZHBseXI6OmNvdW50YCB0byBjb3VudCB0aGUgbnVtYmVyIG9mIHJvd3MgZm9yIGVhY2ggdW5pcXVlIHZhbHVlIG9mIGBhZmZpbGlhdGVkYDsgdXNlIGA/Y291bnRgIHRvIGZpZ3VyZSBvdXQgaG93IHRvIHNvcnQgdGhlIGNvdW50cyBpbiBkZXNjZW5kaW5nIG9yZGVyLgogIDwvZGl2Pgo8L2Rpdj4KCmBgYHtyfQpoZG1fYWZmaWwgJT4lIAogIGRpc3RpbmN0KGFmZmlsaWF0ZWQpCmhkbV9hZmZpbCAlPiUgCiAgY291bnQoYWZmaWxpYXRlZCwgc29ydCA9IFRSVUUpCmBgYAoKCk5vdyBsZXQncyBwbG90IHRoaXMgbmV3IGRhdGEsIGFuZCBmaWxsIHRoZSBjb2x1bW5zIGFjY29yZGluZyB0byBvdXIgbmV3IGBhZmZpbGlhdGVkYCBjb2x1bW4uCgpgYGB7cn0KZ2dwbG90KGhkbV9hZmZpbCwgYWVzKHggPSB5ZWFyLCB5ID0gbnVtX2VhdGVuKSkgKyAKICBnZW9tX2NvbChhZXMoZmlsbCA9IGFmZmlsaWF0ZWQpKSArIAogIGxhYnMoeCA9ICJZZWFyIiwgeSA9ICJIb3QgRG9ncyBhbmQgQnVucyBDb25zdW1lZCIpICsKICBnZ3RpdGxlKCJOYXRoYW4ncyBIb3QgRG9nIEVhdGluZyBDb250ZXN0IFJlc3VsdHMsIDE5ODEtMjAxNyIpCmBgYAoKPGRpdiBjbGFzcz0icGFuZWwgcGFuZWwtc3VjY2VzcyI+CiAgPGRpdiBjbGFzcz0icGFuZWwtaGVhZGluZyI+Q2hhbGxlbmdlICM1OjwvZGl2PgogIDxkaXYgY2xhc3M9InBhbmVsLWJvZHkiPgpEbyB0aGUgZm9sbG93aW5nIHVwZGF0ZXMgdG8gdGhlIGxhc3QgcGxvdCB3ZSBqdXN0IG1hZGU6CgoqIFVwZGF0ZSB0aGUgY29sb3JzIHVzaW5nIGhleCBjb2xvcnM6IGBjKCcjRTk2MDJCJywnIzIyNzdBMCcsJyNDQ0I2ODMnKWAuIAoqIENoYW5nZSB0aGUgbGVnZW5kIHRpdGxlIHRvICJJRk9DRS1hZmZpbGlhdGlvbiIuIAoqIFNhdmUgdGhpcyBwbG90IG9iamVjdCBhcyAiYWZmaWxfcGxvdCIuCiAgPC9kaXY+CjwvZGl2PgoKYGBge3J9CmFmZmlsX3Bsb3QgPC0gZ2dwbG90KGhkbV9hZmZpbCwgYWVzKHggPSB5ZWFyLCB5ID0gbnVtX2VhdGVuKSkgKyAKICBnZW9tX2NvbChhZXMoZmlsbCA9IGFmZmlsaWF0ZWQpKSArIAogIGxhYnMoeCA9ICJZZWFyIiwgeSA9ICJIb3QgRG9ncyBhbmQgQnVucyBDb25zdW1lZCIpICsKICBnZ3RpdGxlKCJOYXRoYW4ncyBIb3QgRG9nIEVhdGluZyBDb250ZXN0IFJlc3VsdHMsIDE5ODEtMjAxNyIpICsKICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCcjRTk2MDJCJywnIzIyNzdBMCcsJyNDQ0I2ODMnKSwKICAgICAgICAgICAgICAgICAgICBuYW1lID0gIklGT0NFLWFmZmlsaWF0aW9uIikKYWZmaWxfcGxvdApgYGAKCiMgUGxheSBXaXRoIFNjYWxlcyAmIENvb3JkaW5hdGVzCgpUaGUgc3BhY2luZydzIGEgbGl0dGxlIGZ1bmt5IGRvd24gbmVhciB0aGUgb3JpZ2luIG9mIHRoZSBwbG90LiBUaGUgW2RvY3VtZW50YXRpb25dKGh0dHA6Ly9nZ3Bsb3QyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL3NjYWxlX2NvbnRpbnVvdXMuaHRtbCkgdGVsbHMgdXMgdGhhdCB0aGUgZGVmYXVsdHMgYXJlIGBjKDAuMDUsIDApYCBmb3IgY29udGludW91cyB2YXJpYWJsZXMuIFRoZSBmaXJzdCBudW1iZXIgaXMgbXVsdGlwbGljYXRpdmUgYW5kIHRoZSBzZWNvbmQgaXMgYWRkaXRpdmUuIAoKVGhlIGRlZmF1bHQgd2FzIHRoYXQgMS44ICgoMjAxNy0xOTgxKSouMDUrMCkgd2FzIGFkZGVkIHRvIHRoZSByaWdodCBhbmQgbGVmdCBzaWRlcyBvZiB0aGUgeC1heGlzIGFzIHBhZGRpbmcsIHNvIHRoZSBlZmZlY3RpdmUgZGVmYXVsdCBsaW1pdHMgd2VyZSBgYygxOTc5LCAyMDE5KWAuCgpMZXQncyB0aWdodGVuIHRoYXQgdXAgd2l0aCB0aGUgYGV4cGFuZGAgcHJvcGVydHkgZm9yIHRoZSBgc2NhbGVfeV9jb250aW51b3VzYCAod2UnbGwgYWxzbyBjaGFuZ2UgdGhlIGJyZWFrcyBmb3IgeS1heGlzIHRpY2sgbWFya3MgaGVyZSkgYW5kIGBzY2FsZV94X2NvbnRpbnVvdXNgIHNldHRpbmdzOgoKYGBge3J9CmFmZmlsX3Bsb3QgPC0gYWZmaWxfcGxvdCArIAogIHNjYWxlX3lfY29udGludW91cyhleHBhbmQgPSBjKDAsIDApLAogICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBzZXEoMCwgNzAsIDEwKSkgKwogIHNjYWxlX3hfY29udGludW91cyhleHBhbmQgPSBjKDAsIDApKQphZmZpbF9wbG90CmBgYAoKCgpCdXQgbm93IHRoZSBwbG90IGxvb2tzIGxpa2UgaXQgaXMgd2VhcmluZyB0aWdodCBwYW50cy4KCiFbXShodHRwczovL21lZGlhLmdpcGh5LmNvbS9tZWRpYS94VDFYSDA3bm8yd1pTcTRtbm0vZ2lwaHkuZ2lmKQoKTGV0J3MgbG9vc2VuIHRoaW5ncyB1cCBhIGJpdCBieSB1cGRhdGluZyB0aGUgcGxvdCBjb29yZGluYXRlcy4gCgo8ZGl2IGNsYXNzPSJwYW5lbCBwYW5lbC1zdWNjZXNzIj4KICA8ZGl2IGNsYXNzPSJwYW5lbC1oZWFkaW5nIj5DaGFsbGVuZ2UgIzY6PC9kaXY+CiAgPGRpdiBjbGFzcz0icGFuZWwtYm9keSI+ClVzZSBgY29vcmRfY2FydGVzaWFuYCB0bzoKCiogU2V0IHRoZSB4LWF4aXMgcmFuZ2UgdG8gMTk4MC0yMDE4CiogU2V0IHRoZSB5LWF4aXMgcmFuZ2UgdG8gMC04MAogIDwvZGl2Pgo8L2Rpdj4KClVzaW5nIGBjb29yZF9jYXJ0ZXNpYW5gIGlzIHRoZSBwcmVmZXJyZWQgbGF5ZXIgaGVyZSBiZWNhdXNlICJzZXR0aW5nIGxpbWl0cyBvbiB0aGUgY29vcmRpbmF0ZSBzeXN0ZW0gd2lsbCB6b29tIHRoZSBwbG90IChsaWtlIHlvdSdyZSBsb29raW5nIGF0IGl0IHdpdGggYSBtYWduaWZ5aW5nIGdsYXNzKSwgYW5kIHdpbGwgbm90IGNoYW5nZSB0aGUgdW5kZXJseWluZyBkYXRhIGxpa2Ugc2V0dGluZyBgbGltaXRzYCBvbiBhIHNjYWxlIHdpbGwuIiAgCgo8ZGl2IGNsYXNzPSJwYW5lbCBwYW5lbC1pbmZvIj4KICA8ZGl2IGNsYXNzPSJwYW5lbC1oZWFkaW5nIj5MZXNzb246PC9kaXY+CiAgPGRpdiBjbGFzcz0icGFuZWwtYm9keSI+CkRvbid0IGNoYW5nZSBgbGltaXRzYCB1bmxlc3MgeW91IHJlYWxseSBrbm93IHdoYXQgeW91IGFyZSBkb2luZyEgTW9zdCBvZiB0aGUgdGltZSwgeW91IHdhbnQgdG8gY2hhbmdlIHRoZSBjb29yZGluYXRlcyBpbnN0ZWFkLgogIDwvZGl2Pgo8L2Rpdj4KCmBgYHtyfQphZmZpbF9wbG90IDwtIGFmZmlsX3Bsb3QgKyAKICBjb29yZF9jYXJ0ZXNpYW4oeGxpbSA9IGMoMTk4MCwgMjAxOCksIHlsaW0gPSBjKDAsIDgwKSkgCmFmZmlsX3Bsb3QKYGBgCgoKIyBQbGF5IFdpdGggVGhlbWUgU2V0dGluZ3MKCkxldCdzIGNoYW5nZSBzb21lIGtleSB0aGVtZSBzZXR0aW5nczoKCmBgYHtyfQphZmZpbF9wbG90ICsKICB0aGVtZShwbG90LnRpdGxlID0gZWxlbWVudF90ZXh0KGhqdXN0ID0gMC41KSkgKwogIHRoZW1lKGF4aXMudGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIpKSArCiAgdGhlbWUocGFuZWwuYmFja2dyb3VuZCA9IGVsZW1lbnRfYmxhbmsoKSkgKwogIHRoZW1lKGF4aXMubGluZS54ID0gZWxlbWVudF9saW5lKGNvbG9yID0gImdyYXk4MCIsIHNpemUgPSAwLjUpKSArCiAgdGhlbWUoYXhpcy50aWNrcyA9IGVsZW1lbnRfbGluZShjb2xvciA9ICJncmF5ODAiLCBzaXplID0gMC41KSkKYGBgCgo8ZGl2IGNsYXNzPSJwYW5lbCBwYW5lbC1pbmZvIj4KICA8ZGl2IGNsYXNzPSJwYW5lbC1oZWFkaW5nIj5MZXNzb246PC9kaXY+CiAgPGRpdiBjbGFzcz0icGFuZWwtYm9keSI+CllvdSBjYW4gY2hhbmdlICphbG1vc3QgYW55dGhpbmcqIHRoYXQgeW91ciBoZWFydCBkZXNpcmVzIHRvIGNoYW5nZSEgCiAgPC9kaXY+CjwvZGl2PgoKCkJ5IGRlZmF1bHQsIHBsb3QgdGl0bGVzIGluIGBnZ3Bsb3QyYCBhcmUgbGVmdC1hbGlnbmVkLiBGb3IgYGhqdXN0YDoKCi0gYDBgID09IGxlZnQKLSBgMC41YCA9PSBjZW50ZXJlZAotIGAxYCA9PSByaWdodAoKV2UgY291bGQgYWxzbyBzYXZlIGFsbCB0aGVzZSBhcyBhIGN1c3RvbSB0aGVtZS4gV2UgYXJlIG5vdCBmYW5zIG9mIHRoZSBkZWZhdWx0IGZvbnQsIHNvIHdlIGFyZSBhbHNvIGdvaW5nIHRvIGNoYW5nZSB0aGlzLiBUbyBkbyB0aGlzLCB5b3UgbmVlZCB0byBpbnN0YWxsIHRoZSAoYGV4dHJhZm9udGAgcGFja2FnZSlbaHR0cHM6Ly9naXRodWIuY29tL3djaC9leHRyYWZvbnRdIGFuZCBmb2xsb3cgaXRzIHNldHVwIGluc3RydWN0aW9ucyBiZWZvcmUgZG9pbmcgdGhpcyBuZXh0IHN0ZXAuCgpgYGB7cn0KaG90X2RpZ2dpdHkgPC0gdGhlbWUocGxvdC50aXRsZSA9IGVsZW1lbnRfdGV4dChoanVzdCA9IDAuNSksCiAgICAgICAgICAgICAgICAgICAgIGF4aXMudGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIpLAogICAgICAgICAgICAgICAgICAgICBwYW5lbC5iYWNrZ3JvdW5kID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgICAgICAgICAgICAgICBheGlzLmxpbmUueCA9IGVsZW1lbnRfbGluZShjb2xvciA9ICJncmF5ODAiLCBzaXplID0gMC41KSwKICAgICAgICAgICAgICAgICAgICAgYXhpcy50aWNrcyA9IGVsZW1lbnRfbGluZShjb2xvciA9ICJncmF5ODAiLCBzaXplID0gMC41KSwKICAgICAgICAgICAgICAgICAgICAgdGV4dCA9IGVsZW1lbnRfdGV4dChmYW1pbHkgPSAiTGF0byIpICMgbmVlZCBleHRyYWZvbnQgZm9yIHRoaXMKICAgICAgICAgICAgICAgICAgICAgKQpgYGAKCgoKYGBge3J9CmFmZmlsX3Bsb3QgKyBob3RfZGlnZ2l0eSAKYGBgCgpXZSBjb3VsZCBhbHNvIHVzZSBzb21lb25lIGVsc2UncyB0aGVtZToKCmBgYHtyfQpsaWJyYXJ5KGdndGhlbWVzKQphZmZpbF9wbG90ICsgdGhlbWVfZml2ZXRoaXJ0eWVpZ2h0KGJhc2VfZmFtaWx5ID0gIkxhdG8iKQphZmZpbF9wbG90ICsgdGhlbWVfdHVmdGUoIGJhc2VfZmFtaWx5ID0gIlBhbGF0aW5vIikKYGBgCgoKCgpUaGUgZmluYWwgdGhpbmcgd2UgaGF2ZSB0byBtZXNzIHdpdGggaXMgdGhlIHgtYXhpcyB0aWNrcyBhbmQgbGFiZWxzLiBXZSdsbCBkbyB0aGlzIGluIHR3byBzdGVwcywgdGhlbiBvdmVycmlkZSBvdXIgcHJldmlvdXMgbGF5ZXIgYHNjYWxlX3hfY29udGludW91c2AuCgpgYGB7cn0KeWVhcnNfdG9fbGFiZWwgPC0gc2VxKGZyb20gPSAxOTgxLCB0byA9IDIwMTcsIGJ5ID0gNCkKeWVhcnNfdG9fbGFiZWwKaGRfeWVhcnMgPC0gaGRtX2FmZmlsICU+JQogIGRpc3RpbmN0KHllYXIpICU+JSAKICBtdXRhdGUoeWVhcl9sYWIgPSBpZmVsc2UoeWVhciAlaW4lIHllYXJzX3RvX2xhYmVsLCB5ZWFyLCAiIikpCmBgYAoKYGBge3J9CmFmZmlsX3Bsb3QgKyAKICBob3RfZGlnZ2l0eSArCiAgc2NhbGVfeF9jb250aW51b3VzKGV4cGFuZCA9IGMoMCwgMCksIAogICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBoZF95ZWFycyR5ZWFyLAogICAgICAgICAgICAgICAgICAgICBsYWJlbHMgPSBoZF95ZWFycyR5ZWFyX2xhYikKYGBgCgojIEZpbmFsIChmaW5hbCwgZmluYWwpIHZlcnNpb24KCkRvbid0IG5hbWUgeW91ciBmaWxlcyAiZmluYWwiIDopCgohW10oaHR0cDovL3d3dy5waGRjb21pY3MuY29tL2NvbWljcy9hcmNoaXZlL3BoZDEwMTIxMnMuZ2lmKQoKQWxsIHRvZ2V0aGVyIGluIG9uZSBjaHVuaywgaGVyZSBpcyBvdXIgZmluYWwgKGZvciBub3cpIHBsb3QhIEknbSBhbHNvIGFkZGluZyBzb21lIGFkZGl0aW9uYWwgZWxlbWVudHMgaGVyZSB0byBzaG93IHlvdSBvcHRpb25zOgoKYGBge3J9Cm5hdGhhbl9wbG90IDwtIGdncGxvdChoZG1fYWZmaWwsIGFlcyh4ID0geWVhciwgeSA9IG51bV9lYXRlbikpICsgCiAgZ2VvbV9jb2woYWVzKGZpbGwgPSBhZmZpbGlhdGVkKSkgKyAKICBsYWJzKHggPSAiWWVhciIsIHkgPSAiSG90IERvZ3MgYW5kIEJ1bnMgQ29uc3VtZWQiKSArCiAgZ2d0aXRsZSgiTmF0aGFuJ3MgSG90IERvZyBFYXRpbmcgQ29udGVzdCBSZXN1bHRzLCAxOTgxLTIwMTciKSArCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gYygnI0U5NjAyQicsJyMyMjc3QTAnLCcjQ0NCNjgzJyksCiAgICAgICAgICAgICAgICAgICAgbmFtZSA9ICJJRk9DRS1hZmZpbGlhdGlvbiIpICsgCiAgaG90X2RpZ2dpdHkgKwogIHNjYWxlX3lfY29udGludW91cyhleHBhbmQgPSBjKDAsIDApLAogICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBzZXEoMCwgNzAsIDEwKSkgKwogIHNjYWxlX3hfY29udGludW91cyhleHBhbmQgPSBjKDAsIDApLCAKICAgICAgICAgICAgICAgICAgICAgYnJlYWtzID0gaGRfeWVhcnMkeWVhciwKICAgICAgICAgICAgICAgICAgICAgbGFiZWxzID0gaGRfeWVhcnMkeWVhcl9sYWIpICsgCiAgY29vcmRfY2FydGVzaWFuKHhsaW0gPSBjKDE5ODAsIDIwMTgpLCB5bGltID0gYygwLCA4MCkpIApuYXRoYW5fcGxvdApgYGAKCkFkZGluZyBzb21lIHBsb3QgYW5ub3RhdGlvbnMgcmF0aGVyIHRoYW4gaGF2aW5nIGEgZmlsbCBsZWdlbmQ6CgpgYGB7cn0KbmF0aGFuX2FubiA8LSBuYXRoYW5fcGxvdCArCiAgZ3VpZGVzKGZpbGwgPSBGQUxTRSkgKwogIGNvb3JkX2NhcnRlc2lhbih4bGltID0gYygxOTgwLCAyMDE5KSwgeWxpbSA9IGMoMCwgODUpKSArCiAgYW5ub3RhdGUoJ3NlZ21lbnQnLCB4PTE5ODAuNzUsIHhlbmQ9MjAwMC4yNSwgeT0gMzAsIHllbmQ9MzAsIHNpemU9MC41LCBjb2xvcj0iI0NDQjY4MyIpKwogIGFubm90YXRlKCdzZWdtZW50JywgeD0xOTgwLjc1LCB4ZW5kPTE5ODAuNzUsIHk9IDMwLCB5ZW5kPTI4LCBzaXplPTAuNSwgY29sb3I9IiNDQ0I2ODMiKSArCiAgYW5ub3RhdGUoJ3NlZ21lbnQnLCB4PTIwMDAuMjUsIHhlbmQ9MjAwMC4yNSwgeT0gMzAsIHllbmQ9MjgsIHNpemU9MC41LCBjb2xvcj0iI0NDQjY4MyIpICsKICBhbm5vdGF0ZSgnc2VnbWVudCcsIHg9MTk5MCwgeGVuZD0xOTkwLCB5PSAzMywgeWVuZD0zMCwgc2l6ZT0wLjUsIGNvbG9yPSIjQ0NCNjgzIikgKwpuYXRoYW5fYW5uCiAgICBhbm5vdGF0ZSgndGV4dCcsIHg9MTk5MCwgeT0zNiwgbGFiZWw9Ik5vIE1MRS9JRk9DRSBBZmZpbGlhdGlvbiIsIGNvbG9yPSIjQ0NCNjgzIiwgZmFtaWx5PSJMYXRvIiwgaGp1c3Q9MC41LCBzaXplID0gMykgKwoKCgogIGFubm90YXRlKCdzZWdtZW50JywgeD0yMDAwLjc1LCB4ZW5kPTIwMDYuMjUsIHk9IDU4LCB5ZW5kPTU4LCBzaXplPTAuNSwgY29sb3I9IiMyMjc3QTAiKSArCiAgYW5ub3RhdGUoJ3NlZ21lbnQnLCB4PTIwMDAuNzUsIHhlbmQ9MjAwMC43NSwgeT0gNTgsIHllbmQ9NTYsIHNpemU9MC41LCBjb2xvcj0iIzIyNzdBMCIpICsKICBhbm5vdGF0ZSgnc2VnbWVudCcsIHg9MjAwNi4yNSwgeGVuZD0yMDA2LjI1LCB5PSA1OCwgeWVuZD01Niwgc2l6ZT0wLjUsIGNvbG9yPSIjMjI3N0EwIikgKwogIGFubm90YXRlKCdzZWdtZW50JywgeD0yMDAzLjUsIHhlbmQ9MjAwMy41LCB5PSA2MSwgeWVuZD01OCwgc2l6ZT0wLjUsIGNvbG9yPSIjMjI3N0EwIikgKwogIGFubm90YXRlKCd0ZXh0JywgeD0yMDAzLjUsIHk9NjUsIGxhYmVsPSJNTEUvSUZPQ0VcbkZvcm1lciBNZW1iZXIiLCBjb2xvcj0iIzIyNzdBMCIsIGZhbWlseT0iTGF0byIsIGhqdXN0PTAuNSwgc2l6ZSA9IDMpICsKCgogIGFubm90YXRlKCdzZWdtZW50JywgeD0yMDA2Ljc1LCB4ZW5kPTIwMTcuMjUsIHk9IDc2LCB5ZW5kPTc2LCBzaXplPTAuNSwgY29sb3I9IiNFOTYwMkIiKSArCiAgYW5ub3RhdGUoJ3NlZ21lbnQnLCB4PTIwMDYuNzUsIHhlbmQ9MjAwNi43NSwgeT0gNzYsIHllbmQ9NzQsIHNpemU9MC41LCBjb2xvcj0iI0U5NjAyQiIpICsKICBhbm5vdGF0ZSgnc2VnbWVudCcsIHg9MjAxNy4yNSwgeGVuZD0yMDE3LjI1LCB5PSA3NiwgeWVuZD03NCwgc2l6ZT0wLjUsIGNvbG9yPSIjRTk2MDJCIikgKwogIGFubm90YXRlKCdzZWdtZW50JywgeD0yMDEyLCB4ZW5kPTIwMTIsIHk9IDc5LCB5ZW5kPTc2LCBzaXplPTAuNSwgY29sb3I9IiNFOTYwMkIiKSArCiAgYW5ub3RhdGUoJ3RleHQnLCB4PTIwMTIsIHk9ODIsIGxhYmVsPSJNTEUvSUZPQ0UgQ3VycmVudCBNZW1iZXIiLCBjb2xvcj0iI0U5NjAyQiIsIGZhbWlseT0iTGF0byIsIGhqdXN0PTAuNSwgc2l6ZSA9IDMpCm5hdGhhbl9hbm4KYGBgCgpGaW5hbGx5LCBhZGRpbmcgaW4gYW5vdGhlciBsYXllciBvZiBkYXRhIGZyb20gZmVtYWxlIGNvbnRlc3RhbnRzOgoKYGBge3J9CmhkbV9mZW1hbGVzIDwtIHJlYWRfY3N2KGhlcmU6OmhlcmUoImRhdGEiLCAiaG90X2RvZ19jb250ZXN0X3dpdGhfYWZmaWxpYXRpb24uY3N2IiksIAogICAgY29sX3R5cGVzID0gY29scygKICAgICAgYWZmaWxpYXRlZCA9IGNvbF9mYWN0b3IobGV2ZWxzID0gTlVMTCksIAogICAgICBnZW5kZXIgPSBjb2xfZmFjdG9yKGxldmVscyA9IE5VTEwpCiAgICAgICkpICU+JSAKICBtdXRhdGUocG9zdF9pZm9jZSA9IHllYXIgPj0gMTk5NykgJT4lIAogIGZpbHRlcih5ZWFyID49IDE5ODEgJiBnZW5kZXIgPT0gImZlbWFsZSIpIApnbGltcHNlKGhkbV9mZW1hbGVzKQpgYGAKCmBgYHtyfQpuYXRoYW5fd19mZW1hbGVzIDwtIG5hdGhhbl9hbm4gKwogICMgYWRkIGluIHRoZSBmZW1hbGUgZGF0YSwgYW5kIG1hbnVhbGx5IHNldCBhIGZpbGwgY29sb3IKICBnZW9tX2NvbChkYXRhID0gaGRtX2ZlbWFsZXMsIAogICAgICAgICAgIHdpZHRoID0gMC43NSwgCiAgICAgICAgICAgZmlsbCA9ICIjRjY4QTM5IikgCm5hdGhhbl93X2ZlbWFsZXMKYGBgCgpBbmQgYWRkaW5nIGEgZmluYWwgY2FwdGlvbjoKCmBgYHtyfQpjYXB0aW9uIDwtIHBhc3RlKHN0cndyYXAoIiogRnJvbSAyMDExIG9uLCBzZXBhcmF0ZSBNZW4ncyBhbmQgV29tZW4ncyBwcml6ZXMgaGF2ZSBiZWVuIGF3YXJkZWQuIEFsbCBmZW1hbGUgY2hhbXBpb25zIHRvIGRhdGUgaGF2ZSBiZWVuIE1MRS9JRk9DRS1hZmZpbGlhdGVkLiIsIDcwKSwgY29sbGFwc2U9IlxuIikKCm5hdGhhbl93X2ZlbWFsZXMgKwogICMgbm93IGFuIGFzdGVyaXNrIHRvIHNldCBvZmYgdGhlIGZlbWFsZSBzY29yZXMsIGFuZCBhIGNhcHRpb24KICBhbm5vdGF0ZSgndGV4dCcsIHggPSAyMDE4LjUsIHkgPSAzOSwgbGFiZWw9IioiLCBmYW1pbHkgPSAiTGF0byIsIHNpemUgPSA4KSArCiAgbGFicyhjYXB0aW9uID0gY2FwdGlvbikgKwogIHRoZW1lKHBsb3QuY2FwdGlvbiA9IGVsZW1lbnRfdGV4dChmYW1pbHkgPSAiTGF0byIsIHNpemU9OCwgaGp1c3Q9MCwgbWFyZ2luPW1hcmdpbih0PTE1KSkpCmBgYAoK

Creative Commons License