Background

The dplyr package provides a language, or grammar, for data manipulation.

The language contains a number of verbs that operate on tables in data frames.

The most commonly used verbs operate on a single data frame:

There are also a number of join verbs that merge several data frames into one.

The tidyr package provides additional verbs, such as pivot_longer and pivot_wider for reshaping data frames.

The single table verbs can also be used with group_by() and ungroup() to work a group at a time instead of applying to the entire data frame.

An experimental alternative to using group_by() and ungroup() is to use the .by argument with thesw verbs.

The design of dplyr is strongly motivated by SQL.

The basic transformation verbs are described in the Data Transformation chapter of R for Data Science.

Utilities in tidyr are described in the Tidy Data chapter.

Some data sets for illustration:

Selecting Variables

Data sets often contain hundreds of even thousands of variables.

A useful first step is to select a group of interesting variables.

A reasonable selection of the EPA vehicle variables:

newmpg1 <- select(newmpg,
                  make, model, year,
                  cty = city08, hwy = highway08,
                  trans = trany, cyl = cylinders,
                  fuel = fuelType1, displ)
head(newmpg1, 2)
## # A tibble: 2 × 9
##   make       model               year   cty   hwy trans          cyl fuel  displ
##   <chr>      <chr>              <dbl> <dbl> <dbl> <chr>        <dbl> <chr> <dbl>
## 1 Alfa Romeo Spider Veloce 2000  1985    19    25 Manual 5-spd     4 Regu…   2  
## 2 Ferrari    Testarossa          1985     9    14 Manual 5-spd    12 Regu…   4.9

Variables can be given new names in a select call; you can also rename them later with rename.

Some variations (the documentation for select() gives full details):

select(newmpg1, year : trans) |> head(2)
## # A tibble: 2 × 4
##    year   cty   hwy trans       
##   <dbl> <dbl> <dbl> <chr>       
## 1  1985    19    25 Manual 5-spd
## 2  1985     9    14 Manual 5-spd
select(newmpg1, -year, -trans) |> head(2)
## # A tibble: 2 × 7
##   make       model                cty   hwy   cyl fuel             displ
##   <chr>      <chr>              <dbl> <dbl> <dbl> <chr>            <dbl>
## 1 Alfa Romeo Spider Veloce 2000    19    25     4 Regular Gasoline   2  
## 2 Ferrari    Testarossa             9    14    12 Regular Gasoline   4.9
select(newmpg1, - (year : trans)) |> head(2)
## # A tibble: 2 × 5
##   make       model                cyl fuel             displ
##   <chr>      <chr>              <dbl> <chr>            <dbl>
## 1 Alfa Romeo Spider Veloce 2000     4 Regular Gasoline   2  
## 2 Ferrari    Testarossa            12 Regular Gasoline   4.9

Numerical column specifications can also be used:

select(newmpg1, 1, 3) |> head(2)
## # A tibble: 2 × 2
##   make        year
##   <chr>      <dbl>
## 1 Alfa Romeo  1985
## 2 Ferrari     1985
select(newmpg1, 1 : 3) |> head(2)
## # A tibble: 2 × 3
##   make       model               year
##   <chr>      <chr>              <dbl>
## 1 Alfa Romeo Spider Veloce 2000  1985
## 2 Ferrari    Testarossa          1985
select(newmpg1, - (1 : 3)) |> head(2)
## # A tibble: 2 × 6
##     cty   hwy trans          cyl fuel             displ
##   <dbl> <dbl> <chr>        <dbl> <chr>            <dbl>
## 1    19    25 Manual 5-spd     4 Regular Gasoline   2  
## 2     9    14 Manual 5-spd    12 Regular Gasoline   4.9

Some helper functions can also be used in the column specifications.

The most useful ones are starts_with, ends_with, and contains.

select(newmpg, starts_with("fuel")) |> head(2)
## # A tibble: 2 × 5
##   fuelCost08 fuelCostA08 fuelType fuelType1        fuelType2
##        <dbl>       <dbl> <chr>    <chr>            <chr>    
## 1       2200           0 Regular  Regular Gasoline <NA>     
## 2       4150           0 Regular  Regular Gasoline <NA>
select(newmpg, contains("city")) |> head(2)
## # A tibble: 2 × 12
##   city08 city08U cityA08 cityA08U cityCD cityE cityUF rangeCity rangeCityA UCity
##    <dbl>   <dbl>   <dbl>    <dbl>  <dbl> <dbl>  <dbl>     <dbl>      <dbl> <dbl>
## 1     19       0       0        0      0     0      0         0          0  23.3
## 2      9       0       0        0      0     0      0         0          0  11  
## # ℹ 2 more variables: UCityA <dbl>, phevCity <dbl>

By default these helpers ignore case.

These can also be preceded by a minus modifier to omit columns:

select(newmpg1, -contains("m")) |> head(2)
## # A tibble: 2 × 7
##    year   cty   hwy trans          cyl fuel             displ
##   <dbl> <dbl> <dbl> <chr>        <dbl> <chr>            <dbl>
## 1  1985    19    25 Manual 5-spd     4 Regular Gasoline   2  
## 2  1985     9    14 Manual 5-spd    12 Regular Gasoline   4.9

contains requires a literal string; matches is analogous but its argument is interpreted as a regular expression pattern to be matched.

select(newmpg, matches("^[Ff]uel")) |> head(2)
## # A tibble: 2 × 5
##   fuelCost08 fuelCostA08 fuelType fuelType1        fuelType2
##        <dbl>       <dbl> <chr>    <chr>            <chr>    
## 1       2200           0 Regular  Regular Gasoline <NA>     
## 2       4150           0 Regular  Regular Gasoline <NA>

Sometimes it is useful to move a few columns to the front; the everything helper can be used for that.

The original NYC flights table:

head(flights, 2)
## # A tibble: 2 × 19
##    year month   day dep_time sched_dep_time dep_delay arr_time sched_arr_time
##   <int> <int> <int>    <int>          <int>     <dbl>    <int>          <int>
## 1  2013     1     1      517            515         2      830            819
## 2  2013     1     1      533            529         4      850            830
## # ℹ 11 more variables: arr_delay <dbl>, carrier <chr>, flight <int>,
## #   tailnum <chr>, origin <chr>, dest <chr>, air_time <dbl>, distance <dbl>,
## #   hour <dbl>, minute <dbl>, time_hour <dttm>

Moving air_time to the front:

select(flights, air_time, everything()) |> head(2)
## # A tibble: 2 × 19
##   air_time  year month   day dep_time sched_dep_time dep_delay arr_time
##      <dbl> <int> <int> <int>    <int>          <int>     <dbl>    <int>
## 1      227  2013     1     1      517            515         2      830
## 2      227  2013     1     1      533            529         4      850
## # ℹ 11 more variables: sched_arr_time <int>, arr_delay <dbl>, carrier <chr>,
## #   flight <int>, tailnum <chr>, origin <chr>, dest <chr>, distance <dbl>,
## #   hour <dbl>, minute <dbl>, time_hour <dttm>

The where helper function allows columns to be selected based on a predicate applied to the full column:

select(newmpg1, where(anyNA)) |> head(2)
## # A tibble: 2 × 3
##   trans          cyl displ
##   <chr>        <dbl> <dbl>
## 1 Manual 5-spd     4   2  
## 2 Manual 5-spd    12   4.9
select(flights, where(anyNA)) |> head(2)
## # A tibble: 2 × 6
##   dep_time dep_delay arr_time arr_delay tailnum air_time
##      <int>     <dbl>    <int>     <dbl> <chr>      <dbl>
## 1      517         2      830        11 N14228       227
## 2      533         4      850        20 N24211       227

If there are missing values in your data it is usually a good idea to review them to make sure you understand what they mean.

Columns can also be selected using list operations:

vars <- c("make", "model", "year", "city08", "trany", "cylinders",
          "fuelType1", "displ")
newmpg[vars] |> head(2)
## # A tibble: 2 × 8
##   make       model               year city08 trany     cylinders fuelType1 displ
##   <chr>      <chr>              <dbl>  <dbl> <chr>         <dbl> <chr>     <dbl>
## 1 Alfa Romeo Spider Veloce 2000  1985     19 Manual 5…         4 Regular …   2  
## 2 Ferrari    Testarossa          1985      9 Manual 5…        12 Regular …   4.9

Filtering Rows

The filter function picks out the subset of rows that satisfy one or more conditions.

All cars with city MPG at or above 140 and model year 2023:

filter(newmpg1, cty >= 140, year == 2023)
## # A tibble: 4 × 9
##   make    model                         year   cty   hwy trans   cyl fuel  displ
##   <chr>   <chr>                        <dbl> <dbl> <dbl> <chr> <dbl> <chr> <dbl>
## 1 Lucid   Air Pure AWD with 19 inch w…  2023   141   140 Auto…    NA Elec…    NA
## 2 Lucid   Air Touring AWD with 19 inc…  2023   141   140 Auto…    NA Elec…    NA
## 3 Hyundai Ioniq 6 Long range RWD (18 …  2023   153   127 Auto…    NA Elec…    NA
## 4 Hyundai Ioniq 6 Standard Range RWD    2023   151   120 Auto…    NA Elec…    NA

All flights from New York to Des Moines on January 1, 2013:

filter(flights, day == 1, month == 1, dest == "DSM") |>
    select(year : dep_time, origin)
## # A tibble: 1 × 5
##    year month   day dep_time origin
##   <int> <int> <int>    <int> <chr> 
## 1  2013     1     1     2119 EWR

Comparisons

The basic comparison operators are

  • ==, !=
  • <, <=
  • >, >=

Be sure to use ==, not a single = character.

Be careful about floating point equality tests:

x <- 1 - 0.9
x == 0.1
## [1] FALSE
near(x, 0.1)
## [1] TRUE

Equality comparisons can be used on character vectors and factors.

Order comparisons can be used on character vectors, but may produce surprising results, or results that vary with different locale settings.

%in% can be used to match against one of several options:

filter(flights, day %in% 1 : 2, month == 1, dest == "DSM") |>
    select(year : dep_time, origin)
## # A tibble: 2 × 5
##    year month   day dep_time origin
##   <int> <int> <int>    <int> <chr> 
## 1  2013     1     1     2119 EWR   
## 2  2013     1     2     1921 EWR

Logical Operations

The basic logical operations on vectors are

  • & – and
  • | – or
  • ! – not
  • xor – exclusive or

Truth tables:

x <- c(TRUE, TRUE,  FALSE, FALSE)
y <- c(TRUE, FALSE, TRUE,  FALSE)
! x
## [1] FALSE FALSE  TRUE  TRUE
x & y
## [1]  TRUE FALSE FALSE FALSE
x | y
## [1]  TRUE  TRUE  TRUE FALSE
xor(x, y)
## [1] FALSE  TRUE  TRUE FALSE

Make sure not to confuse the vectorized logical operators & and | with the scalar flow control operators && and ||.

The previous flights example can also be written as

filter(flights, day == 1 | day == 2, month == 1, dest == "DSM") |>
    select(year : dep_time, origin)
## # A tibble: 2 × 5
##    year month   day dep_time origin
##   <int> <int> <int>    <int> <chr> 
## 1  2013     1     1     2119 EWR   
## 2  2013     1     2     1921 EWR

It can also be written as

filter(flights, (day == 1 | day == 2) & month == 1 & dest == "DSM") |>
    select(year : dep_time, origin)
## # A tibble: 2 × 5
##    year month   day dep_time origin
##   <int> <int> <int>    <int> <chr> 
## 1  2013     1     1     2119 EWR   
## 2  2013     1     2     1921 EWR

Missing Values

filter only keeps values corresponding to TRUE predicate values; FALSE and NA values are dropped.

Arithmetic and comparison operators will always produce NA values if one operand is NA.

1 + NA
## [1] NA
NA < 1
## [1] NA
NA == NA
## [1] NA

In a few cases a non-NA result can be determined:

TRUE | NA
## [1] TRUE
NA & FALSE
## [1] FALSE
NA ^ 0
## [1] 1

is.na can be used to also select rows with NA values.

Non-electric vehicles with zero or NA for displ:

filter(newmpg1, displ == 0 | is.na(displ), fuel != "Electricity")
## # A tibble: 2 × 9
##   make   model     year   cty   hwy trans          cyl fuel             displ
##   <chr>  <chr>    <dbl> <dbl> <dbl> <chr>        <dbl> <chr>            <dbl>
## 1 Subaru RX Turbo  1985    22    28 Manual 5-spd    NA Regular Gasoline    NA
## 2 Subaru RX Turbo  1985    21    27 Manual 5-spd    NA Regular Gasoline    NA

Flights with NA for both dep_time and arr_time:

filter(flights, is.na(dep_time) & is.na(arr_time)) |>
    head(2) |>
    select(year : arr_time)
## # A tibble: 2 × 7
##    year month   day dep_time sched_dep_time dep_delay arr_time
##   <int> <int> <int>    <int>          <int>     <dbl>    <int>
## 1  2013     1     1       NA           1630        NA       NA
## 2  2013     1     1       NA           1935        NA       NA

An alternative approach that does not use filter would be

idx <- with(flights, is.na(dep_time) & is.na(arr_time))
flights[idx, ] |>
    head(2) |>
    select(year : arr_time)
## # A tibble: 2 × 7
##    year month   day dep_time sched_dep_time dep_delay arr_time
##   <int> <int> <int>    <int>          <int>     <dbl>    <int>
## 1  2013     1     1       NA           1630        NA       NA
## 2  2013     1     1       NA           1935        NA       NA

Exploring the EPA Data

Sorting out fuel types:

select(newmpg, contains("fuelType")) |> unique()
## # A tibble: 14 × 3
##    fuelType                    fuelType1         fuelType2  
##    <chr>                       <chr>             <chr>      
##  1 Regular                     Regular Gasoline  <NA>       
##  2 Premium                     Premium Gasoline  <NA>       
##  3 Diesel                      Diesel            <NA>       
##  4 CNG                         Natural Gas       <NA>       
##  5 Gasoline or natural gas     Regular Gasoline  Natural Gas
##  6 Gasoline or E85             Regular Gasoline  E85        
##  7 Electricity                 Electricity       <NA>       
##  8 Gasoline or propane         Regular Gasoline  Propane    
##  9 Premium or E85              Premium Gasoline  E85        
## 10 Midgrade                    Midgrade Gasoline <NA>       
## 11 Premium Gas or Electricity  Premium Gasoline  Electricity
## 12 Regular Gas and Electricity Regular Gasoline  Electricity
## 13 Premium and Electricity     Premium Gasoline  Electricity
## 14 Regular Gas or Electricity  Regular Gasoline  Electricity

Variables with missing values:

select(newmpg, where(anyNA)) |> head(2)
## # A tibble: 2 × 17
##   cylinders displ drive      eng_dscr trany guzzler trans_dscr tCharger sCharger
##       <dbl> <dbl> <chr>      <chr>    <chr> <chr>   <chr>      <lgl>    <chr>   
## 1         4   2   Rear-Whee… (FFS)    Manu… <NA>    <NA>       NA       <NA>    
## 2        12   4.9 Rear-Whee… (GUZZLE… Manu… T       <NA>       NA       <NA>    
## # ℹ 8 more variables: atvType <chr>, fuelType2 <chr>, rangeA <chr>,
## #   evMotor <chr>, mfrCode <chr>, c240Dscr <chr>, c240bDscr <chr>,
## #   startStop <chr>

Among the reduced data set:

select(newmpg1, where(anyNA)) |> names()
## [1] "trans" "cyl"   "displ"

Unique missing value patterns:

incomplete_cases <- function(data) data[! complete.cases(data), ]

select(newmpg1, trans, cyl, displ, fuel) |>
    incomplete_cases() |>
    unique()
## # A tibble: 9 × 4
##   trans                              cyl displ fuel            
##   <chr>                            <dbl> <dbl> <chr>           
## 1 <NA>                                NA  NA   Electricity     
## 2 <NA>                                 8   5.8 Regular Gasoline
## 3 <NA>                                 6   4.1 Regular Gasoline
## 4 Manual 5-spd                        NA  NA   Regular Gasoline
## 5 Manual 5-spd                        NA   1.3 Regular Gasoline
## 6 Automatic (A1)                      NA  NA   Electricity     
## 7 Automatic (variable gear ratios)    NA  NA   Electricity     
## 8 Automatic (A1)                      NA   0   Electricity     
## 9 Automatic (A2)                      NA  NA   Electricity

An alternative to defining incomplete_cases():

select(newmpg1, trans, cyl, displ, fuel) |>
    filter(if_any(everything(), is.na)) |>
    unique()
## # A tibble: 9 × 4
##   trans                              cyl displ fuel            
##   <chr>                            <dbl> <dbl> <chr>           
## 1 <NA>                                NA  NA   Electricity     
## 2 <NA>                                 8   5.8 Regular Gasoline
## 3 <NA>                                 6   4.1 Regular Gasoline
## 4 Manual 5-spd                        NA  NA   Regular Gasoline
## 5 Manual 5-spd                        NA   1.3 Regular Gasoline
## 6 Automatic (A1)                      NA  NA   Electricity     
## 7 Automatic (variable gear ratios)    NA  NA   Electricity     
## 8 Automatic (A1)                      NA   0   Electricity     
## 9 Automatic (A2)                      NA  NA   Electricity

Looking at the counts:

select(newmpg1, trans, cyl, displ, fuel) |>
    incomplete_cases() |>
    count(trans, cyl, displ, fuel)
## # A tibble: 9 × 5
##   trans                              cyl displ fuel                 n
##   <chr>                            <dbl> <dbl> <chr>            <int>
## 1 Automatic (A1)                      NA   0   Electricity          1
## 2 Automatic (A1)                      NA  NA   Electricity        567
## 3 Automatic (A2)                      NA  NA   Electricity         63
## 4 Automatic (variable gear ratios)    NA  NA   Electricity          9
## 5 Manual 5-spd                        NA   1.3 Regular Gasoline     1
## 6 Manual 5-spd                        NA  NA   Regular Gasoline     2
## 7 <NA>                                 6   4.1 Regular Gasoline     1
## 8 <NA>                                 8   5.8 Regular Gasoline     1
## 9 <NA>                                NA  NA   Electricity          9

count(trans, cyl, displ, fuel) is essentially an abbreviation of

group_by(trans, cyl, displ, fuel) |>
    summarize(n = n()) |>
    ungroup()

Another approach that avoids specifying the variables with missing values:

incomplete_cases(newmpg1) |>
    count(across(where(anyNA)), fuel)
## # A tibble: 9 × 5
##   trans                              cyl displ fuel                 n
##   <chr>                            <dbl> <dbl> <chr>            <int>
## 1 Automatic (A1)                      NA   0   Electricity          1
## 2 Automatic (A1)                      NA  NA   Electricity        567
## 3 Automatic (A2)                      NA  NA   Electricity         63
## 4 Automatic (variable gear ratios)    NA  NA   Electricity          9
## 5 Manual 5-spd                        NA   1.3 Regular Gasoline     1
## 6 Manual 5-spd                        NA  NA   Regular Gasoline     2
## 7 <NA>                                 6   4.1 Regular Gasoline     1
## 8 <NA>                                 8   5.8 Regular Gasoline     1
## 9 <NA>                                NA  NA   Electricity          9

Cylinders for electric vehicles:

filter(newmpg1, fuel == "Electricity") |>
    count(is.na(cyl))
## # A tibble: 1 × 2
##   `is.na(cyl)`     n
##   <lgl>        <int>
## 1 TRUE           649

Displacement for electric vehicles:

filter(newmpg1, fuel == "Electricity") |>
    count(is.na(displ))
## # A tibble: 2 × 2
##   `is.na(displ)`     n
##   <lgl>          <int>
## 1 FALSE              1
## 2 TRUE             648

Electric vehicles with non-missing displacement:

filter(newmpg1, fuel == "Electricity", ! is.na(displ))
## # A tibble: 1 × 9
##   make       model   year   cty   hwy trans            cyl fuel        displ
##   <chr>      <chr>  <dbl> <dbl> <dbl> <chr>          <dbl> <chr>       <dbl>
## 1 Mitsubishi i-MiEV  2016   126    99 Automatic (A1)    NA Electricity     0

Some things to think about for electric vehicles:

Other considerations:

Adding New Variables

mutate() can be used to define new variables or modify existing ones.

New variables are added at the end.

Later variables can be defined in terms of earlier ones.

mutate() does not modify the original data frame; it creates a new one with the specified modifications.

fl <- select(flights, year, month, day, air_time, ends_with("delay"))
mutate(fl,
       gain = dep_delay - arr_delay,
       air_hours = air_time / 60,
       gain_per_hour = gain / air_hours) |>
    head(2)
## # A tibble: 2 × 9
##    year month   day air_time dep_delay arr_delay  gain air_hours gain_per_hour
##   <int> <int> <int>    <dbl>     <dbl>     <dbl> <dbl>     <dbl>         <dbl>
## 1  2013     1     1      227         2        11    -9      3.78         -2.38
## 2  2013     1     1      227         4        20   -16      3.78         -4.23

transmute() keeps only the new variables.

transmute(flights,
          gain = dep_delay - arr_delay,
          air_hours = air_time / 60,
          gain_per_hour = gain / air_hours) |>
    head(2)
## # A tibble: 2 × 3
##    gain air_hours gain_per_hour
##   <dbl>     <dbl>         <dbl>
## 1    -9      3.78         -2.38
## 2   -16      3.78         -4.23

Useful operators include:

A reasonable guess is that flights with NA values for both dep_time and arr_time were cancelled:

fl <- mutate(flights,
             cancelled = is.na(dep_time) & is.na(arr_time))

For the geyser data set from the MASS package we used lag to obtain the duration of the previous eruption,

data(geyser, package = "MASS")
geyser2 <- mutate(geyser,
                  prev_duration = lag(duration))
head(geyser2, 4)
##   waiting duration prev_duration
## 1      80 4.016667            NA
## 2      71 2.150000      4.016667
## 3      57 4.000000      2.150000
## 4      80 4.000000      4.000000
thm <- theme_minimal() +
    theme(text = element_text(size = 16))
ggplot(geyser2) +
    geom_point(aes(x = prev_duration,
                   y = waiting)) +
    thm
## Warning: Removed 1 rows containing missing values (`geom_point()`).

The top five vehicles in cty gas mileage (there may be more than five if there is a tie for fifth place):

mutate(newmpg1, cty_rank = min_rank(desc(cty))) |>
    filter(cty_rank <= 5)
## # A tibble: 8 × 10
##   make    model                year   cty   hwy trans   cyl fuel  displ cty_rank
##   <chr>   <chr>               <dbl> <dbl> <dbl> <chr> <dbl> <chr> <dbl>    <int>
## 1 Hyundai Ioniq Electric       2017   150   122 Auto…    NA Elec…    NA        5
## 2 Hyundai Ioniq Electric       2018   150   122 Auto…    NA Elec…    NA        5
## 3 Hyundai Ioniq Electric       2019   150   122 Auto…    NA Elec…    NA        5
## 4 Tesla   Model 3 Standard R…  2021   150   133 Auto…    NA Elec…    NA        5
## 5 Hyundai Ioniq 6 Long range…  2023   153   127 Auto…    NA Elec…    NA        1
## 6 Hyundai Ioniq 6 Standard R…  2023   151   120 Auto…    NA Elec…    NA        3
## 7 Hyundai Ioniq 6 Long range…  2024   153   127 Auto…    NA Elec…    NA        1
## 8 Hyundai Ioniq 6 Standard R…  2024   151   120 Auto…    NA Elec…    NA        3

An alternative to explicitly computing ranks and filtering is the slice_max() utility function:

slice_max(newmpg1, cty, n = 5)
## # A tibble: 8 × 9
##   make    model                         year   cty   hwy trans   cyl fuel  displ
##   <chr>   <chr>                        <dbl> <dbl> <dbl> <chr> <dbl> <chr> <dbl>
## 1 Hyundai Ioniq 6 Long range RWD (18 …  2023   153   127 Auto…    NA Elec…    NA
## 2 Hyundai Ioniq 6 Long range RWD (18 …  2024   153   127 Auto…    NA Elec…    NA
## 3 Hyundai Ioniq 6 Standard Range RWD    2023   151   120 Auto…    NA Elec…    NA
## 4 Hyundai Ioniq 6 Standard Range RWD    2024   151   120 Auto…    NA Elec…    NA
## 5 Hyundai Ioniq Electric                2017   150   122 Auto…    NA Elec…    NA
## 6 Hyundai Ioniq Electric                2018   150   122 Auto…    NA Elec…    NA
## 7 Hyundai Ioniq Electric                2019   150   122 Auto…    NA Elec…    NA
## 8 Tesla   Model 3 Standard Range Plus…  2021   150   133 Auto…    NA Elec…    NA

The percent_rank function produces a ‘percentile rank’ (but between 0 and 1).

The top 5% of countries in life expectancy in 2007 from the gapminder data set:

library(gapminder)
gm <- filter(gapminder, year == 2007)
gm_top <- filter(gm, percent_rank(desc(lifeExp)) <= 0.05)
gm_top
## # A tibble: 8 × 6
##   country          continent  year lifeExp       pop gdpPercap
##   <fct>            <fct>     <int>   <dbl>     <int>     <dbl>
## 1 Australia        Oceania    2007    81.2  20434176    34435.
## 2 Hong Kong, China Asia       2007    82.2   6980412    39725.
## 3 Iceland          Europe     2007    81.8    301931    36181.
## 4 Israel           Asia       2007    80.7   6426679    25523.
## 5 Japan            Asia       2007    82.6 127467972    31656.
## 6 Spain            Europe     2007    80.9  40448191    28821.
## 7 Sweden           Europe     2007    80.9   9031088    33860.
## 8 Switzerland      Europe     2007    81.7   7554661    37506.

And the bottom 5%:

filter(gm, percent_rank(lifeExp) <= 0.05)
## # A tibble: 8 × 6
##   country      continent  year lifeExp      pop gdpPercap
##   <fct>        <fct>     <int>   <dbl>    <int>     <dbl>
## 1 Afghanistan  Asia       2007    43.8 31889923      975.
## 2 Angola       Africa     2007    42.7 12420476     4797.
## 3 Lesotho      Africa     2007    42.6  2012649     1569.
## 4 Mozambique   Africa     2007    42.1 19951656      824.
## 5 Sierra Leone Africa     2007    42.6  6144562      863.
## 6 Swaziland    Africa     2007    39.6  1133066     4513.
## 7 Zambia       Africa     2007    42.4 11746035     1271.
## 8 Zimbabwe     Africa     2007    43.5 12311143      470.

Within these results the original row order is retained.

The arrange function can be used to change this.

Arranging Rows

arrange() reorders the rows according to one or more variables.

By default the order is smallest to largest; using desc() reverses this.

Additional variables will be used to break ties.

Arranging the top 5% of countries in life expectancy in 2007 by life expectancy:

arrange(gm_top, desc(lifeExp))
## # A tibble: 8 × 6
##   country          continent  year lifeExp       pop gdpPercap
##   <fct>            <fct>     <int>   <dbl>     <int>     <dbl>
## 1 Japan            Asia       2007    82.6 127467972    31656.
## 2 Hong Kong, China Asia       2007    82.2   6980412    39725.
## 3 Iceland          Europe     2007    81.8    301931    36181.
## 4 Switzerland      Europe     2007    81.7   7554661    37506.
## 5 Australia        Oceania    2007    81.2  20434176    34435.
## 6 Spain            Europe     2007    80.9  40448191    28821.
## 7 Sweden           Europe     2007    80.9   9031088    33860.
## 8 Israel           Asia       2007    80.7   6426679    25523.

To order by continent name and order by life expectancy within continent:

arrange(gm_top, continent, desc(lifeExp))
## # A tibble: 8 × 6
##   country          continent  year lifeExp       pop gdpPercap
##   <fct>            <fct>     <int>   <dbl>     <int>     <dbl>
## 1 Japan            Asia       2007    82.6 127467972    31656.
## 2 Hong Kong, China Asia       2007    82.2   6980412    39725.
## 3 Israel           Asia       2007    80.7   6426679    25523.
## 4 Iceland          Europe     2007    81.8    301931    36181.
## 5 Switzerland      Europe     2007    81.7   7554661    37506.
## 6 Spain            Europe     2007    80.9  40448191    28821.
## 7 Sweden           Europe     2007    80.9   9031088    33860.
## 8 Australia        Oceania    2007    81.2  20434176    34435.

Missing values are sorted to the end:

df <- data.frame(x = c(5, 2, NA, 3))
arrange(df, x)
##    x
## 1  2
## 2  3
## 3  5
## 4 NA
arrange(df, desc(x))
##    x
## 1  5
## 2  3
## 3  2
## 4 NA

Using is.na you can arrange to put them at the beginning:

arrange(df, desc(is.na(x)), x)
##    x
## 1 NA
## 2  2
## 3  3
## 4  5

Summarizing and Grouping

summarize() collapses a table down to one row of summaries.

The average and maximal departure and arrival delays for flights, along with the total number of flights:

summarize(flights,
          ave_dep_delay = mean(dep_delay, na.rm = TRUE),
          ave_arr_delay = mean(arr_delay, na.rm = TRUE),
          max_dep_delay = max(dep_delay, na.rm = TRUE),
          max_arr_delay = max(arr_delay, na.rm = TRUE),
          n = n())
## # A tibble: 1 × 5
##   ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay      n
##           <dbl>         <dbl>         <dbl>         <dbl>  <int>
## 1          12.6          6.90          1301          1272 336776

Grouped Summaries

summarize() is most commonly used with grouped data to provide group level summaries.

Grouping delay summaries by destination:

fl_dest <-
    group_by(flights, dest) |>
    summarize(ave_dep_delay = mean(dep_delay, na.rm = TRUE),
              ave_arr_delay = mean(arr_delay, na.rm = TRUE),
              max_dep_delay = max(dep_delay, na.rm = TRUE),
              max_arr_delay = max(arr_delay, na.rm = TRUE),
              n = n()) |>
    ungroup()
head(fl_dest, 5)
## # A tibble: 5 × 6
##   dest  ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay     n
##   <chr>         <dbl>         <dbl>         <dbl>         <dbl> <int>
## 1 ABQ           13.7           4.38           142           153   254
## 2 ACK            6.46          4.85           219           221   265
## 3 ALB           23.6          14.4            323           328   439
## 4 ANC           12.9          -2.5             75            39     8
## 5 ATL           12.5          11.3            898           895 17215

Calling ungroup is not always necessary but sometimes it is, so it is a good habit to get into.

An alternative to using group_by() and ungroup() is to use the .by argument to summarize():

flights |>
    summarize(ave_dep_delay = mean(dep_delay, na.rm = TRUE),
              ave_arr_delay = mean(arr_delay, na.rm = TRUE),
              max_dep_delay = max(dep_delay, na.rm = TRUE),
              max_arr_delay = max(arr_delay, na.rm = TRUE),
              n = n(),
              .by = dest) |>
    arrange(dest) |>      ## to match group_by() result
    head()
## # A tibble: 6 × 6
##   dest  ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay     n
##   <chr>         <dbl>         <dbl>         <dbl>         <dbl> <int>
## 1 ABQ           13.7           4.38           142           153   254
## 2 ACK            6.46          4.85           219           221   265
## 3 ALB           23.6          14.4            323           328   439
## 4 ANC           12.9          -2.5             75            39     8
## 5 ATL           12.5          11.3            898           895 17215
## 6 AUS           13.0           6.02           351           349  2439

This is currently marked as Experimental in the documentation.

The group level summaries can be further transformed, for example to identify the top 10 destinations for average delays:

mutate(fl_dest, rank = min_rank(desc(ave_dep_delay))) |>
    filter(rank <= 10)
## # A tibble: 10 × 7
##    dest  ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay     n  rank
##    <chr>         <dbl>         <dbl>         <dbl>         <dbl> <int> <int>
##  1 ALB            23.6          14.4           323           328   439     9
##  2 BHM            29.7          16.9           325           291   297     4
##  3 CAE            35.6          41.8           222           224   116     1
##  4 DSM            26.2          19.0           341           322   569     7
##  5 JAC            26.5          28.1           198           175    25     6
##  6 MSN            23.6          20.2           340           364   572    10
##  7 OKC            30.6          30.6           280           262   346     3
##  8 RIC            23.6          20.1           483           463  2454     8
##  9 TUL            34.9          33.7           251           262   315     2
## 10 TYS            28.5          24.1           291           281   631     5

Missing Values

Most summaries will be NA if any of the data values being summarized are NA.

Typically, summary functions can be called with na.rm = TRUE to produce the summary for all non-missing values.

summarize(flights,
          ave_dep_delay = mean(dep_delay, na.rm = TRUE),
          ave_arr_delay = mean(arr_delay, na.rm = TRUE),
          max_dep_delay = max(dep_delay, na.rm = TRUE),
          max_arr_delay = max(arr_delay, na.rm = TRUE),
          n = n())
## # A tibble: 1 × 5
##   ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay      n
##           <dbl>         <dbl>         <dbl>         <dbl>  <int>
## 1          12.6          6.90          1301          1272 336776

You can also use filter to remove rows with NA values in the variables to be summarized:

filter(flights, ! is.na(dep_delay), ! is.na(arr_delay)) |>
    summarize(ave_dep_delay = mean(dep_delay),
              ave_arr_delay = mean(arr_delay),
              max_dep_delay = max(dep_delay),
              max_arr_delay = max(arr_delay),
              n = n())
## # A tibble: 1 × 5
##   ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay      n
##           <dbl>         <dbl>         <dbl>         <dbl>  <int>
## 1          12.6          6.90          1301          1272 327346

The two approaches do produce different counts.

Counts

The helper function n() provides a count of the number of rows in the table being summarized.

Including counts with grouped summaries is usually a good idea.

Summaries based on small counts are likely to be much less reliable than ones based on larger counts.

This is reflected in higher variability among averages for destinations with fewer flights:

ggplot(fl_dest) +
    geom_point(aes(x = n,
                   y = ave_arr_delay),
               na.rm = TRUE) +
    thm

A plotly version allows the individual destinations to be identified easily:

library(plotly)
p <- ggplot(fl_dest) +
    geom_point(aes(x = n,
                   y = ave_arr_delay,
                   text = dest),
               na.rm = FALSE) +
    thm
ggplotly(p, tooltip = "text")

It would be more user-friendly to show the airport name than the three letter FAA code.

Useful Summaries

Some useful summary functions:

  • Location: mean, median

  • Spread: sd, IQR, mad

  • Rank: min, quantile, max

  • Position: first, nth, last

  • Counts: n.

Airline Arrival Delays

Different Delay Measures

Sometimes it is useful to look at subsets or truncated versions of a variable.

Airlines like to arrange schedules so they are often early.

ggplot(flights, aes(x = arr_delay,
                    y = after_stat(density))) +
    geom_histogram(binwidth = 1,
                   na.rm = TRUE,
                   fill = "deepskyblue3") +
    xlim(c(-100, 250)) +
    thm

This reduces the average arrival delay but may not help travelers much.

Two options:

  • Consider only the positive delays for the average.

  • Treat negative delays as zero (zero-truncation).

The mean of the positive delays can be computed using subsetting:

mean(arr_delay[arr_delay > 0])

The mean of the zero-truncated delays can be computed using the pmax function:

mean(pmax(arr_delay, 0))

The summaries are

fl_arr <- filter(flights, ! is.na(arr_delay))
fl_avs <-
    group_by(fl_arr, dest) |>
    summarize(ave = mean(arr_delay),
              ave_pos = mean(arr_delay[arr_delay > 0]),
              ave_pos_zero = mean(pmax(arr_delay, 0)),
              n = n()) |>
    ungroup()
head(fl_avs)
## # A tibble: 6 × 5
##   dest    ave ave_pos ave_pos_zero     n
##   <chr> <dbl>   <dbl>        <dbl> <int>
## 1 ABQ    4.38    41.9        17.7    254
## 2 ACK    4.85    28.6        11.3    264
## 3 ALB   14.4     52.1        22.9    418
## 4 ANC   -2.5     12.4         7.75     8
## 5 ATL   11.3     37.8        17.8  16837
## 6 AUS    6.02    40.6        16.6   2411

One approach to visualizing the results with color coding the three different summaries:

  • View the fl_avs table as a wide format with avs, ave_pos, and ave_pos_zero as three different measurement types.

  • Use pivot_longer() to convert to a long format with a variable which to hold the summary type and a variable delay to hold the summary value:

library(tidyr)
fl <- pivot_longer(fl_avs, 2 : 4,
                   names_to = "which",
                   values_to = "delay")
arrange(fl, dest)
## # A tibble: 312 × 4
##    dest      n which        delay
##    <chr> <int> <chr>        <dbl>
##  1 ABQ     254 ave           4.38
##  2 ABQ     254 ave_pos      41.9 
##  3 ABQ     254 ave_pos_zero 17.7 
##  4 ACK     264 ave           4.85
##  5 ACK     264 ave_pos      28.6 
##  6 ACK     264 ave_pos_zero 11.3 
##  7 ALB     418 ave          14.4 
##  8 ALB     418 ave_pos      52.1 
##  9 ALB     418 ave_pos_zero 22.9 
## 10 ANC       8 ave          -2.5 
## # ℹ 302 more rows

A plot is then easy to create:

ggplot(fl, aes(x = n,
               y = delay,
               color = which)) +
    geom_point() +
    thm
## Warning: Removed 1 rows containing missing values (`geom_point()`).

An alternative:

  • Use one layer per summary type.

  • Specify the summary type for each layer as a color aesthetic value.

ggplot(fl_avs, aes(x = n)) +
    geom_point(aes(y = ave,
                   color = "ave")) +
    geom_point(aes(y = ave_pos,
                   color = "ave_pos")) +
    geom_point(aes(y = ave_pos_zero,
                   color = "ave_pos_zero")) +
    thm
## Warning: Removed 1 rows containing missing values (`geom_point()`).

The one missing value case:

incomplete_cases(fl_avs)
## # A tibble: 1 × 5
##   dest    ave ave_pos ave_pos_zero     n
##   <chr> <dbl>   <dbl>        <dbl> <int>
## 1 LEX     -22     NaN            0     1

Delay Proportions

Proportions satisfying a condition can be computed as a mean of the logical expression for the condition:

The proportion of all flights that were delayed:

mean(fl_arr$arr_delay > 0)
## [1] 0.4063101

Grouped by destination:

fl_props <-
    group_by(fl_arr, dest) |>
    summarize(
        pd0 = mean(arr_delay > 0),   ## proportion delayed
        pd10 = mean(arr_delay > 10), ## more than 10 minutes
        pd20 = mean(arr_delay > 20), ## more than 20 minutes
        n = n()) |>
    ungroup()
fl_props
## # A tibble: 104 × 5
##    dest    pd0  pd10  pd20     n
##    <chr> <dbl> <dbl> <dbl> <int>
##  1 ABQ   0.421 0.339 0.268   254
##  2 ACK   0.394 0.246 0.167   264
##  3 ALB   0.440 0.333 0.273   418
##  4 ANC   0.625 0.125 0.125     8
##  5 ATL   0.472 0.309 0.211 16837
##  6 AUS   0.408 0.284 0.216  2411
##  7 AVL   0.456 0.257 0.207   261
##  8 BDL   0.350 0.262 0.223   412
##  9 BGR   0.383 0.299 0.246   358
## 10 BHM   0.450 0.361 0.305   269
## # ℹ 94 more rows

These can be visualized similarly:

pivot_longer(fl_props, 2 : 4,
             names_to = "which",
             values_to = "prop") |>
    ggplot(aes(x = n,
               y = prop,
               color = which)) +
    geom_point() +
    thm

Grouped Mutate and Filter

mutate(), filter(), arrange(), slice_max(), and slice_min() can be used on a group level.

Standardizing Within Groups

mutate within groups can be used for standardizing within a group.

For example, to compute the proportion of a continent’s population living in each country for each year:

<!- no longer need pop = as.numeric(pop) since sum() switches to double on overflow–>

gm <- group_by(gapminder, continent, year) |>
    mutate(ppop = pop / sum(pop)) |>
    ungroup()

For Oceania (in this data set) and years 1952 and 2007:

gm_oc <- filter(gm, continent == "Oceania", year %in% c(1952, 2007)) |>
    arrange(year)
gm_oc
## # A tibble: 4 × 7
##   country     continent  year lifeExp      pop gdpPercap  ppop
##   <fct>       <fct>     <int>   <dbl>    <int>     <dbl> <dbl>
## 1 Australia   Oceania    1952    69.1  8691212    10040. 0.813
## 2 New Zealand Oceania    1952    69.4  1994794    10557. 0.187
## 3 Australia   Oceania    2007    81.2 20434176    34435. 0.832
## 4 New Zealand Oceania    2007    80.2  4115771    25185. 0.168

A quick check that within year and continent the proportions add up to one:

group_by(gm_oc, continent, year) |> summarize(sum_ppop = sum(ppop))
## # A tibble: 2 × 3
## # Groups:   continent [1]
##   continent  year sum_ppop
##   <fct>     <int>    <dbl>
## 1 Oceania    1952        1
## 2 Oceania    2007        1

Focus on the 2 countries in each continent with the highest proportions of the continent’s population:

gmt2 <- group_by(gm, continent, year) |>
    slice_max(ppop, n = 2) |>
    ungroup() |>
    filter(continent != "Oceania") ## drop as there are only two
ggplot(gmt2, aes(x = year, y = ppop,
                 group = country, color = country)) +
    geom_line() + geom_point() +
    facet_wrap(~ continent, scales = "free_y") +
    thm

Another approach:

  • Create separate plots for each continent.

  • Adjust the themes a little.

  • Use facet_wrap to get the facet label.

  • Arrange the plots, e.g. using patchwork operators.

A useful feature is that the %+% operator can be used to change the base data frame for a plot:

pAf <- ggplot(filter(gmt2, continent == "Africa"),
              aes(x = year, y = ppop, col = country)) +
    geom_line() + geom_point() +
    facet_wrap(~ continent) +
    thm +
    theme(legend.position = "top",
          legend.title = element_blank())
thm_nx <- theme(axis.title.x = element_blank())
thm_ny <- theme(axis.title.y = element_blank())
pAm <- pAf %+% filter(gmt2, continent == "Americas")
pAs <- pAf %+% filter(gmt2, continent == "Asia")
pEu <- pAf %+% filter(gmt2, continent == "Europe") +
    guides(color = guide_legend(nrow = 2))
library(patchwork)
((pAf + thm_nx) | (pAm + thm_nx + thm_ny)) /
    (pAs | (pEu + thm_ny))

Changes Over Time

Changes over time within a country can also be examined with a grouped mutate and the lag operator.

For variables collected over time the lag function can be used to compute the previous value of a variable:

gm_us <- filter(gapminder,
                country == "United States") |>
    transmute(year,
              lifeExp,
              prev_lifeExp = lag(lifeExp))
gm_us
## # A tibble: 12 × 3
##     year lifeExp prev_lifeExp
##    <int>   <dbl>        <dbl>
##  1  1952    68.4         NA  
##  2  1957    69.5         68.4
##  3  1962    70.2         69.5
##  4  1967    70.8         70.2
##  5  1972    71.3         70.8
##  6  1977    73.4         71.3
##  7  1982    74.6         73.4
##  8  1987    75.0         74.6
##  9  1992    76.1         75.0
## 10  1997    76.8         76.1
## 11  2002    77.3         76.8
## 12  2007    78.2         77.3

This can then be used to compute the change from one period to the next:

mutate(gm_us,
       le_change = lifeExp - prev_lifeExp)
## # A tibble: 12 × 4
##     year lifeExp prev_lifeExp le_change
##    <int>   <dbl>        <dbl>     <dbl>
##  1  1952    68.4         NA      NA    
##  2  1957    69.5         68.4     1.05 
##  3  1962    70.2         69.5     0.720
##  4  1967    70.8         70.2     0.550
##  5  1972    71.3         70.8     0.580
##  6  1977    73.4         71.3     2.04 
##  7  1982    74.6         73.4     1.27 
##  8  1987    75.0         74.6     0.370
##  9  1992    76.1         75.0     1.07 
## 10  1997    76.8         76.1     0.720
## 11  2002    77.3         76.8     0.5  
## 12  2007    78.2         77.3     0.932

Use this idea to compute changes in lifeExp and gdpPercap for all countries in a grouped mutate:

gm <- group_by(gapminder, country) |>
    mutate(
        le_change = lifeExp - lag(lifeExp),
        gdp_change = gdpPercap - lag(gdpPercap),
        gdp_pct_change =
            100 * gdp_change / lag(gdpPercap)) |>
    ungroup()

slice_max() can then be applied to data grouped by country to extract the rows with the highest gdp_pct_change for each country:

group_by(gm, country) |>
    slice_max(gdp_pct_change, n = 1) |>
    ungroup() |>
    select(1 : 3, contains("gdp"))
## # A tibble: 142 × 6
##    country     continent  year gdpPercap gdp_change gdp_pct_change
##    <fct>       <fct>     <int>     <dbl>      <dbl>          <dbl>
##  1 Afghanistan Asia       2007      975.       248.           34.1
##  2 Albania     Europe     2002     4604.      1411.           44.2
##  3 Algeria     Africa     1972     4183.       936.           28.8
##  4 Angola      Africa     2007     4797.      2024.           73.0
##  5 Argentina   Americas   2007    12779.      3982.           45.3
##  6 Australia   Oceania    1967    14526.      2309.           18.9
##  7 Austria     Europe     1957     8843.      2706.           44.1
##  8 Bahrain     Asia       2007    29796.      6392.           27.3
##  9 Bangladesh  Asia       2007     1391.       255.           22.4
## 10 Belgium     Europe     1972    16672.      3523.           26.8
## # ℹ 132 more rows

Grouping by continent allows the worst drop in life expectancy for each continent to be found:

group_by(gm, continent) |>
    slice_min(le_change, n = 1) |>
    ungroup() |>
    select(-contains("gdp"))
## # A tibble: 5 × 6
##   country     continent  year lifeExp      pop le_change
##   <fct>       <fct>     <int>   <dbl>    <int>     <dbl>
## 1 Rwanda      Africa     1992    23.6  7290203   -20.4  
## 2 El Salvador Americas   1977    56.7  4282586    -1.51 
## 3 Cambodia    Asia       1977    31.2  6978607    -9.10 
## 4 Montenegro  Europe     2002    74.0   720230    -1.46 
## 5 Australia   Oceania    1967    71.1 11872264     0.170

Without grouping, the row with best percent GDP change overall can be computed:

slice_max(gm, gdp_pct_change, n = 1) |>
    select(country, continent, year, gdp_pct_change)
## # A tibble: 1 × 4
##   country continent  year gdp_pct_change
##   <fct>   <fct>     <int>          <dbl>
## 1 Libya   Africa     1967           178.

The computation of the best GDP growth years by country can be combined into a single pipe:

group_by(gapminder, country) |>
    mutate(gdp_change = gdpPercap - lag(gdpPercap),
           gdp_pct_change = 100 * gdp_change / lag(gdpPercap)) |>
    ungroup() |>
    group_by(country) |>
    slice_max(gdp_pct_change, n = 1) |>
    ungroup() |>
    select(1 : 3, contains("gdp"))

Since the groupings of the two steps are the same and grouped mutate preserves the group structure, the middle ungroup/group_by can be dropped:

group_by(gapminder, country) |>
    mutate(gdp_change = gdpPercap - lag(gdpPercap),
           gdp_pct_change = 100 * gdp_change / lag(gdpPercap)) |>
    slice_max(gdp_pct_change, n = 1) |>
    ungroup() |>
    select(1 : 3, contains("gdp"))

You need to be very careful making an optimization like this!

Joins

This interactive plot uses the tooltip to show the airport code for the destination:

library(plotly)
p <- ggplot(fl_dest) +
    geom_point(aes(x = n,
                   y = ave_arr_delay,
                   text = dest)) +
    thm
## Warning in geom_point(aes(x = n, y = ave_arr_delay, text = dest)): Ignoring
## unknown aesthetics: text
ggplotly(p, tooltip = "text")

A nicer approach would be to show the airport name.

The airports table provides this information:

head(airports, 2)
## # A tibble: 2 × 8
##   faa   name                            lat   lon   alt    tz dst   tzone       
##   <chr> <chr>                         <dbl> <dbl> <dbl> <dbl> <chr> <chr>       
## 1 04G   Lansdowne Airport              41.1 -80.6  1044    -5 A     America/New…
## 2 06A   Moton Field Municipal Airport  32.5 -85.7   264    -6 A     America/Chi…

We need to get this information into the fl_dest data frame.

As another example, we might want to explore whether aircraft age is related to delays.

The planes table provides the year of manufacture:

head(planes, 2)
## # A tibble: 2 × 9
##   tailnum  year type               manufacturer model engines seats speed engine
##   <chr>   <int> <chr>              <chr>        <chr>   <int> <int> <int> <chr> 
## 1 N10156   2004 Fixed wing multi … EMBRAER      EMB-…       2    55    NA Turbo…
## 2 N102UW   1998 Fixed wing multi … AIRBUS INDU… A320…       2   182    NA Turbo…

Again, we need to merge this information with the data in the flights table.

One way to do this is with a join operation.

Joins are a common operation in data bases.

dplyr distinguishes between two kinds of joins:

These animated explanations may be helpful.

Mutating Joins

Mutating joins combine variables and rows from two tables by matching rows on keys.

  • A primary key is a variable, or combination of variables, in a table that uniquely identify the rows of the table.

  • A foreign key is a variable, or combination of variables, that uniquely identify a row in some table (usually, but not necessarily, a different table).

The simplest join operation is an inner join: Only rows for which the key appears in both tables are retained.

Outer joins retain some other rows with NA values for new variables:

Inner and left joins are the most common.

The key of one of the tables should usually be a primary key that uniquely identifies the table’s rows.

For a left join, the right table key should usually use a primary key.

It is usually a good idea to make sure a primary key candidate really does uniquely identify rows.

For the planes table the tailnum variable is a primary key;

count(planes, tailnum) |> filter(n > 1)
## # A tibble: 0 × 2
## # ℹ 2 variables: tailnum <chr>, n <int>

For the flights data a primary key needs to use multiple variables.

A reasonable guess is to use the date, carrier and flight number, but this isn’t quite enough:

count(flights, year, month, day, flight, carrier) |>
    filter(n > 1) |>
    head(2)
## # A tibble: 2 × 6
##    year month   day flight carrier     n
##   <int> <int> <int>  <int> <chr>   <int>
## 1  2013     6     8   2269 WN          2
## 2  2013     6    15   2269 WN          2

Adding the origin airport code does produce a primary key:

count(flights, year, month, day, flight, carrier, origin) |>
    filter(n > 1)
## # A tibble: 0 × 7
## # ℹ 7 variables: year <int>, month <int>, day <int>, flight <int>,
## #   carrier <chr>, origin <chr>, n <int>

To relate delays to aircraft age we can start with summaries by tailnum and year (year would be needed if we had data for more than 2013):

fl <- filter(flights, ! is.na(dep_time), ! is.na(arr_time)) |>
    group_by(year, tailnum) |>
    summarize(ave_dep_delay = mean(dep_delay, na.rm = TRUE),
              ave_arr_delay = mean(arr_delay, na.rm = TRUE),
              n = n()) |>
    ungroup()
head(fl)
## # A tibble: 6 × 5
##    year tailnum ave_dep_delay ave_arr_delay     n
##   <int> <chr>           <dbl>         <dbl> <int>
## 1  2013 D942DN          31.5          31.5      4
## 2  2013 N0EGMQ           8.49          9.98   354
## 3  2013 N10156          17.8          12.7    146
## 4  2013 N102UW           8             2.94    48
## 5  2013 N103US          -3.20         -6.93    46
## 6  2013 N104UW          10.1           1.80    46

A reduced planes table with year renamed to avoid a conflict:

pl <- select(planes, tailnum, plane_year = year)
pl
## # A tibble: 3,322 × 2
##    tailnum plane_year
##    <chr>        <int>
##  1 N10156        2004
##  2 N102UW        1998
##  3 N103US        1999
##  4 N104UW        1999
##  5 N10575        2002
##  6 N105UW        1999
##  7 N107US        1999
##  8 N108UW        1999
##  9 N109UW        1999
## 10 N110UW        1999
## # ℹ 3,312 more rows

Now

  • use a left join with tailnum as the key to bring the plane_year value into the summary table

  • and then compute the plane age:

fl_age <- left_join(fl, pl, "tailnum") |>
    mutate(plane_age = year - plane_year)
fl_age
## # A tibble: 4,037 × 7
##     year tailnum ave_dep_delay ave_arr_delay     n plane_year plane_age
##    <int> <chr>           <dbl>         <dbl> <int>      <int>     <int>
##  1  2013 D942DN         31.5          31.5       4         NA        NA
##  2  2013 N0EGMQ          8.49          9.98    354         NA        NA
##  3  2013 N10156         17.8          12.7     146       2004         9
##  4  2013 N102UW          8             2.94     48       1998        15
##  5  2013 N103US         -3.20         -6.93     46       1999        14
##  6  2013 N104UW         10.1           1.80     46       1999        14
##  7  2013 N10575         22.3          20.7     271       2002        11
##  8  2013 N105UW          2.58         -0.267    45       1999        14
##  9  2013 N107US         -0.463        -5.73     41       1999        14
## 10  2013 N108UW          4.22         -1.25     60       1999        14
## # ℹ 4,027 more rows

An initial plot:

ggplot(fl_age,
       aes(x = plane_age,
           y = ave_dep_delay)) +
    geom_point() +
    thm

Jittering and adjusting the point size may help:

ggplot(fl_age,
       aes(x = plane_age,
           y = ave_dep_delay)) +
    geom_point(position = "jitter",
               size = 0.1) +
    thm

Adding a smooth and adjusting the ranges may also help:

ggplot(fl_age,
       aes(x = plane_age,
           y = ave_dep_delay)) +
    geom_point(position = "jitter",
               size = 0.1) +
    geom_smooth() +
    xlim(0, 30) +
    ylim(c(-20, 80)) +
    thm

Another option is to compute means or medians within years:

flm <- group_by(fl_age, plane_age) |>
    summarize(ave_dep_delay =
                  mean(ave_dep_delay,
                       na.rm = TRUE)) |>
    ungroup()
ggplot(flm,
       aes(x = plane_age,
           y = ave_dep_delay)) +
    geom_point(na.rm = TRUE) +
    thm

Both the smooth and the means show a modest increase over the first 10 years.

A left join can also be used to add the airport name from the airport table to the fl_dest table.

The primary key in the airport table is the three-letter airport code in the faa variable:

head(airports, 2)
## # A tibble: 2 × 8
##   faa   name                            lat   lon   alt    tz dst   tzone       
##   <chr> <chr>                         <dbl> <dbl> <dbl> <dbl> <chr> <chr>       
## 1 04G   Lansdowne Airport              41.1 -80.6  1044    -5 A     America/New…
## 2 06A   Moton Field Municipal Airport  32.5 -85.7   264    -6 A     America/Chi…

The foreign key in the fl_dest table to match is the dest variable:

head(fl_dest, 2)
## # A tibble: 2 × 6
##   dest  ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay     n
##   <chr>         <dbl>         <dbl>         <dbl>         <dbl> <int>
## 1 ABQ           13.7           4.38           142           153   254
## 2 ACK            6.46          4.85           219           221   265

The left_join function allows this link to be made by specifying the key as c("dest" = "faa"):

fl_dest <- left_join(fl_dest,
                     select(airports, faa, dest_name = name),
                     c("dest" = "faa"))
select(fl_dest, dest_name, everything()) |> head(2)
## # A tibble: 2 × 7
##   dest_name  dest  ave_dep_delay ave_arr_delay max_dep_delay max_arr_delay     n
##   <chr>      <chr>         <dbl>         <dbl>         <dbl>         <dbl> <int>
## 1 Albuquerq… ABQ           13.7           4.38           142           153   254
## 2 Nantucket… ACK            6.46          4.85           219           221   265

Now a tooltip can provide a more accessible label:

p <- ggplot(fl_dest) +
    geom_point(aes(x = n,
                   y = ave_arr_delay,
                   text = paste(dest,
                                dest_name,
                                sep = ": "))) +
    thm
ggplotly(p, tooltip = "text")

Filtering Joins

Filtering joins only affect rows; they do not add variables.

  • semi_join(x, y) returns all rows in x with a match in y:

  • anti_join(x, y) returns all rows in x without a match in y:

semi_join is useful for matching filtered results back to a table.

The plot of delay against age showed some surprisingly old aircraft:

top_age <- slice_max(fl_age, plane_age, n = 2)
top_age
## # A tibble: 3 × 7
##    year tailnum ave_dep_delay ave_arr_delay     n plane_year plane_age
##   <int> <chr>           <dbl>         <dbl> <int>      <int>     <int>
## 1  2013 N381AA           5.95          3.5     22       1956        57
## 2  2013 N201AA          17.4          14.1     51       1959        54
## 3  2013 N567AA           3.72         -4.30    61       1959        54

We can identify these aircraft with a semi-join:

semi_join(planes, top_age, "tailnum")
## # A tibble: 3 × 9
##   tailnum  year type               manufacturer model engines seats speed engine
##   <chr>   <int> <chr>              <chr>        <chr>   <int> <int> <int> <chr> 
## 1 N201AA   1959 Fixed wing single… CESSNA       150         1     2    90 Recip…
## 2 N381AA   1956 Fixed wing multi … DOUGLAS      DC-7…       4   102   232 Recip…
## 3 N567AA   1959 Fixed wing single… DEHAVILLAND  OTTE…       1    16    95 Recip…

The corresponding flights can also be picked out with a semi-join:

top_age_flights <- semi_join(flights, top_age, "tailnum")
head(top_age_flights, 2)
## # A tibble: 2 × 19
##    year month   day dep_time sched_dep_time dep_delay arr_time sched_arr_time
##   <int> <int> <int>    <int>          <int>     <dbl>    <int>          <int>
## 1  2013     1     3      909            700       129     1103            850
## 2  2013     1     3       NA            920        NA       NA           1245
## # ℹ 11 more variables: arr_delay <dbl>, carrier <chr>, flight <int>,
## #   tailnum <chr>, origin <chr>, dest <chr>, air_time <dbl>, distance <dbl>,
## #   hour <dbl>, minute <dbl>, time_hour <dttm>

Looking at the carriers or destinations might help:

select(top_age_flights, carrier, dest, everything()) |> head(4)
## # A tibble: 4 × 19
##   carrier dest   year month   day dep_time sched_dep_time dep_delay arr_time
##   <chr>   <chr> <int> <int> <int>    <int>          <int>     <dbl>    <int>
## 1 AA      ORD    2013     1     3      909            700       129     1103
## 2 AA      DFW    2013     1     3       NA            920        NA       NA
## 3 AA      DFW    2013     1     9     1423           1430        -7     1727
## 4 AA      DFW    2013     1    10     1238           1240        -2     1525
## # ℹ 10 more variables: sched_arr_time <int>, arr_delay <dbl>, flight <int>,
## #   tailnum <chr>, origin <chr>, air_time <dbl>, distance <dbl>, hour <dbl>,
## #   minute <dbl>, time_hour <dttm>

The carrier summary:

count(top_age_flights, carrier)
## # A tibble: 1 × 2
##   carrier     n
##   <chr>   <int>
## 1 AA        139

A check of the documentation for planes shows that American and one other airline report fleet numbers rather than tail numbers.

Anti-joins are useful for detecting join problems.

For example, many flights have tail numbers that do not match entries in planes:

fl <- anti_join(flights, planes, "tailnum")
nrow(fl)
## [1] 52606
count(fl, tailnum)
## # A tibble: 722 × 2
##    tailnum     n
##    <chr>   <int>
##  1 D942DN      4
##  2 N0EGMQ    371
##  3 N14628      1
##  4 N149AT     22
##  5 N16632     11
##  6 N17627      2
##  7 N1EAMQ    210
##  8 N200AA     34
##  9 N24633      8
## 10 N261AV      6
## # ℹ 712 more rows

Join Issues

  • Make sure a primary key you want to use has no missing values.

  • Make sure a primary key you want to use really uniquely identifies a row.

  • Make sure every value of your foreign key you want to use to link to another table matches a row in the linked table (anti_join is a good way to do this).

  • Check that your foreign key has no missing values.

Data Base Interface

Current data on airline flights in the US is available from the Bureau of Transportation Statistics at

http://www.transtats.bts.gov/DL_SelectFields.asp?Table_ID=236

The data are available in one compressed CSV file per month.

Loaded into R, one month of data requires about 300 Mb of memory.

Data for 1987–2008 was compiled for a DataExpo competition of the American Statistical Association held in 2009.

The data are available on the Linux systems in the directory /group/statsoft/data/DataExpo09 as compressed CSV files and as a SQLite data base.

Loading the data into R would require about 70 Gb of memory, which is not practical.

Using the data base is a good alternative.

SQLite is a simple relational data base that works with data stored on a local disk.

More sophisticated data bases allow data to be stored on multiple machines or in the cloud

These data bases are usually accessed by connecting to a server, which usually involves an authentication process.

For SQLite the connection is set up by linking to a file.

The RSQLite package provides an interface to SQLite.

The DBI package provides a high level interface for using data bases.

Analogous packages exist for Python and Perl.

The first step is to connect to the data base.

library(dplyr)
library(DBI)
library(RSQLite)
db <- dbConnect(SQLite(), "/group/statsoft/data/DataExpo09/ontime.sqlite3")

There is only one table:

dbListTables(db)
## [1] "ontime"

tbl creates a tibble-like object for a data base table.

fl <- tbl(db, "ontime")

This object is not a regular tibble, but many tibble operations work on it.

To find the variables in the table:

tbl_vars(fl)
##  [1] "Year"              "Month"             "DayofMonth"
##  [4] "DayOfWeek"         "DepTime"           "CRSDepTime"
##  [7] "ArrTime"           "CRSArrTime"        "UniqueCarrier"
## [10] "FlightNum"         "TailNum"           "ActualElapsedTime"
## [13] "CRSElapsedTime"    "AirTime"           "ArrDelay"
## [16] "DepDelay"          "Origin"            "Dest"
## [19] "Distance"          "TaxiIn"            "TaxiOut"
## [22] "Cancelled"         "CancellationCode"  "Diverted"
## [25] "CarrierDelay"      "WeatherDelay"      "NASDelay"
## [28] "SecurityDelay"     "LateAircraftDelay"

Many dplyr operations are supported:

sm <- summarize(fl, last_year = max(Year, na.rm = TRUE), n = n())

This computation is fast because it does not actually access the data base.

An operation that needs the data, such as printing the result, will access the data base and take time:

sm
## # Source:   lazy query [?? x 2]
## # Database: sqlite 3.37.2
## #   [/mnt/nfs/clasnetappvm/group/statsoft/data/DataExpo09/ontime.sqlite3]
##   last_year         n
##       <int>     <int>
## 1      2008 118914458

The sm object essentially contains a SQL query, which can be examined using sql_render from the dbplyr package:

dbplyr::sql_render(sm)
## <SQL> SELECT MAX(`Year`) AS `last_year`, COUNT(*) AS `n`
## FROM `ontime`

Flights to CID in 2008:

fl_cid <- filter(fl, Year == 2008, Dest == "CID")
dbplyr::sql_render(fl_cid)
## <SQL> SELECT *
## FROM `ontime`
## WHERE ((`Year` = 2008.0) AND (`Dest` = 'CID'))

Number of flights to CID in 2008:

sm <- summarize(fl_cid, n = n())
dbplyr::sql_render(sm)
## <SQL> SELECT COUNT(*) AS `n`
## FROM `ontime`
## WHERE ((`Year` = 2008.0) AND (`Dest` = 'CID'))
sm
## # Source:   lazy query [?? x 1]
## # Database: sqlite 3.37.2
## #   [/mnt/nfs/clasnetappvm/group/statsoft/data/DataExpo09/ontime.sqlite3]
##       n
##   <int>
## 1  2961

Breakdown by where the flights originated:

sm <- group_by(fl_cid, Origin) |>
    summarize(n = n()) |>
    ungroup()
dbplyr::sql_render(sm)
## <SQL> SELECT `Origin`, COUNT(*) AS `n`
## FROM `ontime`
## WHERE ((`Year` = 2008.0) AND (`Dest` = 'CID'))
## GROUP BY `Origin`
sm
## # Source:   lazy query [?? x 2]
## # Database: sqlite 3.37.2
## #   [/mnt/nfs/clasnetappvm/group/statsoft/data/DataExpo09/ontime.sqlite3]
##   Origin     n
##   <chr>  <int>
## 1 ATL      119
## 2 BNA        1
## 3 CVG      292
## 4 DEN      230
## 5 DFW      519
## 6 DTW      345
## 7 MSP      241
## 8 ORD     1214

Many base R functions can be converted to SQL:

sm <- group_by(fl_cid, Origin) |>
    summarize(ave_arr_del_pos = mean(pmax(ArrDelay, 0), na.rm = TRUE)) |>
    ungroup()
dbplyr::sql_render(sm)
## <SQL> SELECT `Origin`, AVG(MAX(`ArrDelay`, 0.0)) AS `ave_arr_del_pos`
## FROM `ontime`
## WHERE ((`Year` = 2008.0) AND (`Dest` = 'CID'))
## GROUP BY `Origin`
sm_cid <- sm
sm
## # Source:   lazy query [?? x 2]
## # Database: sqlite 3.37.2
## #   [/mnt/nfs/clasnetappvm/group/statsoft/data/DataExpo09/ontime.sqlite3]
##   Origin ave_arr_del_pos
##   <chr>            <dbl>
## 1 ATL               17.4
## 2 BNA               18
## 3 CVG               12.4
## 4 DEN               12.1
## 5 DFW               11.3
## 6 DTW               15.0
## 7 MSP               20.1
## 8 ORD               21.6

mutate() can also be used:

fl_cid_gain <- mutate(fl_cid, Gain = DepDelay - ArrDelay)
dbplyr::sql_render(fl_cid_gain)
## <SQL> SELECT `Year`, `Month`, `DayofMonth`, `DayOfWeek`, `DepTime`, `CRSDepTime`, `ArrTime`, `CRSArrTime`, `UniqueCarrier`, `FlightNum`, `TailNum`, `ActualElapsedTime`, `CRSElapsedTime`, `AirTime`, `ArrDelay`, `DepDelay`, `Origin`, `Dest`, `Distance`, `TaxiIn`, `TaxiOut`, `Cancelled`, `CancellationCode`, `Diverted`, `CarrierDelay`, `WeatherDelay`, `NASDelay`, `SecurityDelay`, `LateAircraftDelay`, `DepDelay` - `ArrDelay` AS `Gain`
## FROM `ontime`
## WHERE ((`Year` = 2008.0) AND (`Dest` = 'CID'))
group_by(fl_cid_gain, Origin) |>
    summarize(mean_gain = mean(Gain)) |>
    ungroup()
## # Source:   lazy query [?? x 2]
## # Database: sqlite 3.37.2
## #   [/mnt/nfs/clasnetappvm/group/statsoft/data/DataExpo09/ontime.sqlite3]
##   Origin mean_gain
##   <chr>      <dbl>
## 1 ATL        8.24
## 2 BNA      -26
## 3 CVG       -1.60
## 4 DEN        0.278
## 5 DFW        4.12
## 6 DTW        3.04
## 7 MSP       -0.357
## 8 ORD       -1.36

Once a filtered result is small enough it can be brought over as a regular tibble using

tb <- collect(fl_cid)

There are some limitations to operations that can be done on tables in data bases.

Defining and using a function works for a tibble:

pos_mean <- function(x) mean(pmax(x, 0))
group_by(tb, Origin) |>
    summarize(ave_arr_del_pos = pos_mean(ArrDelay)) |>
    ungroup()

But this fails:

group_by(fl_cid, Origin) |>
    summarize(ave_arr_del_pos = pos_mean(ArrDelay)) |>
    ungroup()
## Error: no such function: pos_mean

Some operations may require some modifications.

Recomputing sm_cid:

fl <- tbl(db, "ontime")
fl_cid <- filter(fl, Year == 2008, Dest == "CID")
sm_cid <- group_by(fl_cid, Origin) |>
    summarize(ave_arr_del_pos = mean(pmax(ArrDelay, 0), na.rm = TRUE)) |>
    ungroup()

Joining with a local table does not work:

ap <- select(nycflights13::airports, faa, name)
sm_cid <- left_join(sm_cid, ap, c(Origin = "faa"))
## Error in `auto_copy()`:
## ! `x` and `y` must share the same src.
## ℹ set `copy` = TRUE (may be slow).

But this does work:

sm_cid <- left_join(sm_cid, ap, c(Origin = "faa"), copy = TRUE)

This copies the ap table to the data base:

dbListTables(db)
## [1] "dbplyr_001"   "ontime"       "sqlite_stat1" "sqlite_stat4"

Finish by disconnecting from the data base:

dbDisconnect(db)

This is not necessary for SQLite but is for other data bases, so it is good practice.

Some notes:

The dbplyr web site provides more information on using data bases with dplyr.

Some other approaches for handling larger-than-memory data:

Other Wrangling Tools

Some additional tools provided by tidyr:

We have already seen pivot_longer() and pivot_wider().

A newer alternative to separate() is to use separate_wider_position() and separate_wider_delim()

Separating Variables

A data set from the WHO on tuberculosis cases:

head(who, 2)
## # A tibble: 2 × 60
##   country   iso2  iso3   year new_sp_m014 new_sp_m1524 new_sp_m2534 new_sp_m3544
##   <chr>     <chr> <chr> <dbl>       <dbl>        <dbl>        <dbl>        <dbl>
## 1 Afghanis… AF    AFG    1980          NA           NA           NA           NA
## 2 Afghanis… AF    AFG    1981          NA           NA           NA           NA
## # ℹ 52 more variables: new_sp_m4554 <dbl>, new_sp_m5564 <dbl>,
## #   new_sp_m65 <dbl>, new_sp_f014 <dbl>, new_sp_f1524 <dbl>,
## #   new_sp_f2534 <dbl>, new_sp_f3544 <dbl>, new_sp_f4554 <dbl>,
## #   new_sp_f5564 <dbl>, new_sp_f65 <dbl>, new_sn_m014 <dbl>,
## #   new_sn_m1524 <dbl>, new_sn_m2534 <dbl>, new_sn_m3544 <dbl>,
## #   new_sn_m4554 <dbl>, new_sn_m5564 <dbl>, new_sn_m65 <dbl>,
## #   new_sn_f014 <dbl>, new_sn_f1524 <dbl>, new_sn_f2534 <dbl>, …

Some of the remaining variable names:

names(who) |> tail(20)
##  [1] "new_ep_f1524" "new_ep_f2534" "new_ep_f3544" "new_ep_f4554" "new_ep_f5564"
##  [6] "new_ep_f65"   "newrel_m014"  "newrel_m1524" "newrel_m2534" "newrel_m3544"
## [11] "newrel_m4554" "newrel_m5564" "newrel_m65"   "newrel_f014"  "newrel_f1524"
## [16] "newrel_f2534" "newrel_f3544" "newrel_f4554" "newrel_f5564" "newrel_f65"

The variable new_sp_m2534, for example, represents the number of cases for

  • type sp (positive pulmonary smear);
  • sex m;
  • age between 25 and 34.

It would be better (tidier) to have separate variables for

  • count
  • type
  • sex
  • lower bound on age bracket
  • upper bound on age bracket

A pipeline to clean this up involves pivoting to long form and several separate and mutate steps.

The original data frame:

who
## # A tibble: 7,240 × 60
##    country  iso2  iso3   year new_sp_m014 new_sp_m1524 new_sp_m2534 new_sp_m3544
##    <chr>    <chr> <chr> <dbl>       <dbl>        <dbl>        <dbl>        <dbl>
##  1 Afghani… AF    AFG    1980          NA           NA           NA           NA
##  2 Afghani… AF    AFG    1981          NA           NA           NA           NA
##  3 Afghani… AF    AFG    1982          NA           NA           NA           NA
##  4 Afghani… AF    AFG    1983          NA           NA           NA           NA
##  5 Afghani… AF    AFG    1984          NA           NA           NA           NA
##  6 Afghani… AF    AFG    1985          NA           NA           NA           NA
##  7 Afghani… AF    AFG    1986          NA           NA           NA           NA
##  8 Afghani… AF    AFG    1987          NA           NA           NA           NA
##  9 Afghani… AF    AFG    1988          NA           NA           NA           NA
## 10 Afghani… AF    AFG    1989          NA           NA           NA           NA
## # ℹ 7,230 more rows
## # ℹ 52 more variables: new_sp_m4554 <dbl>, new_sp_m5564 <dbl>,
## #   new_sp_m65 <dbl>, new_sp_f014 <dbl>, new_sp_f1524 <dbl>,
## #   new_sp_f2534 <dbl>, new_sp_f3544 <dbl>, new_sp_f4554 <dbl>,
## #   new_sp_f5564 <dbl>, new_sp_f65 <dbl>, new_sn_m014 <dbl>,
## #   new_sn_m1524 <dbl>, new_sn_m2534 <dbl>, new_sn_m3544 <dbl>,
## #   new_sn_m4554 <dbl>, new_sn_m5564 <dbl>, new_sn_m65 <dbl>, …

First pivot to longer form:

who_clean <-
    pivot_longer(who,
                 new_sp_m014 : newrel_f65,
                 names_to = "key",
                 values_to = "count")
who_clean
## # A tibble: 405,440 × 6
##    country     iso2  iso3   year key          count
##    <chr>       <chr> <chr> <dbl> <chr>        <dbl>
##  1 Afghanistan AF    AFG    1980 new_sp_m014     NA
##  2 Afghanistan AF    AFG    1980 new_sp_m1524    NA
##  3 Afghanistan AF    AFG    1980 new_sp_m2534    NA
##  4 Afghanistan AF    AFG    1980 new_sp_m3544    NA
##  5 Afghanistan AF    AFG    1980 new_sp_m4554    NA
##  6 Afghanistan AF    AFG    1980 new_sp_m5564    NA
##  7 Afghanistan AF    AFG    1980 new_sp_m65      NA
##  8 Afghanistan AF    AFG    1980 new_sp_f014     NA
##  9 Afghanistan AF    AFG    1980 new_sp_f1524    NA
## 10 Afghanistan AF    AFG    1980 new_sp_f2534    NA
## # ℹ 405,430 more rows

Remove "new" or "new_" prefix:

who_clean <-
    pivot_longer(who,
                 new_sp_m014 : newrel_f65,
                 names_to = "key",
                 values_to = "count") |>
    mutate(key = sub("new_?", "", key))
who_clean
## # A tibble: 405,440 × 6
##    country     iso2  iso3   year key      count
##    <chr>       <chr> <chr> <dbl> <chr>    <dbl>
##  1 Afghanistan AF    AFG    1980 sp_m014     NA
##  2 Afghanistan AF    AFG    1980 sp_m1524    NA
##  3 Afghanistan AF    AFG    1980 sp_m2534    NA
##  4 Afghanistan AF    AFG    1980 sp_m3544    NA
##  5 Afghanistan AF    AFG    1980 sp_m4554    NA
##  6 Afghanistan AF    AFG    1980 sp_m5564    NA
##  7 Afghanistan AF    AFG    1980 sp_m65      NA
##  8 Afghanistan AF    AFG    1980 sp_f014     NA
##  9 Afghanistan AF    AFG    1980 sp_f1524    NA
## 10 Afghanistan AF    AFG    1980 sp_f2534    NA
## # ℹ 405,430 more rows

Separate key into type and sexage:

who_clean <-
    pivot_longer(who,
                 new_sp_m014 : newrel_f65,
                 names_to = "key",
                 values_to = "count") |>
    mutate(key = sub("new_?", "", key)) |>
    separate(key, c("type", "sexage"))
who_clean
## # A tibble: 405,440 × 7
##    country     iso2  iso3   year type  sexage count
##    <chr>       <chr> <chr> <dbl> <chr> <chr>  <dbl>
##  1 Afghanistan AF    AFG    1980 sp    m014      NA
##  2 Afghanistan AF    AFG    1980 sp    m1524     NA
##  3 Afghanistan AF    AFG    1980 sp    m2534     NA
##  4 Afghanistan AF    AFG    1980 sp    m3544     NA
##  5 Afghanistan AF    AFG    1980 sp    m4554     NA
##  6 Afghanistan AF    AFG    1980 sp    m5564     NA
##  7 Afghanistan AF    AFG    1980 sp    m65       NA
##  8 Afghanistan AF    AFG    1980 sp    f014      NA
##  9 Afghanistan AF    AFG    1980 sp    f1524     NA
## 10 Afghanistan AF    AFG    1980 sp    f2534     NA
## # ℹ 405,430 more rows

Separate sexage into sex and age:

who_clean <-
    pivot_longer(who,
                 new_sp_m014 : newrel_f65,
                 names_to = "key",
                 values_to = "count") |>
    mutate(key = sub("new_?", "", key)) |>
    separate(key, c("type", "sexage")) |>
    separate(sexage, c("sex", "age"),
             sep = 1)
who_clean
## # A tibble: 405,440 × 8
##    country     iso2  iso3   year type  sex   age   count
##    <chr>       <chr> <chr> <dbl> <chr> <chr> <chr> <dbl>
##  1 Afghanistan AF    AFG    1980 sp    m     014      NA
##  2 Afghanistan AF    AFG    1980 sp    m     1524     NA
##  3 Afghanistan AF    AFG    1980 sp    m     2534     NA
##  4 Afghanistan AF    AFG    1980 sp    m     3544     NA
##  5 Afghanistan AF    AFG    1980 sp    m     4554     NA
##  6 Afghanistan AF    AFG    1980 sp    m     5564     NA
##  7 Afghanistan AF    AFG    1980 sp    m     65       NA
##  8 Afghanistan AF    AFG    1980 sp    f     014      NA
##  9 Afghanistan AF    AFG    1980 sp    f     1524     NA
## 10 Afghanistan AF    AFG    1980 sp    f     2534     NA
## # ℹ 405,430 more rows

Fix up age categories:

who_clean <-
    pivot_longer(who,
                 new_sp_m014 : newrel_f65,
                 names_to = "key",
                 values_to = "count") |>
    mutate(key = sub("new_?", "", key)) |>
    separate(key, c("type", "sexage")) |>
    separate(sexage, c("sex", "age"),
             sep = 1) |>
    mutate(age = sub("014", "0014", age)) |>
    mutate(age = sub("65", "65Inf", age))
who_clean
## # A tibble: 405,440 × 8
##    country     iso2  iso3   year type  sex   age   count
##    <chr>       <chr> <chr> <dbl> <chr> <chr> <chr> <dbl>
##  1 Afghanistan AF    AFG    1980 sp    m     0014     NA
##  2 Afghanistan AF    AFG    1980 sp    m     1524     NA
##  3 Afghanistan AF    AFG    1980 sp    m     2534     NA
##  4 Afghanistan AF    AFG    1980 sp    m     3544     NA
##  5 Afghanistan AF    AFG    1980 sp    m     4554     NA
##  6 Afghanistan AF    AFG    1980 sp    m     5564     NA
##  7 Afghanistan AF    AFG    1980 sp    m     65Inf    NA
##  8 Afghanistan AF    AFG    1980 sp    f     0014     NA
##  9 Afghanistan AF    AFG    1980 sp    f     1524     NA
## 10 Afghanistan AF    AFG    1980 sp    f     2534     NA
## # ℹ 405,430 more rows

Finally, split age into age_lo and age-hi:

who_clean <-
    pivot_longer(who,
                 new_sp_m014 : newrel_f65,
                 names_to = "key",
                 values_to = "count") |>
    mutate(key = sub("new_?", "", key)) |>
    separate(key, c("type", "sexage")) |>
    separate(sexage, c("sex", "age"),
             sep = 1) |>
    mutate(age = sub("014", "0014", age)) |>
    mutate(age = sub("65", "65Inf", age)) |>
    separate(age, c("age_lo", "age_hi"),
             sep = 2,
             convert = TRUE)
who_clean
## # A tibble: 405,440 × 9
##    country     iso2  iso3   year type  sex   age_lo age_hi count
##    <chr>       <chr> <chr> <dbl> <chr> <chr>  <int>  <dbl> <dbl>
##  1 Afghanistan AF    AFG    1980 sp    m          0     14    NA
##  2 Afghanistan AF    AFG    1980 sp    m         15     24    NA
##  3 Afghanistan AF    AFG    1980 sp    m         25     34    NA
##  4 Afghanistan AF    AFG    1980 sp    m         35     44    NA
##  5 Afghanistan AF    AFG    1980 sp    m         45     54    NA
##  6 Afghanistan AF    AFG    1980 sp    m         55     64    NA
##  7 Afghanistan AF    AFG    1980 sp    m         65    Inf    NA
##  8 Afghanistan AF    AFG    1980 sp    f          0     14    NA
##  9 Afghanistan AF    AFG    1980 sp    f         15     24    NA
## 10 Afghanistan AF    AFG    1980 sp    f         25     34    NA
## # ℹ 405,430 more rows

A plot of total numbers of cases for a few countries over time:

filter(who_clean,
       iso3 %in% c("CAN", "DEU", "GBR", "USA")) |>
    group_by(year, iso3) |>
    summarize(count = sum(count, na.rm = TRUE)) |>
    ungroup() |>
    ggplot() +
    geom_line(aes(x = year,
                  y = count,
                  color = iso3)) +
    thm

Filling In Omitted Values

The Refugee Processing Center provides information from the United States Refugee Admissions Program (USRAP), including arrivals by state and nationality.

Three files are available locally:

These are Excel spread sheets.

Reading in the third file, with data for October 2017 through February 2018:

ref_url <-
    "http://homepage.stat.uiowa.edu/~luke/data/Arrivals-2018-03-05.xls"
ref_file <- "Arrivals-2018-03-05.xls"
if (! file.exists(ref_file))
    download.file(ref_url, ref_file)

library(readxl)
ref <- read_excel(ref_file, skip = 16) ## skip header
ref <- head(ref, -2) ## drop last two rows
names(ref)[1] <- "Destination"

The Destination column needs filling in (this is common in spread sheet data):

ref
## # A tibble: 500 × 18
##    Destination Nationality   Oct   Nov   Dec   Jan   Feb   Mar   Apr   May   Jun
##    <chr>       <chr>       <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
##  1 Alabama     <NA>            0     0     3     1     4     0     0     0     0
##  2 <NA>        Dem. Rep. …     0     0     3     0     4     0     0     0     0
##  3 <NA>        El Salvador     0     0     0     1     0     0     0     0     0
##  4 Alaska      <NA>            9     2     6     0     0     0     0     0     0
##  5 <NA>        Russia          0     0     6     0     0     0     0     0     0
##  6 <NA>        Ukraine         9     2     0     0     0     0     0     0     0
##  7 Arizona     <NA>           26    47    71    73   110     0     0     0     0
##  8 <NA>        Afghanistan     0     1    14    10    17     0     0     0     0
##  9 <NA>        Algeria         0     0     0     0     1     0     0     0     0
## 10 <NA>        Bhutan          2     3     8     2     0     0     0     0     0
## # ℹ 490 more rows
## # ℹ 7 more variables: Jul <dbl>, Aug <dbl>, Sep <dbl>, Cases <dbl>, Inds <dbl>,
## #   State <dbl>, U.S. <dbl>

Fill down the Destination:

ref_clean <-
    fill(ref, Destination)
ref_clean
## # A tibble: 500 × 18
##    Destination Nationality   Oct   Nov   Dec   Jan   Feb   Mar   Apr   May   Jun
##    <chr>       <chr>       <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
##  1 Alabama     <NA>            0     0     3     1     4     0     0     0     0
##  2 Alabama     Dem. Rep. …     0     0     3     0     4     0     0     0     0
##  3 Alabama     El Salvador     0     0     0     1     0     0     0     0     0
##  4 Alaska      <NA>            9     2     6     0     0     0     0     0     0
##  5 Alaska      Russia          0     0     6     0     0     0     0     0     0
##  6 Alaska      Ukraine         9     2     0     0     0     0     0     0     0
##  7 Arizona     <NA>           26    47    71    73   110     0     0     0     0
##  8 Arizona     Afghanistan     0     1    14    10    17     0     0     0     0
##  9 Arizona     Algeria         0     0     0     0     1     0     0     0     0
## 10 Arizona     Bhutan          2     3     8     2     0     0     0     0     0
## # ℹ 490 more rows
## # ℹ 7 more variables: Jul <dbl>, Aug <dbl>, Sep <dbl>, Cases <dbl>, Inds <dbl>,
## #   State <dbl>, U.S. <dbl>

Drop the totals:

ref_clean <-
    fill(ref, Destination) |>
    filter(! is.na(Nationality))
ref_clean
## # A tibble: 451 × 18
##    Destination Nationality   Oct   Nov   Dec   Jan   Feb   Mar   Apr   May   Jun
##    <chr>       <chr>       <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
##  1 Alabama     Dem. Rep. …     0     0     3     0     4     0     0     0     0
##  2 Alabama     El Salvador     0     0     0     1     0     0     0     0     0
##  3 Alaska      Russia          0     0     6     0     0     0     0     0     0
##  4 Alaska      Ukraine         9     2     0     0     0     0     0     0     0
##  5 Arizona     Afghanistan     0     1    14    10    17     0     0     0     0
##  6 Arizona     Algeria         0     0     0     0     1     0     0     0     0
##  7 Arizona     Bhutan          2     3     8     2     0     0     0     0     0
##  8 Arizona     Burma           6     3     7     1     9     0     0     0     0
##  9 Arizona     Burundi         0     0     0     5     7     0     0     0     0
## 10 Arizona     Colombia        0     0     2     0     0     0     0     0     0
## # ℹ 441 more rows
## # ℹ 7 more variables: Jul <dbl>, Aug <dbl>, Sep <dbl>, Cases <dbl>, Inds <dbl>,
## #   State <dbl>, U.S. <dbl>

Pivot to a tidier form with one row per month:

ref_clean <-
    fill(ref, Destination) |>
    filter(! is.na(Nationality)) |>
    pivot_longer(
        Oct : Sep,
        names_to = "month",
        values_to = "count")
ref_clean
## # A tibble: 5,412 × 8
##    Destination Nationality     Cases  Inds State     U.S. month count
##    <chr>       <chr>           <dbl> <dbl> <dbl>    <dbl> <chr> <dbl>
##  1 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Oct       0
##  2 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Nov       0
##  3 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Dec       3
##  4 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Jan       0
##  5 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Feb       4
##  6 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Mar       0
##  7 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Apr       0
##  8 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 May       0
##  9 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Jun       0
## 10 Alabama     Dem. Rep. Congo     2     7 0.875 0.000811 Jul       0
## # ℹ 5,402 more rows

Top 5 destination states:

group_by(ref_clean, Destination) |>
    summarize(count = sum(count)) |>
    ungroup() |>
    slice_max(count, n = 5)
## # A tibble: 5 × 2
##   Destination count
##   <chr>       <dbl>
## 1 Ohio          750
## 2 Texas         602
## 3 Washington    562
## 4 California    545
## 5 New York      540

Top 5 origin nationalities:

group_by(ref_clean, Nationality) |>
    summarize(count = sum(count)) |>
    ungroup() |>
    slice_max(count, n = 5)
## # A tibble: 5 × 2
##   Nationality     count
##   <chr>           <dbl>
## 1 Dem. Rep. Congo  1968
## 2 Bhutan           1884
## 3 Burma            1240
## 4 Ukraine          1021
## 5 Eritrea           648

Completing Missing Factor Level Combinations

This example is based on a blog post using data from the Significant Earthquake Database.

The data used is a selection for North America and Hawaii.

Reading in the data:

if (! file.exists("earthquakes.tsv"))
    download.file("http://homepage.stat.uiowa.edu/~luke/data/earthquakes.tsv",
                  "earthquakes.tsv")
eq <- read.table("earthquakes.tsv",
                 header = TRUE,
                 sep = "\t")
eq <- as_tibble(eq)
select(eq, STATE, everything())
## # A tibble: 254 × 47
##    STATE   I_D FLAG_TSUNAMI  YEAR MONTH   DAY  HOUR MINUTE SECOND FOCAL_DEPTH
##    <chr> <int> <chr>        <int> <int> <int> <int>  <int>  <dbl>       <int>
##  1 HI     6697 Tsu           1500    NA    NA    NA     NA     NA          NA
##  2 MA     6013 Tsu           1668     4    13    NA     NA     NA          NA
##  3 OR     9954 Tsu           1700     1    27     5      0     NA          NA
##  4 MA     5828 Tsu           1755    11    18     9     11     35          NA
##  5 AK     5926 Tsu           1788     7    21    NA     NA     NA          NA
##  6 AK     5927 Tsu           1788     8     6    NA     NA     NA          NA
##  7 AK     7057 Tsu           1792    NA    NA    NA     NA     NA          NA
##  8 CA     5857 Tsu           1806     3    25     8     NA     NA          NA
##  9 AR     1594 Tsu           1811    12    16     8     15     NA          NA
## 10 AR     7058 Tsu           1811    12    16    14     15     NA          NA
## # ℹ 244 more rows
## # ℹ 37 more variables: EQ_PRIMARY <dbl>, EQ_MAG_MW <dbl>, EQ_MAG_MS <dbl>,
## #   EQ_MAG_MB <dbl>, EQ_MAG_ML <dbl>, EQ_MAG_MFA <dbl>, EQ_MAG_UNK <dbl>,
## #   INTENSITY <int>, COUNTRY <chr>, LOCATION_NAME <chr>, LATITUDE <dbl>,
## #   LONGITUDE <dbl>, REGION_CODE <int>, DEATHS <int>, DEATHS_DESCRIPTION <int>,
## #   MISSING <lgl>, MISSING_DESCRIPTION <lgl>, INJURIES <int>,
## #   INJURIES_DESCRIPTION <int>, DAMAGE_MILLIONS_DOLLARS <dbl>, …

Some STATE entries are blank, but the location name identifies the state:

filter(eq, STATE == "") |>
    select(STATE, LOCATION_NAME)
## # A tibble: 9 × 2
##   STATE LOCATION_NAME                        
##   <chr> <chr>                                
## 1 ""    MONTANA:  CLARKSTON VALLEY           
## 2 ""    ILLINOIS:  WEST SALEM                
## 3 ""    NEVADA:  FALLON                      
## 4 ""    CALIFORNIA:  OCOTILLO                
## 5 ""    COLORADO:  PAONIA                    
## 6 ""    ALASKA: ALEUTIAN ISLANDS: FOX ISLANDS
## 7 ""    OKLAHOMA:  SPARKS                    
## 8 ""    OKLAHOMA:  SPARKS, PRAGUE            
## 9 ""    TEXAS:  WEST

One way to fix this:

badidx <- which(eq$STATE == "")
badstate <- sub(":.*", "",
                eq$LOCATION_NAME[badidx])
eq$STATE[badidx] <-
    state.abb[match(tolower(badstate),
                    tolower(state.name))]

Add full state names as a factor:

states <- data.frame(STATE = state.abb,
                     STATE_NAME = state.name)
eq <- left_join(eq, states, "STATE")
eq <- mutate(eq,
             STATE_NAME = factor(STATE_NAME,
                                 state.name))

Look at total number of earthquakes by state:

tbl <- count(eq, STATE_NAME)
head(tbl, 6)
## # A tibble: 6 × 2
##   STATE_NAME      n
##   <fct>       <int>
## 1 Alabama         1
## 2 Alaska         78
## 3 Arkansas        2
## 4 California    100
## 5 Colorado        2
## 6 Connecticut     1
library(forcats)
pb <- ggplot(tbl, aes(y = fct_rev(STATE_NAME), x = n)) +
    geom_col() +
    thm +
    theme(axis.title.y = element_blank())
ps <- ggplot(tbl, aes(y = fct_rev(STATE_NAME), x = n)) +
    geom_point() +
    geom_segment(aes(xend = 0, yend = fct_rev(STATE_NAME))) +
    thm +
    theme(axis.title.y = element_blank())
pb | ps

Many states, including Iowa, are missing.

Their counts should be zero.

This often impacts visualizations.

complete() can be used to fill in the zero values.

tbl <-
    count(eq, STATE_NAME) |>
    complete(STATE_NAME, fill = list(n = 0))
tbl
## # A tibble: 50 × 2
##    STATE_NAME      n
##    <fct>       <int>
##  1 Alabama         1
##  2 Alaska         78
##  3 Arizona         0
##  4 Arkansas        2
##  5 California    100
##  6 Colorado        2
##  7 Connecticut     1
##  8 Delaware        0
##  9 Florida         0
## 10 Georgia         0
## # ℹ 40 more rows
thm_ytxt_sm <- theme(axis.text.y = element_text(size = 10))
(pb %+% tbl + thm_ytxt_sm) | (ps %+% tbl + thm_ytxt_sm)

Reading

Chapter Data transformation of R for Data Science.

Chapter Tidy data of R for Data Science.

Interactive Tutorial

An interactive learnr tutorial for these notes is available.

You can run the tutorial with

STAT4580::runTutorial("dplyr")

You can install the current version of the STAT4580 package with

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

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

Exercises

  1. To bring in dplyr and the mpg data, start by evaluating

    library(dplyr)
    library(ggplot2)

    The select function allows variables to be specified in a variety of ways. Which of the following does not produce a data frame with only the variables manufacturer, model, cty, hwy?

    1. select(mpg, 1:2, 7:8)
    2. select(mpg, starts_with("m"), ends_with("y"))
    3. select(mpg, 1:2, cty : hwy)
    4. select(mpg, -(displ : drv), -(fl : class))
  2. Consider the code

    library(dplyr)
    library(ggplot2)
    filter(mpg, ---) |> nrow()

    Which of the replacements for --- computes the number of Ford vehicles with more than 6 cylinders in the mpg table?

    1. model == "ford", cyl <= 6
    2. manufacturer == "ford" | cyl > 6
    3. manufacturer == "ford", cyl > 6
    4. manufacturer == "ford", cylinders > 6
  3. In the 2013 NYC flights data provided by the nycflights13 package how many flights were there to Des Moines (FAA code DSM) from NYC in the first three months of 2013?

    1. 13
    2. 98
    3. 64
    4. 77
  4. To bring in dplyr and the mpg data, start by evaluating

    library(dplyr)
    library(ggplot2)

    Which of the following sorts the rows for mpg by increasing cyl value and, within each cyl value sorts the rows from largest to smallest hwy value.

    1. arrange(mpg, desc(hwy), cyl)
    2. arrange(mpg, cyl, desc(hwy))
    3. arrange(mpg, desc(cyl), hwy)
    4. arrange(mpg, cyl, hwy)
  5. To bring in dplyr and the flights data, start by evaluating

    library(dplyr)
    library(nycflights13)

    The dep_time variable in the flights data set from the nycflights13 package is convenient to look at (529 means 5:29 AM and 22:12 means 10:12 PM), but hard to compute with. Which of the following adds variables dep_hour and dep_min containing hour and minute of the departure time?

    1. mutate(flights, dep_hour = dep_time %/% 60, dep_min = dep_time %% 60)
    2. mutate(flights, dep_hour = dep_time %/% 100, dep_min = dep_time %% 100)
    3. mutate(flights, dep_hour = dep_time / 100, dep_min = dep_time - dep_hour)
    4. mutate(flights, dep_hour = dep_time %/% 60, dep_min = dep_hour %% 60)
  6. Using the gapminder data set, the following code computes population totals for each continent and each year:

    library(dplyr)
    library(gapminder)
    cpops <- group_by(gapminder, continent, year) |>
        summarize(pop = sum(pop)) |>
        ungroup()

    To produce a plot to compare population growth over the years for the continents it is useful to standardize the population data for each continent, for example by dividing the population values by the average population size for each continent. One way to do this is with a grouped mutate. The first line of your result should be

    head(cpops_std, 1)
    ## # A tibble: 1 × 4
    ##   continent  year       pop stdpop
    ##   <fct>     <int>     <dbl>  <dbl>
    ## 1 Africa     1952 237640501  0.461
    library(ggplot2)
    ggplot(cpops_std, aes(x = year, y = stdpop, color = continent)) +
        geom_line()

    Which of the following produces the correct result:

    1. cpops_std <- group_by(cpops, continent) |> mutate(stdpop = pop / mean(year)) |> ungroup()
    2. cpops_std <- group_by(cpops, year) |> mutate(stdpop = pop / mean(pop)) |> ungroup()
    3. cpops_std <- group_by(cpops, continent) |> mutate(stdpop = pop / mean(pop)) |> ungroup()
    4. cpops_std <- group_by(cpops, year) |> mutate(stdpop = pop / mean(year)) |> ungroup()
  7. Another approach to the previous exercise first creates a table of population averages with

    cpops_avg <- group_by(cpops, continent) |>
        summarize(avg_pop = mean(pop))

    Then use a left join to add avg_pop to the cpops table, followed by an ordinary mutate step:

    left_join(---) |>
        mutate(stdpop = pop / avg_pop)

    Which is the correct replacement for ---?

    1. cpops_avg, cpops, "continent"
    2. cpops, cpops_avg, "continent"
    3. cpops, cpops_avg, "year"
    4. cpops_avg, cpops, "year"
LS0tCnRpdGxlOiAiQSBHcmFtbWFyIG9mIERhdGEgTWFuaXB1bGF0aW9uIgpvdXRwdXQ6CiAgaHRtbF9kb2N1bWVudDoKICAgIHRvYzogeWVzCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCi0tLQoKPGxpbmsgcmVsPSJzdHlsZXNoZWV0IiBocmVmPSJzdGF0NDU4MC5jc3MiIHR5cGU9InRleHQvY3NzIiAvPgo8c3R5bGUgdHlwZT0idGV4dC9jc3MiPiAucmVtYXJrLWNvZGUgeyBmb250LXNpemU6IDg1JTsgfSA8L3N0eWxlPgoKYGBge3Igc2V0dXAsIGluY2x1ZGUgPSBGQUxTRSwgbWVzc2FnZSA9IEZBTFNFfQpzb3VyY2UoaGVyZTo6aGVyZSgic2V0dXAuUiIpKQprbml0cjo6b3B0c19jaHVuayRzZXQoY29sbGFwc2UgPSBUUlVFLCBtZXNzYWdlID0gRkFMU0UsCiAgICAgICAgICAgICAgICAgICAgICBmaWcuaGVpZ2h0ID0gNSwgZmlnLndpZHRoID0gNiwgZmlnLmFsaWduID0gImNlbnRlciIpCgpzZXQuc2VlZCgxMjM0NSkKbGlicmFyeShkcGx5cikKbGlicmFyeShnZ3Bsb3QyKQpsaWJyYXJ5KGxhdHRpY2UpCmBgYAoKCiMjIEJhY2tncm91bmQKClRoZSBgZHBseXJgIHBhY2thZ2UgcHJvdmlkZXMgYSBsYW5ndWFnZSwgb3IgZ3JhbW1hciwgZm9yIGRhdGEKbWFuaXB1bGF0aW9uLgoKVGhlIGxhbmd1YWdlIGNvbnRhaW5zIGEgbnVtYmVyIG9mIF92ZXJic18gdGhhdCBvcGVyYXRlIG9uIHRhYmxlcyBpbgpkYXRhIGZyYW1lcy4KClRoZSBtb3N0IGNvbW1vbmx5IHVzZWQgdmVyYnMgb3BlcmF0ZSBvbiBhIHNpbmdsZSBkYXRhIGZyYW1lOgoKKiBbYHNlbGVjdCgpYF0oaHR0cHM6Ly9kcGx5ci50aWR5dmVyc2Uub3JnL3JlZmVyZW5jZS9zZWxlY3QuaHRtbCk6CiAgcGljayB2YXJpYWJsZXMgYnkgdGhlaXIgbmFtZXMKCiogW2BmaWx0ZXIoKWBdKGh0dHBzOi8vZHBseXIudGlkeXZlcnNlLm9yZy9yZWZlcmVuY2UvZmlsdGVyLmh0bWwpOgogIGNob29zZSByb3dzIHRoYXQgc2F0aXNmeSBzb21lIGNyaXRlcmlhCgoqIFtgbXV0YXRlKClgXShodHRwczovL2RwbHlyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL211dGF0ZS5odG1sKToKICBjcmVhdGUgdHJhbnNmb3JtZWQgb3IgZGVyaXZlZCB2YXJpYWJsZXMKCiogW2BhcnJhbmdlKClgXShodHRwczovL2RwbHlyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL2FycmFuZ2UuaHRtbCk6CiAgcmVvcmRlciB0aGUgcm93cwoKKiBbYHN1bW1hcml6ZSgpYF0oaHR0cHM6Ly9kcGx5ci50aWR5dmVyc2Uub3JnL3JlZmVyZW5jZS9zdW1tYXJpc2UuaHRtbCk6CiAgY29sbGFwc2Ugcm93cyBkb3duIHRvIHN1bW1hcmllcwoKVGhlcmUgYXJlIGFsc28gYSBudW1iZXIgb2YgYGpvaW5gIHZlcmJzIHRoYXQgbWVyZ2Ugc2V2ZXJhbCBkYXRhIGZyYW1lcwppbnRvIG9uZS4KClRoZSBgdGlkeXJgIHBhY2thZ2UgcHJvdmlkZXMgYWRkaXRpb25hbCB2ZXJicywgc3VjaCBhcyBgcGl2b3RfbG9uZ2VyYAphbmQgYHBpdm90X3dpZGVyYCBmb3IgcmVzaGFwaW5nIGRhdGEgZnJhbWVzLgoKVGhlIHNpbmdsZSB0YWJsZSB2ZXJicyBjYW4gYWxzbyBiZSB1c2VkIHdpdGgKW2Bncm91cF9ieSgpYF0oaHR0cHM6Ly9kcGx5ci50aWR5dmVyc2Uub3JnL3JlZmVyZW5jZS9ncm91cF9ieS5odG1sKSBhbmQKW2B1bmdyb3VwKClgXShodHRwczovL2RwbHlyLnRpZHl2ZXJzZS5vcmcvcmVmZXJlbmNlL2dyb3VwX2J5Lmh0bWwpIHRvCndvcmsgYSBncm91cCBhdCBhIHRpbWUgaW5zdGVhZCBvZiBhcHBseWluZyB0byB0aGUgZW50aXJlIGRhdGEgZnJhbWUuCgpBbiBleHBlcmltZW50YWwgYWx0ZXJuYXRpdmUgdG8gdXNpbmcgYGdyb3VwX2J5KClgIGFuZCBgdW5ncm91cCgpYCBpcwp0byB1c2UgdGhlIGAuYnlgIGFyZ3VtZW50IHdpdGggdGhlc3cgdmVyYnMuCgpUaGUgZGVzaWduIG9mIGBkcGx5cmAgaXMgc3Ryb25nbHkgbW90aXZhdGVkIGJ5IFNRTC4KCiogYGRwbHlyYCBjYW4gYWxzbyBiZSB1c2VkIHRvIG9wZXJhdGUgb24KICBbdGFibGVzIHN0b3JlZCBpbiBkYXRhIGJhc2VzLl0oaHR0cHM6Ly9kYnBseXIudGlkeXZlcnNlLm9yZy9pbmRleC5odG1sKQoKVGhlIGJhc2ljIHRyYW5zZm9ybWF0aW9uIHZlcmJzIGFyZSBkZXNjcmliZWQgaW4gdGhlCltEYXRhIFRyYW5zZm9ybWF0aW9uIGNoYXB0ZXJdKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovdHJhbnNmb3JtLmh0bWwpIG9mCltSIGZvciBEYXRhIFNjaWVuY2VdKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovKS4KClV0aWxpdGllcyBpbiBgdGlkeXJgIGFyZSBkZXNjcmliZWQgaW4gdGhlIFtUaWR5IERhdGEKY2hhcHRlcl0oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei9kYXRhLXRpZHkuaHRtbCkuCgpTb21lIGRhdGEgc2V0cyBmb3IgaWxsdXN0cmF0aW9uOgoKKiBFUEEgdmVoaWNsZSBkYXRhIHVzZWQgaW4gSFc0CgogIGBgYHtyLCBtZXNzYWdlID0gRkFMU0V9CmxpYnJhcnkocmVhZHIpCmlmICghIGZpbGUuZXhpc3RzKCJ2ZWhpY2xlcy5jc3YuemlwIikpCiAgICBkb3dubG9hZC5maWxlKCJodHRwOi8vd3d3LnN0YXQudWlvd2EuZWR1L35sdWtlL2RhdGEvdmVoaWNsZXMuY3N2LnppcCIsCiAgICAgICAgICAgICAgICAgICJ2ZWhpY2xlcy5jc3YuemlwIikKbmV3bXBnIDwtIHJlYWRfY3N2KCJ2ZWhpY2xlcy5jc3YuemlwIiwgZ3Vlc3NfbWF4ID0gMTAwMDAwKQogIGBgYAoKKiBUaGUgYG55Y2ZsaWdodHMxM2AgcGFja2FnZSBwcm92aWRlcyBkYXRhIG9uIGFsbCBmbGlnaHRzCiAgb3JpZ2luYXRpbmcgZnJvbSBvbmUgb2YgdGhlIHRocmVlIG1haW4gTmV3IFlvcmsgQ2l0eSBhaXJwb3J0cyBpbgogIDIwMTMgYW5kIGhlYWRpbmcgdG8gYWlycG9ydHMgd2l0aGluIHRoZSBVUy4KCiAgYGBge3J9CmxpYnJhcnkobnljZmxpZ2h0czEzKQogIGBgYAoKKiBUaGUgW2BhbnlmbGlnaHRzYF0oaHR0cHM6Ly9jcmFuLnItcHJvamVjdC5vcmcvcGFja2FnZT1hbnlmbGlnaHRzKQogIHBhY2thZ2UgY2FuIGJlIHVzZWQgdG8gb2J0YWluIGRhdGEgZm9yIG90aGVyIGNpdGllcyBhbmQgcGVyaW9kcy4KCiogVGhlIGBzdG9ybXNgIGRhdGEgZnJhbWUsIGluY2x1ZGVkIGluIGBkcGx5cmAgd2l0aCBkYXRhIG9uCiAgaHVycmljYW5lcyBiZXR3ZWVuIDE5NzUgYW5kIDIwMTUuCgoKIyMgU2VsZWN0aW5nIFZhcmlhYmxlcwoKRGF0YSBzZXRzIG9mdGVuIGNvbnRhaW4gaHVuZHJlZHMgb2YgZXZlbiB0aG91c2FuZHMgb2YgdmFyaWFibGVzLgoKQSB1c2VmdWwgZmlyc3Qgc3RlcCBpcyB0byBzZWxlY3QgYSBncm91cCBvZiBpbnRlcmVzdGluZyB2YXJpYWJsZXMuCgpBIHJlYXNvbmFibGUgc2VsZWN0aW9uIG9mIHRoZSBbRVBBIHZlaGljbGUKdmFyaWFibGVzXShodHRwczovL3d3dy5mdWVsZWNvbm9teS5nb3YvZmVnL3dzL2luZGV4LnNodG1sI3ZlaGljbGUpOgoKYGBge3J9Cm5ld21wZzEgPC0gc2VsZWN0KG5ld21wZywKICAgICAgICAgICAgICAgICAgbWFrZSwgbW9kZWwsIHllYXIsCiAgICAgICAgICAgICAgICAgIGN0eSA9IGNpdHkwOCwgaHd5ID0gaGlnaHdheTA4LAogICAgICAgICAgICAgICAgICB0cmFucyA9IHRyYW55LCBjeWwgPSBjeWxpbmRlcnMsCiAgICAgICAgICAgICAgICAgIGZ1ZWwgPSBmdWVsVHlwZTEsIGRpc3BsKQpoZWFkKG5ld21wZzEsIDIpCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+ClZhcmlhYmxlcyBjYW4gYmUgZ2l2ZW4gbmV3IG5hbWVzIGluIGEgYHNlbGVjdGAgY2FsbDsgeW91IGNhbiBhbHNvCnJlbmFtZSB0aGVtIGxhdGVyIHdpdGggYHJlbmFtZWAuCjwvZGl2PgoKU29tZSB2YXJpYXRpb25zICh0aGUgZG9jdW1lbnRhdGlvbiBmb3IKW2BzZWxlY3QoKWBdKGh0dHBzOi8vZHBseXIudGlkeXZlcnNlLm9yZy9yZWZlcmVuY2Uvc2VsZWN0Lmh0bWwpIGdpdmVzCmZ1bGwgZGV0YWlscyk6CgpgYGB7cn0Kc2VsZWN0KG5ld21wZzEsIHllYXIgOiB0cmFucykgfD4gaGVhZCgyKQpgYGAKCmBgYHtyfQpzZWxlY3QobmV3bXBnMSwgLXllYXIsIC10cmFucykgfD4gaGVhZCgyKQpgYGAKCmBgYHtyfQpzZWxlY3QobmV3bXBnMSwgLSAoeWVhciA6IHRyYW5zKSkgfD4gaGVhZCgyKQpgYGAKCk51bWVyaWNhbCBjb2x1bW4gc3BlY2lmaWNhdGlvbnMgY2FuIGFsc28gYmUgdXNlZDoKCmBgYHtyfQpzZWxlY3QobmV3bXBnMSwgMSwgMykgfD4gaGVhZCgyKQpgYGAKCmBgYHtyfQpzZWxlY3QobmV3bXBnMSwgMSA6IDMpIHw+IGhlYWQoMikKYGBgCgpgYGB7cn0Kc2VsZWN0KG5ld21wZzEsIC0gKDEgOiAzKSkgfD4gaGVhZCgyKQpgYGAKClNvbWUgX2hlbHBlciBmdW5jdGlvbnNfIGNhbiBhbHNvIGJlIHVzZWQgaW4gdGhlIGNvbHVtbiBzcGVjaWZpY2F0aW9ucy4KClRoZSBtb3N0IHVzZWZ1bCBvbmVzIGFyZSBgc3RhcnRzX3dpdGhgLCBgZW5kc193aXRoYCwgYW5kIGBjb250YWluc2AuCgpgYGB7cn0Kc2VsZWN0KG5ld21wZywgc3RhcnRzX3dpdGgoImZ1ZWwiKSkgfD4gaGVhZCgyKQpgYGAKCmBgYHtyfQpzZWxlY3QobmV3bXBnLCBjb250YWlucygiY2l0eSIpKSB8PiBoZWFkKDIpCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CkJ5IGRlZmF1bHQgdGhlc2UgaGVscGVycyBpZ25vcmUgY2FzZS4KPC9kaXY+CgpUaGVzZSBjYW4gYWxzbyBiZSBwcmVjZWRlZCBieSBhIG1pbnVzIG1vZGlmaWVyIHRvIG9taXQgY29sdW1uczoKCmBgYHtyfQpzZWxlY3QobmV3bXBnMSwgLWNvbnRhaW5zKCJtIikpIHw+IGhlYWQoMikKYGBgCgpgY29udGFpbnNgIHJlcXVpcmVzIGEgbGl0ZXJhbCBzdHJpbmc7IGBtYXRjaGVzYCBpcyBhbmFsb2dvdXMgYnV0IGl0cwphcmd1bWVudCBpcyBpbnRlcnByZXRlZCBhcyBhIFtfcmVndWxhciBleHByZXNzaW9uX10ocmVnZXguaHRtbCkKcGF0dGVybiB0byBiZSBtYXRjaGVkLgoKYGBge3J9CnNlbGVjdChuZXdtcGcsIG1hdGNoZXMoIl5bRmZddWVsIikpIHw+IGhlYWQoMikKYGBgCgpTb21ldGltZXMgaXQgaXMgdXNlZnVsIHRvIG1vdmUgYSBmZXcgY29sdW1ucyB0byB0aGUgZnJvbnQ7IHRoZSBgZXZlcnl0aGluZ2AKaGVscGVyIGNhbiBiZSB1c2VkIGZvciB0aGF0LgoKVGhlIG9yaWdpbmFsIE5ZQyBgZmxpZ2h0c2AgdGFibGU6CgpgYGB7cn0KaGVhZChmbGlnaHRzLCAyKQpgYGAKCk1vdmluZyBgYWlyX3RpbWVgIHRvIHRoZSBmcm9udDoKCmBgYHtyfQpzZWxlY3QoZmxpZ2h0cywgYWlyX3RpbWUsIGV2ZXJ5dGhpbmcoKSkgfD4gaGVhZCgyKQpgYGAKClRoZSBgd2hlcmVgIGhlbHBlciBmdW5jdGlvbiBhbGxvd3MgY29sdW1ucyB0byBiZSBzZWxlY3RlZCBiYXNlZCBvbiBhCnByZWRpY2F0ZSBhcHBsaWVkIHRvIHRoZSBmdWxsIGNvbHVtbjoKCmBgYHtyfQpzZWxlY3QobmV3bXBnMSwgd2hlcmUoYW55TkEpKSB8PiBoZWFkKDIpCmBgYAoKYGBge3J9CnNlbGVjdChmbGlnaHRzLCB3aGVyZShhbnlOQSkpIHw+IGhlYWQoMikKYGBgCgo8ZGl2IGNsYXNzID0gImFsZXJ0Ij4KSWYgdGhlcmUgYXJlIG1pc3NpbmcgdmFsdWVzIGluIHlvdXIgZGF0YSBpdCBpcyB1c3VhbGx5IGEgZ29vZCBpZGVhIHRvCnJldmlldyB0aGVtIHRvIG1ha2Ugc3VyZSB5b3UgdW5kZXJzdGFuZCB3aGF0IHRoZXkgbWVhbi4KPC9kaXY+CgpDb2x1bW5zIGNhbiBhbHNvIGJlIHNlbGVjdGVkIHVzaW5nIGxpc3Qgb3BlcmF0aW9uczoKCmBgYHtyfQp2YXJzIDwtIGMoIm1ha2UiLCAibW9kZWwiLCAieWVhciIsICJjaXR5MDgiLCAidHJhbnkiLCAiY3lsaW5kZXJzIiwKICAgICAgICAgICJmdWVsVHlwZTEiLCAiZGlzcGwiKQpuZXdtcGdbdmFyc10gfD4gaGVhZCgyKQpgYGAKCgojIyBGaWx0ZXJpbmcgUm93cwoKVGhlIGBmaWx0ZXJgIGZ1bmN0aW9uIHBpY2tzIG91dCB0aGUgc3Vic2V0IG9mIHJvd3MgdGhhdCBzYXRpc2Z5IG9uZSBvcgptb3JlIGNvbmRpdGlvbnMuCgpBbGwgY2FycyB3aXRoIGNpdHkgTVBHIGF0IG9yIGFib3ZlIDE0MCBhbmQgbW9kZWwgeWVhciAyMDIzOgoKYGBge3J9CmZpbHRlcihuZXdtcGcxLCBjdHkgPj0gMTQwLCB5ZWFyID09IDIwMjMpCmBgYAoKQWxsIGZsaWdodHMgZnJvbSBOZXcgWW9yayB0byBEZXMgTW9pbmVzIG9uIEphbnVhcnkgMSwgMjAxMzoKCmBgYHtyfQpmaWx0ZXIoZmxpZ2h0cywgZGF5ID09IDEsIG1vbnRoID09IDEsIGRlc3QgPT0gIkRTTSIpIHw+CiAgICBzZWxlY3QoeWVhciA6IGRlcF90aW1lLCBvcmlnaW4pCmBgYAoKCiMjIyBDb21wYXJpc29ucwoKVGhlIGJhc2ljIGNvbXBhcmlzb24gb3BlcmF0b3JzIGFyZQoKKiBgPT1gLCBgIT1gCiogYDxgLCBgPD1gCiogYD5gLCBgPj1gCgo8ZGl2IGNsYXNzID0gImFsZXJ0Ij4KQmUgc3VyZSB0byB1c2UgYD09YCwgbm90IGEgc2luZ2xlIGA9YCBjaGFyYWN0ZXIuCjwvZGl2PgoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CkJlIGNhcmVmdWwgYWJvdXQgW2Zsb2F0aW5nCnBvaW50XShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9GbG9hdGluZy1wb2ludF9hcml0aG1ldGljKQplcXVhbGl0eSB0ZXN0czoKCmBgYHtyfQp4IDwtIDEgLSAwLjkKeCA9PSAwLjEKbmVhcih4LCAwLjEpCmBgYAo8L2Rpdj4KCjxkaXYgY2xhc3MgPSAiYWxlcnQiPgpFcXVhbGl0eSBjb21wYXJpc29ucyBjYW4gYmUgdXNlZCBvbiBjaGFyYWN0ZXIgdmVjdG9ycyBhbmQgZmFjdG9ycy4KCk9yZGVyIGNvbXBhcmlzb25zIGNhbiBiZSB1c2VkIG9uIGNoYXJhY3RlciB2ZWN0b3JzLCBidXQgbWF5IHByb2R1Y2UKc3VycHJpc2luZyByZXN1bHRzLCBvciByZXN1bHRzIHRoYXQgdmFyeSB3aXRoIGRpZmZlcmVudCBsb2NhbGUKc2V0dGluZ3MuCjwvZGl2PgoKYCVpbiVgIGNhbiBiZSB1c2VkIHRvIG1hdGNoIGFnYWluc3Qgb25lIG9mIHNldmVyYWwgb3B0aW9uczoKCmBgYHtyfQpmaWx0ZXIoZmxpZ2h0cywgZGF5ICVpbiUgMSA6IDIsIG1vbnRoID09IDEsIGRlc3QgPT0gIkRTTSIpIHw+CiAgICBzZWxlY3QoeWVhciA6IGRlcF90aW1lLCBvcmlnaW4pCmBgYAoKCiMjIyBMb2dpY2FsIE9wZXJhdGlvbnMKClRoZSBiYXNpYyBsb2dpY2FsIG9wZXJhdGlvbnMgb24gdmVjdG9ycyBhcmUKCiogYCZgIC0tIGFuZAoqIGB8YCAtLSBvcgoqIGAhYCAtLSBub3QKKiBgeG9yYCAtLSBleGNsdXNpdmUgb3IKClRydXRoIHRhYmxlczoKCjwhLS0gIyBub2xpbnQgc3RhcnQgLS0+CgpgYGB7cn0KeCA8LSBjKFRSVUUsIFRSVUUsICBGQUxTRSwgRkFMU0UpCnkgPC0gYyhUUlVFLCBGQUxTRSwgVFJVRSwgIEZBTFNFKQohIHgKeCAmIHkKeCB8IHkKeG9yKHgsIHkpCmBgYAo8IS0tICMgbm9saW50IGVuZCAtLT4KCjxkaXYgY2xhc3MgPSAiYWxlcnQiPgpNYWtlIHN1cmUgbm90IHRvIGNvbmZ1c2UgdGhlIHZlY3Rvcml6ZWQgbG9naWNhbCBvcGVyYXRvcnMgYCZgIGFuZCBgfGAKd2l0aCB0aGUgc2NhbGFyIGZsb3cgY29udHJvbCBvcGVyYXRvcnMgYCYmYCBhbmQgYHx8YC4KPC9kaXY+CgpUaGUgcHJldmlvdXMgYGZsaWdodHNgIGV4YW1wbGUgY2FuIGFsc28gYmUgd3JpdHRlbiBhcwoKYGBge3J9CmZpbHRlcihmbGlnaHRzLCBkYXkgPT0gMSB8IGRheSA9PSAyLCBtb250aCA9PSAxLCBkZXN0ID09ICJEU00iKSB8PgogICAgc2VsZWN0KHllYXIgOiBkZXBfdGltZSwgb3JpZ2luKQpgYGAKCkl0IGNhbiBhbHNvIGJlIHdyaXR0ZW4gYXMKCmBgYHtyfQpmaWx0ZXIoZmxpZ2h0cywgKGRheSA9PSAxIHwgZGF5ID09IDIpICYgbW9udGggPT0gMSAmIGRlc3QgPT0gIkRTTSIpIHw+CiAgICBzZWxlY3QoeWVhciA6IGRlcF90aW1lLCBvcmlnaW4pCmBgYAoKCiMjIyBNaXNzaW5nIFZhbHVlcwoKYGZpbHRlcmAgb25seSBrZWVwcyB2YWx1ZXMgY29ycmVzcG9uZGluZyB0byBgVFJVRWAgcHJlZGljYXRlIHZhbHVlczsKYEZBTFNFYCBhbmQgYE5BYCB2YWx1ZXMgYXJlIGRyb3BwZWQuCgpBcml0aG1ldGljIGFuZCBjb21wYXJpc29uIG9wZXJhdG9ycyB3aWxsIGFsd2F5cyBwcm9kdWNlIGBOQWAgdmFsdWVzCmlmIG9uZSBvcGVyYW5kIGlzIGBOQWAuCgo8IS0tICMgbm9saW50IHN0YXJ0IC0tPgpgYGB7cn0KMSArIE5BCk5BIDwgMQpOQSA9PSBOQQpgYGAKPCEtLSAjIG5vbGludCBlbmQgLS0+CgpJbiBhIGZldyBjYXNlcyBhIG5vbi1gTkFgIHJlc3VsdCBjYW4gYmUgZGV0ZXJtaW5lZDoKCmBgYHtyfQpUUlVFIHwgTkEKTkEgJiBGQUxTRQpOQSBeIDAKYGBgCgpgaXMubmFgIGNhbiBiZSB1c2VkIHRvIGFsc28gc2VsZWN0IHJvd3Mgd2l0aCBgTkFgIHZhbHVlcy4KCk5vbi1lbGVjdHJpYyB2ZWhpY2xlcyB3aXRoIHplcm8gb3IgYE5BYCBmb3IgYGRpc3BsYDoKCmBgYHtyfQpmaWx0ZXIobmV3bXBnMSwgZGlzcGwgPT0gMCB8IGlzLm5hKGRpc3BsKSwgZnVlbCAhPSAiRWxlY3RyaWNpdHkiKQpgYGAKCkZsaWdodHMgd2l0aCBgTkFgIGZvciBib3RoIGBkZXBfdGltZWAgYW5kIGBhcnJfdGltZWA6CgpgYGB7cn0KZmlsdGVyKGZsaWdodHMsIGlzLm5hKGRlcF90aW1lKSAmIGlzLm5hKGFycl90aW1lKSkgfD4KICAgIGhlYWQoMikgfD4KICAgIHNlbGVjdCh5ZWFyIDogYXJyX3RpbWUpCmBgYAoKQW4gYWx0ZXJuYXRpdmUgYXBwcm9hY2ggdGhhdCBkb2VzIG5vdCB1c2UgYGZpbHRlcmAgd291bGQgYmUKCmBgYHtyfQppZHggPC0gd2l0aChmbGlnaHRzLCBpcy5uYShkZXBfdGltZSkgJiBpcy5uYShhcnJfdGltZSkpCmZsaWdodHNbaWR4LCBdIHw+CiAgICBoZWFkKDIpIHw+CiAgICBzZWxlY3QoeWVhciA6IGFycl90aW1lKQpgYGAKCgojIyBFeHBsb3JpbmcgdGhlIEVQQSBEYXRhCgpTb3J0aW5nIG91dCBmdWVsIHR5cGVzOgoKYGBge3J9CnNlbGVjdChuZXdtcGcsIGNvbnRhaW5zKCJmdWVsVHlwZSIpKSB8PiB1bmlxdWUoKQpgYGAKClZhcmlhYmxlcyB3aXRoIG1pc3NpbmcgdmFsdWVzOgoKYGBge3J9CnNlbGVjdChuZXdtcGcsIHdoZXJlKGFueU5BKSkgfD4gaGVhZCgyKQpgYGAKCkFtb25nIHRoZSByZWR1Y2VkIGRhdGEgc2V0OgoKYGBge3J9CnNlbGVjdChuZXdtcGcxLCB3aGVyZShhbnlOQSkpIHw+IG5hbWVzKCkKYGBgCgpVbmlxdWUgbWlzc2luZyB2YWx1ZSBwYXR0ZXJuczoKCmBgYHtyLCBldmFsID0gRkFMU0UsIGVjaG8gPSBGQUxTRX0KIyMgVGhpcyBtYXkgYmUgbW9yZSByb2J1c3Q6CmluY29tcGxldGVfY2FzZXMgPC0gZnVuY3Rpb24oZGF0YSkgZGF0YVtSZWR1Y2UoYHxgLCBsYXBwbHkoZGF0YSwgaXMubmEpKSwgXQpgYGAKYGBge3J9CmluY29tcGxldGVfY2FzZXMgPC0gZnVuY3Rpb24oZGF0YSkgZGF0YVshIGNvbXBsZXRlLmNhc2VzKGRhdGEpLCBdCgpzZWxlY3QobmV3bXBnMSwgdHJhbnMsIGN5bCwgZGlzcGwsIGZ1ZWwpIHw+CiAgICBpbmNvbXBsZXRlX2Nhc2VzKCkgfD4KICAgIHVuaXF1ZSgpCmBgYAoKQW4gYWx0ZXJuYXRpdmUgdG8gZGVmaW5pbmcgYGluY29tcGxldGVfY2FzZXMoKWA6CgpgYGB7cn0Kc2VsZWN0KG5ld21wZzEsIHRyYW5zLCBjeWwsIGRpc3BsLCBmdWVsKSB8PgogICAgZmlsdGVyKGlmX2FueShldmVyeXRoaW5nKCksIGlzLm5hKSkgfD4KICAgIHVuaXF1ZSgpCmBgYAoKTG9va2luZyBhdCB0aGUgY291bnRzOgoKYGBge3J9CnNlbGVjdChuZXdtcGcxLCB0cmFucywgY3lsLCBkaXNwbCwgZnVlbCkgfD4KICAgIGluY29tcGxldGVfY2FzZXMoKSB8PgogICAgY291bnQodHJhbnMsIGN5bCwgZGlzcGwsIGZ1ZWwpCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CmBjb3VudCh0cmFucywgY3lsLCBkaXNwbCwgZnVlbClgIGlzIGVzc2VudGlhbGx5IGFuIGFiYnJldmlhdGlvbiBvZgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ3JvdXBfYnkodHJhbnMsIGN5bCwgZGlzcGwsIGZ1ZWwpIHw+CiAgICBzdW1tYXJpemUobiA9IG4oKSkgfD4KICAgIHVuZ3JvdXAoKQpgYGAKPC9kaXY+CgpBbm90aGVyIGFwcHJvYWNoIHRoYXQgYXZvaWRzIHNwZWNpZnlpbmcgdGhlIHZhcmlhYmxlcyB3aXRoIG1pc3NpbmcgdmFsdWVzOgoKYGBge3J9CmluY29tcGxldGVfY2FzZXMobmV3bXBnMSkgfD4KICAgIGNvdW50KGFjcm9zcyh3aGVyZShhbnlOQSkpLCBmdWVsKQpgYGAKCkN5bGluZGVycyBmb3IgZWxlY3RyaWMgdmVoaWNsZXM6CgpgYGB7cn0KZmlsdGVyKG5ld21wZzEsIGZ1ZWwgPT0gIkVsZWN0cmljaXR5IikgfD4KICAgIGNvdW50KGlzLm5hKGN5bCkpCmBgYAoKRGlzcGxhY2VtZW50IGZvciBlbGVjdHJpYyB2ZWhpY2xlczoKCmBgYHtyfQpmaWx0ZXIobmV3bXBnMSwgZnVlbCA9PSAiRWxlY3RyaWNpdHkiKSB8PgogICAgY291bnQoaXMubmEoZGlzcGwpKQpgYGAKCkVsZWN0cmljIHZlaGljbGVzIHdpdGggbm9uLW1pc3NpbmcgZGlzcGxhY2VtZW50OgoKYGBge3J9CmZpbHRlcihuZXdtcGcxLCBmdWVsID09ICJFbGVjdHJpY2l0eSIsICEgaXMubmEoZGlzcGwpKQpgYGAKClNvbWUgdGhpbmdzIHRvIHRoaW5rIGFib3V0IGZvciBlbGVjdHJpYyB2ZWhpY2xlczoKCiogU2hvdWxkIGVsZWN0cmljIHZlaGljbGVzIGJlIGRyb3BwZWQgZnJvbSBhbmFseXNlcyBvciB2aXN1YWxpemF0aW9ucwogIGludm9sdmluZyBjeWxpbmRlcnMgb3IgZGlzcGxhY2VtZW50PwoKKiBTaG91bGQgYGN5bGAgb3IgYGRpc3BsYCBiZSByZWNvZGVkLCBwZXJoYXBzIGFzIHplcm8/CgoqIFdoeSBhcmUgbWlsZXMgcGVyIGdhbGxvbiBfbm90XyBtaXNzaW5nLCBvciBpbmZpbml0ZSwgZm9yIGVsZWN0cmljCiAgdmVoaWNsZXM/IFRoZSBbZG9jdW1lbnRhdGlvbiBmb3IgdGhlCiAgZGF0YV0oaHR0cHM6Ly93d3cuZnVlbGVjb25vbXkuZ292L2ZlZy93cy9pbmRleC5zaHRtbCN2ZWhpY2xlKSBoYXMKICBzb21lIHBvaW50ZXJzLgoKT3RoZXIgY29uc2lkZXJhdGlvbnM6CgoqIFNob3VsZCBzb21lIHRyYW5zbWlzc2lvbiBsZXZlbHMgYmUgY29tYmluZWQ/CgoqIFNob3VsZCBzb21lIGZ1ZWwgdHlwZSBsZXZlbHMgYmUgY29tYmluZWQ/CgoKIyMgQWRkaW5nIE5ldyBWYXJpYWJsZXMKCmBtdXRhdGUoKWAgY2FuIGJlIHVzZWQgdG8gZGVmaW5lIG5ldyB2YXJpYWJsZXMgb3IgbW9kaWZ5IGV4aXN0aW5nIG9uZXMuCgpOZXcgdmFyaWFibGVzIGFyZSBhZGRlZCBhdCB0aGUgZW5kLgoKTGF0ZXIgdmFyaWFibGVzIGNhbiBiZSBkZWZpbmVkIGluIHRlcm1zIG9mIGVhcmxpZXIgb25lcy4KCjxkaXYgY2xhc3MgPSAiYWxlcnQiPgpgbXV0YXRlKClgIGRvZXMgbm90IG1vZGlmeSB0aGUgb3JpZ2luYWwgZGF0YSBmcmFtZTsgaXQgY3JlYXRlcyBhIG5ldyBvbmUKd2l0aCB0aGUgc3BlY2lmaWVkIG1vZGlmaWNhdGlvbnMuCjwvZGl2PgoKYGBge3J9CmZsIDwtIHNlbGVjdChmbGlnaHRzLCB5ZWFyLCBtb250aCwgZGF5LCBhaXJfdGltZSwgZW5kc193aXRoKCJkZWxheSIpKQptdXRhdGUoZmwsCiAgICAgICBnYWluID0gZGVwX2RlbGF5IC0gYXJyX2RlbGF5LAogICAgICAgYWlyX2hvdXJzID0gYWlyX3RpbWUgLyA2MCwKICAgICAgIGdhaW5fcGVyX2hvdXIgPSBnYWluIC8gYWlyX2hvdXJzKSB8PgogICAgaGVhZCgyKQpgYGAKCmB0cmFuc211dGUoKWAga2VlcHMgb25seSB0aGUgbmV3IHZhcmlhYmxlcy4KCmBgYHtyfQp0cmFuc211dGUoZmxpZ2h0cywKICAgICAgICAgIGdhaW4gPSBkZXBfZGVsYXkgLSBhcnJfZGVsYXksCiAgICAgICAgICBhaXJfaG91cnMgPSBhaXJfdGltZSAvIDYwLAogICAgICAgICAgZ2Fpbl9wZXJfaG91ciA9IGdhaW4gLyBhaXJfaG91cnMpIHw+CiAgICBoZWFkKDIpCmBgYAoKVXNlZnVsIG9wZXJhdG9ycyBpbmNsdWRlOgoKKiBCYXNpYyBhcml0aG1ldGljIG9wZXJhdG9ycywgbG9nYXJpdGhtcywgZXRjLgoKKiBNb2R1bGFyIGFyaXRobWV0aWMgb3BlcmF0b3JzIGAlLyVgIGFuZCBgJSVgCgoqIExvZ2ljYWwgb3BlcmF0b3JzIGFuZCBjb21wYXJpc29uIG9wZXJhdG9ycy4KCiogT2Zmc2V0IG9wZXJhdG9ycyBgbGVhZCgpYCBhbmQgYGxhZygpYC4KCiogY3VtdWxhdGl2ZSBhZ2dyZWdhdGVzIHN1Y2ggYXMgYGN1bXN1bSgpYCwgYGN1bW1lYW4oKWAuCgoqIFJhbmtpbmcgb3BlcmF0b3JzLCBzdWNoIGFzIGBtaW5fcmFuaygpYCwgYHBlcmNlbnRfcmFuaygpYC4KCiogUHJlZGljYXRlcyBsaWtlIGBpcy5uYWAuCgpBIHJlYXNvbmFibGUgZ3Vlc3MgaXMgdGhhdCBmbGlnaHRzIHdpdGggYE5BYCB2YWx1ZXMgZm9yIGJvdGgKYGRlcF90aW1lYCBhbmQgYGFycl90aW1lYCB3ZXJlIGNhbmNlbGxlZDoKCmBgYHtyfQpmbCA8LSBtdXRhdGUoZmxpZ2h0cywKICAgICAgICAgICAgIGNhbmNlbGxlZCA9IGlzLm5hKGRlcF90aW1lKSAmIGlzLm5hKGFycl90aW1lKSkKYGBgCgpGb3IgdGhlIGBnZXlzZXJgIGRhdGEgc2V0IGZyb20gdGhlIGBNQVNTYCBwYWNrYWdlIHdlIHVzZWQgYGxhZ2AgdG8Kb2J0YWluIHRoZSBkdXJhdGlvbiBvZiB0aGUgcHJldmlvdXMgZXJ1cHRpb24sCgpgYGB7cn0KZGF0YShnZXlzZXIsIHBhY2thZ2UgPSAiTUFTUyIpCmdleXNlcjIgPC0gbXV0YXRlKGdleXNlciwKICAgICAgICAgICAgICAgICAgcHJldl9kdXJhdGlvbiA9IGxhZyhkdXJhdGlvbikpCmhlYWQoZ2V5c2VyMiwgNCkKYGBgCgpgYGB7ciBnZXlzZXItbGFnLCBldmFsID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRX0KdGhtIDwtIHRoZW1lX21pbmltYWwoKSArCiAgICB0aGVtZSh0ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxNikpCmdncGxvdChnZXlzZXIyKSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gcHJldl9kdXJhdGlvbiwKICAgICAgICAgICAgICAgICAgIHkgPSB3YWl0aW5nKSkgKwogICAgdGhtCmBgYAoKYGBge3IgZ2V5c2VyLWxhZywgZWNobyA9IEZBTFNFfQpgYGAKClRoZSB0b3AgZml2ZSB2ZWhpY2xlcyBpbiBgY3R5YCBnYXMgbWlsZWFnZSAodGhlcmUgbWF5IGJlIG1vcmUgdGhhbgpmaXZlIGlmIHRoZXJlIGlzIGEgdGllIGZvciBmaWZ0aCBwbGFjZSk6CgpgYGB7cn0KbXV0YXRlKG5ld21wZzEsIGN0eV9yYW5rID0gbWluX3JhbmsoZGVzYyhjdHkpKSkgfD4KICAgIGZpbHRlcihjdHlfcmFuayA8PSA1KQpgYGAKCkFuIGFsdGVybmF0aXZlIHRvIGV4cGxpY2l0bHkgY29tcHV0aW5nIHJhbmtzIGFuZCBmaWx0ZXJpbmcgaXMgdGhlCmBzbGljZV9tYXgoKWAgdXRpbGl0eSBmdW5jdGlvbjoKCmBgYHtyfQpzbGljZV9tYXgobmV3bXBnMSwgY3R5LCBuID0gNSkKYGBgCgpUaGUgYHBlcmNlbnRfcmFua2AgZnVuY3Rpb24gcHJvZHVjZXMgYSAncGVyY2VudGlsZSByYW5rJyAoYnV0IGJldHdlZW4KMCBhbmQgMSkuCgpUaGUgdG9wIDUlIG9mIGNvdW50cmllcyBpbiBsaWZlIGV4cGVjdGFuY3kgaW4gMjAwNyBmcm9tIHRoZQpgZ2FwbWluZGVyYCBkYXRhIHNldDoKCmBgYHtyfQpsaWJyYXJ5KGdhcG1pbmRlcikKZ20gPC0gZmlsdGVyKGdhcG1pbmRlciwgeWVhciA9PSAyMDA3KQpnbV90b3AgPC0gZmlsdGVyKGdtLCBwZXJjZW50X3JhbmsoZGVzYyhsaWZlRXhwKSkgPD0gMC4wNSkKZ21fdG9wCmBgYAoKQW5kIHRoZSBib3R0b20gNSU6CmBgYHtyfQpmaWx0ZXIoZ20sIHBlcmNlbnRfcmFuayhsaWZlRXhwKSA8PSAwLjA1KQpgYGAKCldpdGhpbiB0aGVzZSByZXN1bHRzIHRoZSBvcmlnaW5hbCByb3cgb3JkZXIgaXMgcmV0YWluZWQuCgpUaGUgYGFycmFuZ2VgIGZ1bmN0aW9uIGNhbiBiZSB1c2VkIHRvIGNoYW5nZSB0aGlzLgoKCiMjIEFycmFuZ2luZyBSb3dzCgpgYXJyYW5nZSgpYCByZW9yZGVycyB0aGUgcm93cyBhY2NvcmRpbmcgdG8gb25lIG9yIG1vcmUgdmFyaWFibGVzLgoKQnkgZGVmYXVsdCB0aGUgb3JkZXIgaXMgc21hbGxlc3QgdG8gbGFyZ2VzdDsgdXNpbmcgYGRlc2MoKWAgcmV2ZXJzZXMgdGhpcy4KCkFkZGl0aW9uYWwgdmFyaWFibGVzIHdpbGwgYmUgdXNlZCB0byBicmVhayB0aWVzLgoKQXJyYW5naW5nIHRoZSB0b3AgNSUgb2YgY291bnRyaWVzIGluIGxpZmUgZXhwZWN0YW5jeSBpbiAyMDA3IGJ5IGxpZmUKZXhwZWN0YW5jeToKCmBgYHtyfQphcnJhbmdlKGdtX3RvcCwgZGVzYyhsaWZlRXhwKSkKYGBgCgpUbyBvcmRlciBieSBjb250aW5lbnQgbmFtZSBhbmQgb3JkZXIgYnkgbGlmZSBleHBlY3RhbmN5IHdpdGhpbiBjb250aW5lbnQ6CgpgYGB7cn0KYXJyYW5nZShnbV90b3AsIGNvbnRpbmVudCwgZGVzYyhsaWZlRXhwKSkKYGBgCgpNaXNzaW5nIHZhbHVlcyBhcmUgc29ydGVkIHRvIHRoZSBlbmQ6CgpgYGB7cn0KZGYgPC0gZGF0YS5mcmFtZSh4ID0gYyg1LCAyLCBOQSwgMykpCmFycmFuZ2UoZGYsIHgpCmFycmFuZ2UoZGYsIGRlc2MoeCkpCmBgYAoKVXNpbmcgYGlzLm5hYCB5b3UgY2FuIGFycmFuZ2UgdG8gcHV0IHRoZW0gYXQgdGhlIGJlZ2lubmluZzoKCmBgYHtyfQphcnJhbmdlKGRmLCBkZXNjKGlzLm5hKHgpKSwgeCkKYGBgCgoKIyMgU3VtbWFyaXppbmcgYW5kIEdyb3VwaW5nCgpgc3VtbWFyaXplKClgIGNvbGxhcHNlcyBhIHRhYmxlIGRvd24gdG8gb25lIHJvdyBvZiBzdW1tYXJpZXMuCgpUaGUgYXZlcmFnZSBhbmQgbWF4aW1hbCBkZXBhcnR1cmUgYW5kIGFycml2YWwgZGVsYXlzIGZvciBmbGlnaHRzLAphbG9uZyB3aXRoIHRoZSB0b3RhbCBudW1iZXIgb2YgZmxpZ2h0czoKCmBgYHtyfQpzdW1tYXJpemUoZmxpZ2h0cywKICAgICAgICAgIGF2ZV9kZXBfZGVsYXkgPSBtZWFuKGRlcF9kZWxheSwgbmEucm0gPSBUUlVFKSwKICAgICAgICAgIGF2ZV9hcnJfZGVsYXkgPSBtZWFuKGFycl9kZWxheSwgbmEucm0gPSBUUlVFKSwKICAgICAgICAgIG1heF9kZXBfZGVsYXkgPSBtYXgoZGVwX2RlbGF5LCBuYS5ybSA9IFRSVUUpLAogICAgICAgICAgbWF4X2Fycl9kZWxheSA9IG1heChhcnJfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICBuID0gbigpKQpgYGAKCgojIyMgR3JvdXBlZCBTdW1tYXJpZXMKCmBzdW1tYXJpemUoKWAgaXMgbW9zdCBjb21tb25seSB1c2VkIHdpdGggX2dyb3VwZWQgZGF0YV8gdG8gcHJvdmlkZSBncm91cApsZXZlbCBzdW1tYXJpZXMuCgpHcm91cGluZyBkZWxheSBzdW1tYXJpZXMgYnkgZGVzdGluYXRpb246CgpgYGB7ciwgd2FybmluZyA9IEZBTFNFfQpmbF9kZXN0IDwtCiAgICBncm91cF9ieShmbGlnaHRzLCBkZXN0KSB8PgogICAgc3VtbWFyaXplKGF2ZV9kZXBfZGVsYXkgPSBtZWFuKGRlcF9kZWxheSwgbmEucm0gPSBUUlVFKSwKICAgICAgICAgICAgICBhdmVfYXJyX2RlbGF5ID0gbWVhbihhcnJfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICAgICAgbWF4X2RlcF9kZWxheSA9IG1heChkZXBfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICAgICAgbWF4X2Fycl9kZWxheSA9IG1heChhcnJfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICAgICAgbiA9IG4oKSkgfD4KICAgIHVuZ3JvdXAoKQpoZWFkKGZsX2Rlc3QsIDUpCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CkNhbGxpbmcgYHVuZ3JvdXBgIGlzIG5vdCBhbHdheXMgbmVjZXNzYXJ5IGJ1dCBzb21ldGltZXMgaXQgaXMsIHNvIGl0CmlzIGEgZ29vZCBoYWJpdCB0byBnZXQgaW50by4KPC9kaXY+CgpBbiBhbHRlcm5hdGl2ZSB0byB1c2luZyBgZ3JvdXBfYnkoKWAgYW5kIGB1bmdyb3VwKClgIGlzIHRvIHVzZSB0aGUKYC5ieWAgYXJndW1lbnQgdG8gYHN1bW1hcml6ZSgpYDoKCmBgYHtyLCB3YXJuaW5nID0gRkFMU0V9CmZsaWdodHMgfD4KICAgIHN1bW1hcml6ZShhdmVfZGVwX2RlbGF5ID0gbWVhbihkZXBfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICAgICAgYXZlX2Fycl9kZWxheSA9IG1lYW4oYXJyX2RlbGF5LCBuYS5ybSA9IFRSVUUpLAogICAgICAgICAgICAgIG1heF9kZXBfZGVsYXkgPSBtYXgoZGVwX2RlbGF5LCBuYS5ybSA9IFRSVUUpLAogICAgICAgICAgICAgIG1heF9hcnJfZGVsYXkgPSBtYXgoYXJyX2RlbGF5LCBuYS5ybSA9IFRSVUUpLAogICAgICAgICAgICAgIG4gPSBuKCksCiAgICAgICAgICAgICAgLmJ5ID0gZGVzdCkgfD4KICAgIGFycmFuZ2UoZGVzdCkgfD4gICAgICAjIyB0byBtYXRjaCBncm91cF9ieSgpIHJlc3VsdAogICAgaGVhZCgpCmBgYAoKVGhpcyBpcyBjdXJyZW50bHkgbWFya2VkIGFzIF9FeHBlcmltZW50YWxfIGluIHRoZSBkb2N1bWVudGF0aW9uLgoKVGhlIGdyb3VwIGxldmVsIHN1bW1hcmllcyBjYW4gYmUgZnVydGhlciB0cmFuc2Zvcm1lZCwgZm9yIGV4YW1wbGUKdG8gaWRlbnRpZnkgdGhlIHRvcCAxMCBkZXN0aW5hdGlvbnMgZm9yIGF2ZXJhZ2UgZGVsYXlzOgoKYGBge3J9Cm11dGF0ZShmbF9kZXN0LCByYW5rID0gbWluX3JhbmsoZGVzYyhhdmVfZGVwX2RlbGF5KSkpIHw+CiAgICBmaWx0ZXIocmFuayA8PSAxMCkKYGBgCgoKIyMjIE1pc3NpbmcgVmFsdWVzCgpNb3N0IHN1bW1hcmllcyB3aWxsIGJlIGBOQWAgaWYgYW55IG9mIHRoZSBkYXRhIHZhbHVlcyBiZWluZyBzdW1tYXJpemVkCmFyZSBgTkFgLgoKVHlwaWNhbGx5LCBzdW1tYXJ5IGZ1bmN0aW9ucyBjYW4gYmUgY2FsbGVkIHdpdGggYG5hLnJtID0gVFJVRWAgdG8KcHJvZHVjZSB0aGUgc3VtbWFyeSBmb3IgYWxsIG5vbi1taXNzaW5nIHZhbHVlcy4KCmBgYHtyfQpzdW1tYXJpemUoZmxpZ2h0cywKICAgICAgICAgIGF2ZV9kZXBfZGVsYXkgPSBtZWFuKGRlcF9kZWxheSwgbmEucm0gPSBUUlVFKSwKICAgICAgICAgIGF2ZV9hcnJfZGVsYXkgPSBtZWFuKGFycl9kZWxheSwgbmEucm0gPSBUUlVFKSwKICAgICAgICAgIG1heF9kZXBfZGVsYXkgPSBtYXgoZGVwX2RlbGF5LCBuYS5ybSA9IFRSVUUpLAogICAgICAgICAgbWF4X2Fycl9kZWxheSA9IG1heChhcnJfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICBuID0gbigpKQpgYGAKCllvdSBjYW4gYWxzbyB1c2UgYGZpbHRlcmAgdG8gcmVtb3ZlIHJvd3Mgd2l0aCBgTkFgIHZhbHVlcyBpbiB0aGUKdmFyaWFibGVzIHRvIGJlIHN1bW1hcml6ZWQ6CgpgYGB7cn0KZmlsdGVyKGZsaWdodHMsICEgaXMubmEoZGVwX2RlbGF5KSwgISBpcy5uYShhcnJfZGVsYXkpKSB8PgogICAgc3VtbWFyaXplKGF2ZV9kZXBfZGVsYXkgPSBtZWFuKGRlcF9kZWxheSksCiAgICAgICAgICAgICAgYXZlX2Fycl9kZWxheSA9IG1lYW4oYXJyX2RlbGF5KSwKICAgICAgICAgICAgICBtYXhfZGVwX2RlbGF5ID0gbWF4KGRlcF9kZWxheSksCiAgICAgICAgICAgICAgbWF4X2Fycl9kZWxheSA9IG1heChhcnJfZGVsYXkpLAogICAgICAgICAgICAgIG4gPSBuKCkpCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+ClRoZSB0d28gYXBwcm9hY2hlcyBkbyBwcm9kdWNlIGRpZmZlcmVudCBjb3VudHMuCjwvZGl2PgoKCiMjIyBDb3VudHMKClRoZSBoZWxwZXIgZnVuY3Rpb24gYG4oKWAgcHJvdmlkZXMgYSBjb3VudCBvZiB0aGUgbnVtYmVyIG9mIHJvd3MgaW4KdGhlIHRhYmxlIGJlaW5nIHN1bW1hcml6ZWQuCgpJbmNsdWRpbmcgY291bnRzIHdpdGggZ3JvdXBlZCBzdW1tYXJpZXMgaXMgdXN1YWxseSBhIGdvb2QgaWRlYS4KClN1bW1hcmllcyBiYXNlZCBvbiBzbWFsbCBjb3VudHMgYXJlIGxpa2VseSB0byBiZSBtdWNoIGxlc3MgcmVsaWFibGUKdGhhbiBvbmVzIGJhc2VkIG9uIGxhcmdlciBjb3VudHMuCgpUaGlzIGlzIHJlZmxlY3RlZCBpbiBoaWdoZXIgdmFyaWFiaWxpdHkgYW1vbmcgYXZlcmFnZXMgZm9yCmRlc3RpbmF0aW9ucyB3aXRoIGZld2VyIGZsaWdodHM6CgpgYGB7ciBhdmVfYXJyX2RlbGF5LCBldmFsID0gRkFMU0V9CmdncGxvdChmbF9kZXN0KSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gbiwKICAgICAgICAgICAgICAgICAgIHkgPSBhdmVfYXJyX2RlbGF5KSwKICAgICAgICAgICAgICAgbmEucm0gPSBUUlVFKSArCiAgICB0aG0KYGBgCgpgYGB7ciBhdmVfYXJyX2RlbGF5LCBlY2hvID0gRkFMU0V9CmBgYAoKQSBgcGxvdGx5YCB2ZXJzaW9uIGFsbG93cyB0aGUgaW5kaXZpZHVhbCBkZXN0aW5hdGlvbnMgdG8gYmUgaWRlbnRpZmllZAplYXNpbHk6CgpgYGB7ciBhdmVfYXJyX2RlbGF5X3Bsb3RseSwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KHBsb3RseSkKcCA8LSBnZ3Bsb3QoZmxfZGVzdCkgKwogICAgZ2VvbV9wb2ludChhZXMoeCA9IG4sCiAgICAgICAgICAgICAgICAgICB5ID0gYXZlX2Fycl9kZWxheSwKICAgICAgICAgICAgICAgICAgIHRleHQgPSBkZXN0KSwKICAgICAgICAgICAgICAgbmEucm0gPSBGQUxTRSkgKwogICAgdGhtCmdncGxvdGx5KHAsIHRvb2x0aXAgPSAidGV4dCIpCmBgYAoKYGBge3IgYXZlX2Fycl9kZWxheV9wbG90bHksIGVjaG8gPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFfQpgYGAKCkl0IHdvdWxkIGJlIG1vcmUgdXNlci1mcmllbmRseSB0byBzaG93IHRoZSBhaXJwb3J0IG5hbWUgdGhhbiB0aGUgdGhyZWUKbGV0dGVyIEZBQSBjb2RlLgoKCiMjIyBVc2VmdWwgU3VtbWFyaWVzCgpTb21lIHVzZWZ1bCBzdW1tYXJ5IGZ1bmN0aW9uczoKCiogTG9jYXRpb246IGBtZWFuYCwgYG1lZGlhbmAKCiogU3ByZWFkOiBgc2RgLCBgSVFSYCwgYG1hZGAKCiogUmFuazogYG1pbmAsIGBxdWFudGlsZWAsIGBtYXhgCgoqIFBvc2l0aW9uOiBgZmlyc3RgLCBgbnRoYCwgYGxhc3RgCgoqIENvdW50czogYG5gLgoKCiMjIEFpcmxpbmUgQXJyaXZhbCBEZWxheXMKCgojIyMgRGlmZmVyZW50IERlbGF5IE1lYXN1cmVzCgpTb21ldGltZXMgaXQgaXMgdXNlZnVsIHRvIGxvb2sgYXQgc3Vic2V0cyBvciB0cnVuY2F0ZWQgdmVyc2lvbnMgb2YgYQp2YXJpYWJsZS4KCkFpcmxpbmVzIGxpa2UgdG8gYXJyYW5nZSBzY2hlZHVsZXMgc28gdGhleSBhcmUgb2Z0ZW4gZWFybHkuCgpgYGB7ciwgZmlnLmhlaWdodCA9IDQsIGNsYXNzLnNvdXJjZSA9ICJmb2xkLWhpZGUifQpnZ3Bsb3QoZmxpZ2h0cywgYWVzKHggPSBhcnJfZGVsYXksCiAgICAgICAgICAgICAgICAgICAgeSA9IGFmdGVyX3N0YXQoZGVuc2l0eSkpKSArCiAgICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aCA9IDEsCiAgICAgICAgICAgICAgICAgICBuYS5ybSA9IFRSVUUsCiAgICAgICAgICAgICAgICAgICBmaWxsID0gImRlZXBza3libHVlMyIpICsKICAgIHhsaW0oYygtMTAwLCAyNTApKSArCiAgICB0aG0KYGBgCgpUaGlzIHJlZHVjZXMgdGhlIGF2ZXJhZ2UgYXJyaXZhbCBkZWxheSBidXQgbWF5IG5vdCBoZWxwIHRyYXZlbGVycyBtdWNoLgoKVHdvIG9wdGlvbnM6CgoqIENvbnNpZGVyIG9ubHkgdGhlIHBvc2l0aXZlIGRlbGF5cyBmb3IgdGhlIGF2ZXJhZ2UuCgoqIFRyZWF0IG5lZ2F0aXZlIGRlbGF5cyBhcyB6ZXJvICh6ZXJvLXRydW5jYXRpb24pLgoKVGhlIG1lYW4gb2YgdGhlIHBvc2l0aXZlIGRlbGF5cyBjYW4gYmUgY29tcHV0ZWQgdXNpbmcgX3N1YnNldHRpbmdfOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KbWVhbihhcnJfZGVsYXlbYXJyX2RlbGF5ID4gMF0pCmBgYAoKVGhlIG1lYW4gb2YgdGhlIHplcm8tdHJ1bmNhdGVkIGRlbGF5cyBjYW4gYmUgY29tcHV0ZWQgdXNpbmcgdGhlIGBwbWF4YApmdW5jdGlvbjoKCmBgYHtyLCBldmFsID0gRkFMU0V9Cm1lYW4ocG1heChhcnJfZGVsYXksIDApKQpgYGAKVGhlIHN1bW1hcmllcyBhcmUKYGBge3J9CmZsX2FyciA8LSBmaWx0ZXIoZmxpZ2h0cywgISBpcy5uYShhcnJfZGVsYXkpKQpmbF9hdnMgPC0KICAgIGdyb3VwX2J5KGZsX2FyciwgZGVzdCkgfD4KICAgIHN1bW1hcml6ZShhdmUgPSBtZWFuKGFycl9kZWxheSksCiAgICAgICAgICAgICAgYXZlX3BvcyA9IG1lYW4oYXJyX2RlbGF5W2Fycl9kZWxheSA+IDBdKSwKICAgICAgICAgICAgICBhdmVfcG9zX3plcm8gPSBtZWFuKHBtYXgoYXJyX2RlbGF5LCAwKSksCiAgICAgICAgICAgICAgbiA9IG4oKSkgfD4KICAgIHVuZ3JvdXAoKQpoZWFkKGZsX2F2cykKYGBgCgpPbmUgYXBwcm9hY2ggdG8gdmlzdWFsaXppbmcgdGhlIHJlc3VsdHMgd2l0aCBjb2xvciBjb2RpbmcgdGhlIHRocmVlCmRpZmZlcmVudCBzdW1tYXJpZXM6CgoqIFZpZXcgdGhlIGBmbF9hdnNgIHRhYmxlIGFzIGEgd2lkZSBmb3JtYXQgd2l0aCBgYXZzYCwgYGF2ZV9wb3NgLCBhbmQKICBgYXZlX3Bvc196ZXJvYCBhcyB0aHJlZSBkaWZmZXJlbnQgbWVhc3VyZW1lbnQgdHlwZXMuCgoqIFVzZSBgcGl2b3RfbG9uZ2VyKClgIHRvIGNvbnZlcnQgdG8gYSBsb25nIGZvcm1hdCB3aXRoIGEgdmFyaWFibGUKICBgd2hpY2hgIHRvIGhvbGQgdGhlIHN1bW1hcnkgdHlwZSBhbmQgYSB2YXJpYWJsZSBgZGVsYXlgIHRvCiAgaG9sZCB0aGUgc3VtbWFyeSB2YWx1ZToKCmBgYHtyfQpsaWJyYXJ5KHRpZHlyKQpmbCA8LSBwaXZvdF9sb25nZXIoZmxfYXZzLCAyIDogNCwKICAgICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gIndoaWNoIiwKICAgICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJkZWxheSIpCmFycmFuZ2UoZmwsIGRlc3QpCmBgYAoKQSBwbG90IGlzIHRoZW4gZWFzeSB0byBjcmVhdGU6CgpgYGB7ciBhdmVzX3Bpdm90LCBldmFsID0gRkFMU0V9CmdncGxvdChmbCwgYWVzKHggPSBuLAogICAgICAgICAgICAgICB5ID0gZGVsYXksCiAgICAgICAgICAgICAgIGNvbG9yID0gd2hpY2gpKSArCiAgICBnZW9tX3BvaW50KCkgKwogICAgdGhtCmBgYAoKYGBge3IgYXZlc19waXZvdCwgZWNobyA9IEZBTFNFfQpgYGAKCkFuIGFsdGVybmF0aXZlOgoKKiBVc2Ugb25lIGxheWVyIHBlciBzdW1tYXJ5IHR5cGUuCgoqIFNwZWNpZnkgdGhlIHN1bW1hcnkgdHlwZSBmb3IgZWFjaCBsYXllciBhcyBhIGBjb2xvcmAgYWVzdGhldGljIHZhbHVlLgoKYGBge3IgYXZlc19sYXllciwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZmxfYXZzLCBhZXMoeCA9IG4pKSArCiAgICBnZW9tX3BvaW50KGFlcyh5ID0gYXZlLAogICAgICAgICAgICAgICAgICAgY29sb3IgPSAiYXZlIikpICsKICAgIGdlb21fcG9pbnQoYWVzKHkgPSBhdmVfcG9zLAogICAgICAgICAgICAgICAgICAgY29sb3IgPSAiYXZlX3BvcyIpKSArCiAgICBnZW9tX3BvaW50KGFlcyh5ID0gYXZlX3Bvc196ZXJvLAogICAgICAgICAgICAgICAgICAgY29sb3IgPSAiYXZlX3Bvc196ZXJvIikpICsKICAgIHRobQpgYGAKCmBgYHtyIGF2ZXNfbGF5ZXIsIGVjaG8gPSBGQUxTRX0KYGBgCgpUaGUgb25lIG1pc3NpbmcgdmFsdWUgY2FzZToKCmBgYHtyfQppbmNvbXBsZXRlX2Nhc2VzKGZsX2F2cykKYGBgCgoKIyMjIERlbGF5IFByb3BvcnRpb25zCgpQcm9wb3J0aW9ucyBzYXRpc2Z5aW5nIGEgY29uZGl0aW9uIGNhbiBiZSBjb21wdXRlZCBhcyBhIG1lYW4Kb2YgdGhlIGxvZ2ljYWwgZXhwcmVzc2lvbiBmb3IgdGhlIGNvbmRpdGlvbjoKClRoZSBwcm9wb3J0aW9uIG9mIGFsbCBmbGlnaHRzIHRoYXQgd2VyZSBkZWxheWVkOgoKYGBge3J9Cm1lYW4oZmxfYXJyJGFycl9kZWxheSA+IDApCmBgYAoKR3JvdXBlZCBieSBkZXN0aW5hdGlvbjoKCmBgYHtyfQpmbF9wcm9wcyA8LQogICAgZ3JvdXBfYnkoZmxfYXJyLCBkZXN0KSB8PgogICAgc3VtbWFyaXplKAogICAgICAgIHBkMCA9IG1lYW4oYXJyX2RlbGF5ID4gMCksICAgIyMgcHJvcG9ydGlvbiBkZWxheWVkCiAgICAgICAgcGQxMCA9IG1lYW4oYXJyX2RlbGF5ID4gMTApLCAjIyBtb3JlIHRoYW4gMTAgbWludXRlcwogICAgICAgIHBkMjAgPSBtZWFuKGFycl9kZWxheSA+IDIwKSwgIyMgbW9yZSB0aGFuIDIwIG1pbnV0ZXMKICAgICAgICBuID0gbigpKSB8PgogICAgdW5ncm91cCgpCmZsX3Byb3BzCmBgYAoKVGhlc2UgY2FuIGJlIHZpc3VhbGl6ZWQgc2ltaWxhcmx5OgoKYGBge3IgcHJvcF9kZWxheWVkX3Bpdm90LCBldmFsID0gRkFMU0V9CnBpdm90X2xvbmdlcihmbF9wcm9wcywgMiA6IDQsCiAgICAgICAgICAgICBuYW1lc190byA9ICJ3aGljaCIsCiAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAicHJvcCIpIHw+CiAgICBnZ3Bsb3QoYWVzKHggPSBuLAogICAgICAgICAgICAgICB5ID0gcHJvcCwKICAgICAgICAgICAgICAgY29sb3IgPSB3aGljaCkpICsKICAgIGdlb21fcG9pbnQoKSArCiAgICB0aG0KYGBgCgpgYGB7ciBwcm9wX2RlbGF5ZWRfcGl2b3QsIGVjaG8gPSBGQUxTRX0KYGBgCgoKIyMgR3JvdXBlZCBNdXRhdGUgYW5kIEZpbHRlcgoKYG11dGF0ZSgpYCwgYGZpbHRlcigpYCwgYGFycmFuZ2UoKWAsIGBzbGljZV9tYXgoKWAsIGFuZCBgc2xpY2VfbWluKClgCmNhbiBiZSB1c2VkIG9uIGEgZ3JvdXAgbGV2ZWwuCgoKIyMjIFN0YW5kYXJkaXppbmcgV2l0aGluIEdyb3VwcwoKYG11dGF0ZWAgd2l0aGluIGdyb3VwcyBjYW4gYmUgdXNlZCBmb3Igc3RhbmRhcmRpemluZyB3aXRoaW4gYSBncm91cC4KCkZvciBleGFtcGxlLCB0byBjb21wdXRlIHRoZSBwcm9wb3J0aW9uIG9mIGEgY29udGluZW50J3MgcG9wdWxhdGlvbgpsaXZpbmcgaW4gZWFjaCBjb3VudHJ5IGZvciBlYWNoIHllYXI6Cgo8IS0gbm8gbG9uZ2VyIG5lZWQgcG9wID0gYXMubnVtZXJpYyhwb3ApIHNpbmNlIHN1bSgpIHN3aXRjaGVzIHRvCglkb3VibGUgb24gb3ZlcmZsb3ctLT4KCmBgYHtyfQpnbSA8LSBncm91cF9ieShnYXBtaW5kZXIsIGNvbnRpbmVudCwgeWVhcikgfD4KICAgIG11dGF0ZShwcG9wID0gcG9wIC8gc3VtKHBvcCkpIHw+CiAgICB1bmdyb3VwKCkKYGBgCgpGb3IgT2NlYW5pYSAoaW4gdGhpcyBkYXRhIHNldCkgYW5kIHllYXJzIDE5NTIgYW5kIDIwMDc6CgpgYGB7cn0KZ21fb2MgPC0gZmlsdGVyKGdtLCBjb250aW5lbnQgPT0gIk9jZWFuaWEiLCB5ZWFyICVpbiUgYygxOTUyLCAyMDA3KSkgfD4KICAgIGFycmFuZ2UoeWVhcikKZ21fb2MKYGBgCgpBIHF1aWNrIGNoZWNrIHRoYXQgd2l0aGluIHllYXIgYW5kIGNvbnRpbmVudCB0aGUgcHJvcG9ydGlvbnMgYWRkCnVwIHRvIG9uZToKCmBgYHtyfQpncm91cF9ieShnbV9vYywgY29udGluZW50LCB5ZWFyKSB8PiBzdW1tYXJpemUoc3VtX3Bwb3AgPSBzdW0ocHBvcCkpCmBgYAoKRm9jdXMgb24gdGhlIDIgY291bnRyaWVzIGluIGVhY2ggY29udGluZW50IHdpdGggdGhlIGhpZ2hlc3QKcHJvcG9ydGlvbnMgb2YgdGhlIGNvbnRpbmVudCdzIHBvcHVsYXRpb246CgpgYGB7ciwgZmlnLndpZHRoID0gOSwgY2xhc3Muc291cmNlID0gImZvbGQtaGlkZSJ9CmdtdDIgPC0gZ3JvdXBfYnkoZ20sIGNvbnRpbmVudCwgeWVhcikgfD4KICAgIHNsaWNlX21heChwcG9wLCBuID0gMikgfD4KICAgIHVuZ3JvdXAoKSB8PgogICAgZmlsdGVyKGNvbnRpbmVudCAhPSAiT2NlYW5pYSIpICMjIGRyb3AgYXMgdGhlcmUgYXJlIG9ubHkgdHdvCmdncGxvdChnbXQyLCBhZXMoeCA9IHllYXIsIHkgPSBwcG9wLAogICAgICAgICAgICAgICAgIGdyb3VwID0gY291bnRyeSwgY29sb3IgPSBjb3VudHJ5KSkgKwogICAgZ2VvbV9saW5lKCkgKyBnZW9tX3BvaW50KCkgKwogICAgZmFjZXRfd3JhcCh+IGNvbnRpbmVudCwgc2NhbGVzID0gImZyZWVfeSIpICsKICAgIHRobQpgYGAKCkFub3RoZXIgYXBwcm9hY2g6CgoqIENyZWF0ZSBzZXBhcmF0ZSBwbG90cyBmb3IgZWFjaCBjb250aW5lbnQuCgoqIEFkanVzdCB0aGUgdGhlbWVzIGEgbGl0dGxlLgoKKiBVc2UgYGZhY2V0X3dyYXBgIHRvIGdldCB0aGUgZmFjZXQgbGFiZWwuCgoqIEFycmFuZ2UgdGhlIHBsb3RzLCBlLmcuIHVzaW5nIGBwYXRjaHdvcmtgIG9wZXJhdG9ycy4KCkEgdXNlZnVsIGZlYXR1cmUgaXMgdGhhdCB0aGUgYCUrJWAgb3BlcmF0b3IgY2FuIGJlIHVzZWQgdG8gY2hhbmdlIHRoZQpiYXNlIGRhdGEgZnJhbWUgZm9yIGEgcGxvdDoKCmBgYHtyLCBmaWcuaGVpZ2h0ID0gNi41LCBmaWcud2lkdGggPSA4LCBjbGFzcy5zb3VyY2UgPSAiZm9sZC1oaWRlIn0KcEFmIDwtIGdncGxvdChmaWx0ZXIoZ210MiwgY29udGluZW50ID09ICJBZnJpY2EiKSwKICAgICAgICAgICAgICBhZXMoeCA9IHllYXIsIHkgPSBwcG9wLCBjb2wgPSBjb3VudHJ5KSkgKwogICAgZ2VvbV9saW5lKCkgKyBnZW9tX3BvaW50KCkgKwogICAgZmFjZXRfd3JhcCh+IGNvbnRpbmVudCkgKwogICAgdGhtICsKICAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJ0b3AiLAogICAgICAgICAgbGVnZW5kLnRpdGxlID0gZWxlbWVudF9ibGFuaygpKQp0aG1fbnggPC0gdGhlbWUoYXhpcy50aXRsZS54ID0gZWxlbWVudF9ibGFuaygpKQp0aG1fbnkgPC0gdGhlbWUoYXhpcy50aXRsZS55ID0gZWxlbWVudF9ibGFuaygpKQpwQW0gPC0gcEFmICUrJSBmaWx0ZXIoZ210MiwgY29udGluZW50ID09ICJBbWVyaWNhcyIpCnBBcyA8LSBwQWYgJSslIGZpbHRlcihnbXQyLCBjb250aW5lbnQgPT0gIkFzaWEiKQpwRXUgPC0gcEFmICUrJSBmaWx0ZXIoZ210MiwgY29udGluZW50ID09ICJFdXJvcGUiKSArCiAgICBndWlkZXMoY29sb3IgPSBndWlkZV9sZWdlbmQobnJvdyA9IDIpKQpsaWJyYXJ5KHBhdGNod29yaykKKChwQWYgKyB0aG1fbngpIHwgKHBBbSArIHRobV9ueCArIHRobV9ueSkpIC8KICAgIChwQXMgfCAocEV1ICsgdGhtX255KSkKYGBgCgoKIyMjIENoYW5nZXMgT3ZlciBUaW1lCgpDaGFuZ2VzIG92ZXIgdGltZSB3aXRoaW4gYSBjb3VudHJ5IGNhbiBhbHNvIGJlIGV4YW1pbmVkIHdpdGggYSBncm91cGVkCmBtdXRhdGVgIGFuZCB0aGUgYGxhZ2Agb3BlcmF0b3IuCgpGb3IgdmFyaWFibGVzIGNvbGxlY3RlZCBvdmVyIHRpbWUgdGhlIGBsYWdgIGZ1bmN0aW9uIGNhbiBiZSB1c2VkIHRvCmNvbXB1dGUgdGhlIHByZXZpb3VzIHZhbHVlIG9mIGEgdmFyaWFibGU6CgpgYGB7cn0KZ21fdXMgPC0gZmlsdGVyKGdhcG1pbmRlciwKICAgICAgICAgICAgICAgIGNvdW50cnkgPT0gIlVuaXRlZCBTdGF0ZXMiKSB8PgogICAgdHJhbnNtdXRlKHllYXIsCiAgICAgICAgICAgICAgbGlmZUV4cCwKICAgICAgICAgICAgICBwcmV2X2xpZmVFeHAgPSBsYWcobGlmZUV4cCkpCmdtX3VzCmBgYAoKVGhpcyBjYW4gdGhlbiBiZSB1c2VkIHRvIGNvbXB1dGUgdGhlIGNoYW5nZSBmcm9tIG9uZSBwZXJpb2QgdG8gdGhlIG5leHQ6CgpgYGB7cn0KbXV0YXRlKGdtX3VzLAogICAgICAgbGVfY2hhbmdlID0gbGlmZUV4cCAtIHByZXZfbGlmZUV4cCkKYGBgCgpVc2UgdGhpcyBpZGVhIHRvIGNvbXB1dGUgY2hhbmdlcyBpbiBgbGlmZUV4cGAgYW5kIGBnZHBQZXJjYXBgIGZvciBhbGwKY291bnRyaWVzIGluIGEgZ3JvdXBlZCBgbXV0YXRlYDoKCmBgYHtyfQpnbSA8LSBncm91cF9ieShnYXBtaW5kZXIsIGNvdW50cnkpIHw+CiAgICBtdXRhdGUoCiAgICAgICAgbGVfY2hhbmdlID0gbGlmZUV4cCAtIGxhZyhsaWZlRXhwKSwKICAgICAgICBnZHBfY2hhbmdlID0gZ2RwUGVyY2FwIC0gbGFnKGdkcFBlcmNhcCksCiAgICAgICAgZ2RwX3BjdF9jaGFuZ2UgPQogICAgICAgICAgICAxMDAgKiBnZHBfY2hhbmdlIC8gbGFnKGdkcFBlcmNhcCkpIHw+CiAgICB1bmdyb3VwKCkKYGBgCgpgc2xpY2VfbWF4KClgIGNhbiB0aGVuIGJlIGFwcGxpZWQgdG8gZGF0YSBncm91cGVkIGJ5IGNvdW50cnkgdG8KZXh0cmFjdCB0aGUgcm93cyB3aXRoIHRoZSBoaWdoZXN0IGBnZHBfcGN0X2NoYW5nZWAgZm9yIGVhY2ggY291bnRyeToKCmBgYHtyfQpncm91cF9ieShnbSwgY291bnRyeSkgfD4KICAgIHNsaWNlX21heChnZHBfcGN0X2NoYW5nZSwgbiA9IDEpIHw+CiAgICB1bmdyb3VwKCkgfD4KICAgIHNlbGVjdCgxIDogMywgY29udGFpbnMoImdkcCIpKQpgYGAKCkdyb3VwaW5nIGJ5IGNvbnRpbmVudCBhbGxvd3MgdGhlIHdvcnN0IGRyb3AgaW4gbGlmZSBleHBlY3RhbmN5IGZvcgplYWNoIGNvbnRpbmVudCB0byBiZSBmb3VuZDoKCmBgYHtyfQpncm91cF9ieShnbSwgY29udGluZW50KSB8PgogICAgc2xpY2VfbWluKGxlX2NoYW5nZSwgbiA9IDEpIHw+CiAgICB1bmdyb3VwKCkgfD4KICAgIHNlbGVjdCgtY29udGFpbnMoImdkcCIpKQpgYGAKCldpdGhvdXQgZ3JvdXBpbmcsIHRoZSByb3cgd2l0aCBiZXN0IHBlcmNlbnQgR0RQIGNoYW5nZSBvdmVyYWxsIGNhbiBiZSBjb21wdXRlZDoKCmBgYHtyfQpzbGljZV9tYXgoZ20sIGdkcF9wY3RfY2hhbmdlLCBuID0gMSkgfD4KICAgIHNlbGVjdChjb3VudHJ5LCBjb250aW5lbnQsIHllYXIsIGdkcF9wY3RfY2hhbmdlKQpgYGAKClRoZSBjb21wdXRhdGlvbiBvZiB0aGUgYmVzdCBHRFAgZ3Jvd3RoIHllYXJzIGJ5IGNvdW50cnkgY2FuIGJlCmNvbWJpbmVkIGludG8gYSBzaW5nbGUgcGlwZToKCmBgYHtyLCByZXN1bHRzID0gImhpZGUifQpncm91cF9ieShnYXBtaW5kZXIsIGNvdW50cnkpIHw+CiAgICBtdXRhdGUoZ2RwX2NoYW5nZSA9IGdkcFBlcmNhcCAtIGxhZyhnZHBQZXJjYXApLAogICAgICAgICAgIGdkcF9wY3RfY2hhbmdlID0gMTAwICogZ2RwX2NoYW5nZSAvIGxhZyhnZHBQZXJjYXApKSB8PgogICAgdW5ncm91cCgpIHw+CiAgICBncm91cF9ieShjb3VudHJ5KSB8PgogICAgc2xpY2VfbWF4KGdkcF9wY3RfY2hhbmdlLCBuID0gMSkgfD4KICAgIHVuZ3JvdXAoKSB8PgogICAgc2VsZWN0KDEgOiAzLCBjb250YWlucygiZ2RwIikpCmBgYAoKU2luY2UgdGhlIGdyb3VwaW5ncyBvZiB0aGUgdHdvIHN0ZXBzIGFyZSB0aGUgc2FtZSBhbmQgZ3JvdXBlZCBgbXV0YXRlYApwcmVzZXJ2ZXMgdGhlIGdyb3VwIHN0cnVjdHVyZSwgdGhlIG1pZGRsZSBgdW5ncm91cGAvYGdyb3VwX2J5YCBjYW4gYmUKZHJvcHBlZDoKCmBgYHtyLCByZXN1bHRzID0gImhpZGUifQpncm91cF9ieShnYXBtaW5kZXIsIGNvdW50cnkpIHw+CiAgICBtdXRhdGUoZ2RwX2NoYW5nZSA9IGdkcFBlcmNhcCAtIGxhZyhnZHBQZXJjYXApLAogICAgICAgICAgIGdkcF9wY3RfY2hhbmdlID0gMTAwICogZ2RwX2NoYW5nZSAvIGxhZyhnZHBQZXJjYXApKSB8PgogICAgc2xpY2VfbWF4KGdkcF9wY3RfY2hhbmdlLCBuID0gMSkgfD4KICAgIHVuZ3JvdXAoKSB8PgogICAgc2VsZWN0KDEgOiAzLCBjb250YWlucygiZ2RwIikpCmBgYAoKPGRpdiBjbGFzcyA9ICJhbGVydCI+CllvdSBuZWVkIHRvIGJlIF92ZXJ5IGNhcmVmdWxfIG1ha2luZyBhbiBvcHRpbWl6YXRpb24gbGlrZSB0aGlzIQo8L2Rpdj4KCgojIyBKb2lucwoKPCEtLSBDaGVjayBvdXQgaHR0cHM6Ly93d3cuZ2Fycmlja2FkZW5idWllLmNvbS9wcm9qZWN0L3RpZHlleHBsYWluLyAtLT4KClRoaXMgaW50ZXJhY3RpdmUgcGxvdCB1c2VzIHRoZSBgdG9vbHRpcGAgdG8gc2hvdyB0aGUgYWlycG9ydCBjb2RlIGZvcgp0aGUgZGVzdGluYXRpb246CgpgYGB7ciwgY2xhc3Muc291cmNlID0gImZvbGQtaGlkZSJ9CmxpYnJhcnkocGxvdGx5KQpwIDwtIGdncGxvdChmbF9kZXN0KSArCiAgICBnZW9tX3BvaW50KGFlcyh4ID0gbiwKICAgICAgICAgICAgICAgICAgIHkgPSBhdmVfYXJyX2RlbGF5LAogICAgICAgICAgICAgICAgICAgdGV4dCA9IGRlc3QpKSArCiAgICB0aG0KZ2dwbG90bHkocCwgdG9vbHRpcCA9ICJ0ZXh0IikKYGBgCgpBIG5pY2VyIGFwcHJvYWNoIHdvdWxkIGJlIHRvIHNob3cgdGhlIGFpcnBvcnQgbmFtZS4KClRoZSBgYWlycG9ydHNgIHRhYmxlIHByb3ZpZGVzIHRoaXMgaW5mb3JtYXRpb246CgpgYGB7cn0KaGVhZChhaXJwb3J0cywgMikKYGBgCgpXZSBuZWVkIHRvIGdldCB0aGlzIGluZm9ybWF0aW9uIGludG8gdGhlIGBmbF9kZXN0YCBkYXRhIGZyYW1lLgoKQXMgYW5vdGhlciBleGFtcGxlLCB3ZSBtaWdodCB3YW50IHRvIGV4cGxvcmUgd2hldGhlciBhaXJjcmFmdCBhZ2UKaXMgcmVsYXRlZCB0byBkZWxheXMuCgpUaGUgYHBsYW5lc2AgdGFibGUgcHJvdmlkZXMgdGhlIHllYXIgb2YgbWFudWZhY3R1cmU6CgpgYGB7cn0KaGVhZChwbGFuZXMsIDIpCmBgYAoKQWdhaW4sIHdlIG5lZWQgdG8gbWVyZ2UgdGhpcyBpbmZvcm1hdGlvbiB3aXRoIHRoZSBkYXRhIGluIHRoZSBgZmxpZ2h0c2AgdGFibGUuCgpPbmUgd2F5IHRvIGRvIHRoaXMgaXMgd2l0aCBhIF9qb2luIG9wZXJhdGlvbl8uCgpKb2lucyBhcmUgYSBjb21tb24gb3BlcmF0aW9uIGluIGRhdGEgYmFzZXMuCgoqIE11cmVsbCdzIF9EYXRhIFRlY2hub2xvZ2llc18gYm9vayBkZXNjcmliZXMgZGF0YSBiYXNlcyBpbgogIFNlY3Rpb24gNS42IGFuZCBkYXRhIGJhc2UgcXVlcmllcyB1c2luZyBTUUwgaW4gQ2hhcHRlciA3LgoKKiBfUiBmb3IgRGF0YSBTY2llbmNlXyBkZXNjcmliZXMgcmVsYXRpb25hbCBkYXRhIGluCiAgW0NoYXB0ZXIgMTNdKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovam9pbnMuaHRtbCkuCgpgZHBseXJgIGRpc3Rpbmd1aXNoZXMgYmV0d2VlbiB0d28ga2luZHMgb2Ygam9pbnM6CgoqIF9tdXRhdGluZyBqb2luc187CgoqIF9maWx0ZXJpbmcgam9pbnNfLgoKVGhlc2UgW2FuaW1hdGVkCmV4cGxhbmF0aW9uc10oaHR0cHM6Ly93d3cuZ2Fycmlja2FkZW5idWllLmNvbS9wcm9qZWN0L3RpZHlleHBsYWluLykgbWF5CmJlIGhlbHBmdWwuCgoKIyMjIE11dGF0aW5nIEpvaW5zCgpNdXRhdGluZyBqb2lucyBjb21iaW5lIHZhcmlhYmxlcyBhbmQgcm93cyBmcm9tIHR3byB0YWJsZXMgYnkgbWF0Y2hpbmcKcm93cyBvbiBfa2V5c18uCgoqIEEgX3ByaW1hcnkga2V5XyBpcyBhIHZhcmlhYmxlLCBvciBjb21iaW5hdGlvbiBvZiB2YXJpYWJsZXMsIGluIGEKICB0YWJsZSB0aGF0IHVuaXF1ZWx5IGlkZW50aWZ5IHRoZSByb3dzIG9mIHRoZSB0YWJsZS4KCiogQSBfZm9yZWlnbiBrZXlfIGlzIGEgdmFyaWFibGUsIG9yIGNvbWJpbmF0aW9uIG9mIHZhcmlhYmxlcywgdGhhdAogIHVuaXF1ZWx5IGlkZW50aWZ5IGEgcm93IGluIHNvbWUgdGFibGUgKHVzdWFsbHksIGJ1dCBub3QgbmVjZXNzYXJpbHksIGEKICBkaWZmZXJlbnQgdGFibGUpLgoKVGhlIHNpbXBsZXN0IGpvaW4gb3BlcmF0aW9uIGlzIGFuIF9pbm5lciBqb2luXzogT25seSByb3dzIGZvciB3aGljaAp0aGUga2V5IGFwcGVhcnMgaW4gYm90aCB0YWJsZXMgYXJlIHJldGFpbmVkLgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgb3V0LndpZHRoID0gIjcwJSJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKElNRygiam9pbi1pbm5lci5wbmciKSkKYGBgCgpfT3V0ZXIgam9pbnNfIHJldGFpbiBzb21lIG90aGVyIHJvd3Mgd2l0aCBgTkFgIHZhbHVlcyBmb3IgbmV3IHZhcmlhYmxlczoKCmBgYHtyLCBlY2hvID0gRkFMU0UsIG91dC53aWR0aCA9ICI3MCUifQprbml0cjo6aW5jbHVkZV9ncmFwaGljcyhJTUcoImpvaW4tb3V0ZXIucG5nIikpCmBgYAoKSW5uZXIgYW5kIGxlZnQgam9pbnMgYXJlIHRoZSBtb3N0IGNvbW1vbi4KClRoZSBrZXkgb2Ygb25lIG9mIHRoZSB0YWJsZXMgc2hvdWxkIHVzdWFsbHkgYmUgYSBfcHJpbWFyeSBrZXlfIHRoYXQKdW5pcXVlbHkgaWRlbnRpZmllcyB0aGUgdGFibGUncyByb3dzLgoKRm9yIGEgbGVmdCBqb2luLCB0aGUgcmlnaHQgdGFibGUga2V5IHNob3VsZCB1c3VhbGx5IHVzZSBhIHByaW1hcnkga2V5LgoKSXQgaXMgdXN1YWxseSBhIGdvb2QgaWRlYSB0byBtYWtlIHN1cmUgYSBwcmltYXJ5IGtleSBjYW5kaWRhdGUgcmVhbGx5CmRvZXMgdW5pcXVlbHkgaWRlbnRpZnkgcm93cy4KCkZvciB0aGUgYHBsYW5lc2AgdGFibGUgdGhlIGB0YWlsbnVtYCB2YXJpYWJsZSBpcyBhIHByaW1hcnkga2V5OwoKYGBge3J9CmNvdW50KHBsYW5lcywgdGFpbG51bSkgfD4gZmlsdGVyKG4gPiAxKQpgYGAKCkZvciB0aGUgYGZsaWdodHNgIGRhdGEgYSBwcmltYXJ5IGtleSBuZWVkcyB0byB1c2UgbXVsdGlwbGUgdmFyaWFibGVzLgoKQSByZWFzb25hYmxlIGd1ZXNzIGlzIHRvIHVzZSB0aGUgZGF0ZSwgY2FycmllciBhbmQgZmxpZ2h0IG51bWJlciwgYnV0CnRoaXMgaXNuJ3QgcXVpdGUgZW5vdWdoOgoKYGBge3J9CmNvdW50KGZsaWdodHMsIHllYXIsIG1vbnRoLCBkYXksIGZsaWdodCwgY2FycmllcikgfD4KICAgIGZpbHRlcihuID4gMSkgfD4KICAgIGhlYWQoMikKYGBgCgpBZGRpbmcgdGhlIGBvcmlnaW5gIGFpcnBvcnQgY29kZSBkb2VzIHByb2R1Y2UgYSBwcmltYXJ5IGtleToKCmBgYHtyfQpjb3VudChmbGlnaHRzLCB5ZWFyLCBtb250aCwgZGF5LCBmbGlnaHQsIGNhcnJpZXIsIG9yaWdpbikgfD4KICAgIGZpbHRlcihuID4gMSkKYGBgCgpUbyByZWxhdGUgZGVsYXlzIHRvIGFpcmNyYWZ0IGFnZSB3ZSBjYW4gc3RhcnQgd2l0aCBzdW1tYXJpZXMgYnkKYHRhaWxudW1gIGFuZCBgeWVhcmAgKGB5ZWFyYCB3b3VsZCBiZSBuZWVkZWQgaWYgd2UgaGFkIGRhdGEgZm9yIG1vcmUKdGhhbiAyMDEzKToKCmBgYHtyfQpmbCA8LSBmaWx0ZXIoZmxpZ2h0cywgISBpcy5uYShkZXBfdGltZSksICEgaXMubmEoYXJyX3RpbWUpKSB8PgogICAgZ3JvdXBfYnkoeWVhciwgdGFpbG51bSkgfD4KICAgIHN1bW1hcml6ZShhdmVfZGVwX2RlbGF5ID0gbWVhbihkZXBfZGVsYXksIG5hLnJtID0gVFJVRSksCiAgICAgICAgICAgICAgYXZlX2Fycl9kZWxheSA9IG1lYW4oYXJyX2RlbGF5LCBuYS5ybSA9IFRSVUUpLAogICAgICAgICAgICAgIG4gPSBuKCkpIHw+CiAgICB1bmdyb3VwKCkKYGBgCgpgYGB7cn0KaGVhZChmbCkKYGBgCgpBIHJlZHVjZWQgYHBsYW5lc2AgdGFibGUgd2l0aCBgeWVhcmAgcmVuYW1lZCB0byBhdm9pZCBhIGNvbmZsaWN0OgoKYGBge3J9CnBsIDwtIHNlbGVjdChwbGFuZXMsIHRhaWxudW0sIHBsYW5lX3llYXIgPSB5ZWFyKQpgYGAKCmBgYHtyfQpwbApgYGAKCk5vdwoKKiB1c2UgYSBsZWZ0IGpvaW4gd2l0aCBgdGFpbG51bWAgYXMgdGhlIGtleSB0byBicmluZyB0aGUgYHBsYW5lX3llYXJgCiAgdmFsdWUgaW50byB0aGUgc3VtbWFyeSB0YWJsZQoKKiBhbmQgdGhlbiBjb21wdXRlIHRoZSBwbGFuZSBhZ2U6CgpgYGB7cn0KZmxfYWdlIDwtIGxlZnRfam9pbihmbCwgcGwsICJ0YWlsbnVtIikgfD4KICAgIG11dGF0ZShwbGFuZV9hZ2UgPSB5ZWFyIC0gcGxhbmVfeWVhcikKZmxfYWdlCmBgYApgYGB7ciwgaW5jbHVkZSA9IEZBTFNFfQojIyBub3Qgc3VyZSBpZiB0aGlzIGlzIHN0aWxsIG5lZWRlZApmbCA8LSBmbF9hZ2UKYGBgCgpBbiBpbml0aWFsIHBsb3Q6CgpgYGB7ciBwbGFuZS1hZ2UtaW5pdGlhbCwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZmxfYWdlLAogICAgICAgYWVzKHggPSBwbGFuZV9hZ2UsCiAgICAgICAgICAgeSA9IGF2ZV9kZXBfZGVsYXkpKSArCiAgICBnZW9tX3BvaW50KCkgKwogICAgdGhtCmBgYAoKYGBge3IgcGxhbmUtYWdlLWluaXRpYWwsIGVjaG8gPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFfQpgYGAKCkppdHRlcmluZyBhbmQgYWRqdXN0aW5nIHRoZSBwb2ludCBzaXplIG1heSBoZWxwOgoKYGBge3IgcGxhbmUtYWdlLWppdHRlciwgZXZhbCA9IEZBTFNFfQpnZ3Bsb3QoZmxfYWdlLAogICAgICAgYWVzKHggPSBwbGFuZV9hZ2UsCiAgICAgICAgICAgeSA9IGF2ZV9kZXBfZGVsYXkpKSArCiAgICBnZW9tX3BvaW50KHBvc2l0aW9uID0gImppdHRlciIsCiAgICAgICAgICAgICAgIHNpemUgPSAwLjEpICsKICAgIHRobQpgYGAKCmBgYHtyIHBsYW5lLWFnZS1qaXR0ZXIsIGVjaG8gPSBGQUxTRSwgd2FybmluZyA9IEZBTFNFfQpgYGAKCkFkZGluZyBhIHNtb290aCBhbmQgYWRqdXN0aW5nIHRoZSByYW5nZXMgbWF5IGFsc28gaGVscDoKCmBgYHtyLCBwbGFuZS1hZ2Utc21vb3RoLCBldmFsID0gRkFMU0V9CmdncGxvdChmbF9hZ2UsCiAgICAgICBhZXMoeCA9IHBsYW5lX2FnZSwKICAgICAgICAgICB5ID0gYXZlX2RlcF9kZWxheSkpICsKICAgIGdlb21fcG9pbnQocG9zaXRpb24gPSAiaml0dGVyIiwKICAgICAgICAgICAgICAgc2l6ZSA9IDAuMSkgKwogICAgZ2VvbV9zbW9vdGgoKSArCiAgICB4bGltKDAsIDMwKSArCiAgICB5bGltKGMoLTIwLCA4MCkpICsKICAgIHRobQpgYGAKCmBgYHtyLCBwbGFuZS1hZ2Utc21vb3RoLCBlY2hvID0gRkFMU0UsIHdhcm5pbmcgPSBGQUxTRX0KYGBgCgpBbm90aGVyIG9wdGlvbiBpcyB0byBjb21wdXRlIG1lYW5zIG9yIG1lZGlhbnMgd2l0aGluIHllYXJzOgoKYGBge3IgcGxhbmUtYWdlLW1lYW5zLCBldmFsID0gRkFMU0V9CmZsbSA8LSBncm91cF9ieShmbF9hZ2UsIHBsYW5lX2FnZSkgfD4KICAgIHN1bW1hcml6ZShhdmVfZGVwX2RlbGF5ID0KICAgICAgICAgICAgICAgICAgbWVhbihhdmVfZGVwX2RlbGF5LAogICAgICAgICAgICAgICAgICAgICAgIG5hLnJtID0gVFJVRSkpIHw+CiAgICB1bmdyb3VwKCkKZ2dwbG90KGZsbSwKICAgICAgIGFlcyh4ID0gcGxhbmVfYWdlLAogICAgICAgICAgIHkgPSBhdmVfZGVwX2RlbGF5KSkgKwogICAgZ2VvbV9wb2ludChuYS5ybSA9IFRSVUUpICsKICAgIHRobQpgYGAKCmBgYHtyIHBsYW5lLWFnZS1tZWFucywgZWNobyA9IEZBTFNFfQpgYGAKCkJvdGggdGhlIHNtb290aCBhbmQgdGhlIG1lYW5zIHNob3cgYSBtb2Rlc3QgaW5jcmVhc2Ugb3ZlciB0aGUgZmlyc3QgMTAgeWVhcnMuCgpBIGxlZnQgam9pbiBjYW4gYWxzbyBiZSB1c2VkIHRvIGFkZCB0aGUgYWlycG9ydCBuYW1lIGZyb20gdGhlIGFpcnBvcnQKdGFibGUgdG8gdGhlIGBmbF9kZXN0YCB0YWJsZS4KClRoZSBwcmltYXJ5IGtleSBpbiB0aGUgYGFpcnBvcnRgIHRhYmxlIGlzIHRoZSB0aHJlZS1sZXR0ZXIgYWlycG9ydCBjb2RlCmluIHRoZSBgZmFhYCB2YXJpYWJsZToKCmBgYHtyfQpoZWFkKGFpcnBvcnRzLCAyKQpgYGAKClRoZSBmb3JlaWduIGtleSBpbiB0aGUgYGZsX2Rlc3RgIHRhYmxlIHRvIG1hdGNoIGlzIHRoZSBgZGVzdGAgdmFyaWFibGU6CgpgYGB7cn0KaGVhZChmbF9kZXN0LCAyKQpgYGAKClRoZSBgbGVmdF9qb2luYCBmdW5jdGlvbiBhbGxvd3MgdGhpcyBsaW5rIHRvIGJlIG1hZGUgYnkgc3BlY2lmeWluZyB0aGUKa2V5IGFzIGBjKCJkZXN0IiA9ICJmYWEiKWA6CgpgYGB7cn0KZmxfZGVzdCA8LSBsZWZ0X2pvaW4oZmxfZGVzdCwKICAgICAgICAgICAgICAgICAgICAgc2VsZWN0KGFpcnBvcnRzLCBmYWEsIGRlc3RfbmFtZSA9IG5hbWUpLAogICAgICAgICAgICAgICAgICAgICBjKCJkZXN0IiA9ICJmYWEiKSkKc2VsZWN0KGZsX2Rlc3QsIGRlc3RfbmFtZSwgZXZlcnl0aGluZygpKSB8PiBoZWFkKDIpCmBgYAoKTm93IGEgdG9vbHRpcCBjYW4gcHJvdmlkZSBhIG1vcmUgYWNjZXNzaWJsZSBsYWJlbDoKCmBgYHtyIGF2ZV9hcnJfZGVzdF9uYW1lLCBldmFsID0gRkFMU0V9CnAgPC0gZ2dwbG90KGZsX2Rlc3QpICsKICAgIGdlb21fcG9pbnQoYWVzKHggPSBuLAogICAgICAgICAgICAgICAgICAgeSA9IGF2ZV9hcnJfZGVsYXksCiAgICAgICAgICAgICAgICAgICB0ZXh0ID0gcGFzdGUoZGVzdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZXN0X25hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2VwID0gIjogIikpKSArCiAgICB0aG0KZ2dwbG90bHkocCwgdG9vbHRpcCA9ICJ0ZXh0IikKYGBgCgpgYGB7ciBhdmVfYXJyX2Rlc3RfbmFtZSwgZWNobyA9IEZBTFNFLCB3YXJuaW5nID0gRkFMU0V9CmBgYAoKCiMjIyBGaWx0ZXJpbmcgSm9pbnMKCkZpbHRlcmluZyBqb2lucyBvbmx5IGFmZmVjdCByb3dzOyB0aGV5IGRvIG5vdCBhZGQgdmFyaWFibGVzLgoKKiBgc2VtaV9qb2luKHgsIHkpYCByZXR1cm5zIGFsbCByb3dzIGluIGB4YCB3aXRoIGEgbWF0Y2ggaW4gYHlgOgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgb3V0LndpZHRoID0gIjcwJSJ9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKElNRygiam9pbi1zZW1pLnBuZyIpKQpgYGAKCiogYGFudGlfam9pbih4LCB5KWAgcmV0dXJucyBhbGwgcm93cyBpbiBgeGAgd2l0aG91dCBhIG1hdGNoIGluIGB5YDoKCmBgYHtyLCBlY2hvID0gRkFMU0UsIG91dC53aWR0aCA9ICI3MCUifQprbml0cjo6aW5jbHVkZV9ncmFwaGljcyhJTUcoImpvaW4tYW50aS5wbmciKSkKYGBgCgpgc2VtaV9qb2luYCBpcyB1c2VmdWwgZm9yIG1hdGNoaW5nIGZpbHRlcmVkIHJlc3VsdHMgYmFjayB0byBhIHRhYmxlLgoKVGhlIHBsb3Qgb2YgZGVsYXkgYWdhaW5zdCBhZ2Ugc2hvd2VkIHNvbWUgc3VycHJpc2luZ2x5IG9sZCBhaXJjcmFmdDoKCmBgYHtyfQp0b3BfYWdlIDwtIHNsaWNlX21heChmbF9hZ2UsIHBsYW5lX2FnZSwgbiA9IDIpCnRvcF9hZ2UKYGBgCgpXZSBjYW4gaWRlbnRpZnkgdGhlc2UgYWlyY3JhZnQgd2l0aCBhIHNlbWktam9pbjoKCmBgYHtyfQpzZW1pX2pvaW4ocGxhbmVzLCB0b3BfYWdlLCAidGFpbG51bSIpCmBgYAoKVGhlIGNvcnJlc3BvbmRpbmcgZmxpZ2h0cyBjYW4gYWxzbyBiZSBwaWNrZWQgb3V0IHdpdGggYSBzZW1pLWpvaW46CgpgYGB7cn0KdG9wX2FnZV9mbGlnaHRzIDwtIHNlbWlfam9pbihmbGlnaHRzLCB0b3BfYWdlLCAidGFpbG51bSIpCmhlYWQodG9wX2FnZV9mbGlnaHRzLCAyKQpgYGAKCkxvb2tpbmcgYXQgdGhlIGNhcnJpZXJzIG9yIGRlc3RpbmF0aW9ucyBtaWdodCBoZWxwOgoKYGBge3J9CnNlbGVjdCh0b3BfYWdlX2ZsaWdodHMsIGNhcnJpZXIsIGRlc3QsIGV2ZXJ5dGhpbmcoKSkgfD4gaGVhZCg0KQpgYGAKClRoZSBjYXJyaWVyIHN1bW1hcnk6CgpgYGB7cn0KY291bnQodG9wX2FnZV9mbGlnaHRzLCBjYXJyaWVyKQpgYGAKCkEgY2hlY2sgb2YgdGhlIGRvY3VtZW50YXRpb24gZm9yIGBwbGFuZXNgIHNob3dzIHRoYXQgQW1lcmljYW4gYW5kIG9uZQpvdGhlciBhaXJsaW5lIHJlcG9ydCBfZmxlZXQgbnVtYmVyc18gcmF0aGVyIHRoYW4gdGFpbCBudW1iZXJzLgoKQW50aS1qb2lucyBhcmUgdXNlZnVsIGZvciBkZXRlY3Rpbmcgam9pbiBwcm9ibGVtcy4KCkZvciBleGFtcGxlLCBtYW55IGZsaWdodHMgaGF2ZSB0YWlsIG51bWJlcnMgdGhhdCBkbyBub3QgbWF0Y2ggZW50cmllcwppbiBgcGxhbmVzYDoKCmBgYHtyfQpmbCA8LSBhbnRpX2pvaW4oZmxpZ2h0cywgcGxhbmVzLCAidGFpbG51bSIpCm5yb3coZmwpCmBgYAoKYGBge3J9CmNvdW50KGZsLCB0YWlsbnVtKQpgYGAKCgojIyMgSm9pbiBJc3N1ZXMKCiogTWFrZSBzdXJlIGEgcHJpbWFyeSBrZXkgeW91IHdhbnQgdG8gdXNlIGhhcyBubyBtaXNzaW5nIHZhbHVlcy4KCiogTWFrZSBzdXJlIGEgcHJpbWFyeSBrZXkgeW91IHdhbnQgdG8gdXNlIHJlYWxseSB1bmlxdWVseSBpZGVudGlmaWVzCiAgYSByb3cuCgoqIE1ha2Ugc3VyZSBldmVyeSB2YWx1ZSBvZiB5b3VyIGZvcmVpZ24ga2V5IHlvdSB3YW50IHRvIHVzZSB0byBsaW5rIHRvCiAgYW5vdGhlciB0YWJsZSBtYXRjaGVzIGEgcm93IGluIHRoZSBsaW5rZWQgdGFibGUgKGBhbnRpX2pvaW5gIGlzIGEKICBnb29kIHdheSB0byBkbyB0aGlzKS4KCiogQ2hlY2sgdGhhdCB5b3VyIGZvcmVpZ24ga2V5IGhhcyBubyBtaXNzaW5nIHZhbHVlcy4KCjwhLS0KLSBtb3JlIGV4YW1wbGVzCiAgICAtIG1heWJlIHNlcGFyYXRlIHNlY3Rpb24KICAgIC0gd2l0aCBwbG90cwotIGpvaW5zCiAgICAtIG1vdmUgb3ZlciBzb21lIG9mIHRoZSBsYXRlciBzdHVmZgogICAgLSBvciBtYXliZSBzZXBhcmF0ZSBkb2N1bWVudAotLT4KCgojIyBEYXRhIEJhc2UgSW50ZXJmYWNlCgpDdXJyZW50IGRhdGEgb24gYWlybGluZSBmbGlnaHRzIGluIHRoZSBVUyBpcyBhdmFpbGFibGUgZnJvbSB0aGUgQnVyZWF1Cm9mIFRyYW5zcG9ydGF0aW9uIFN0YXRpc3RpY3MgYXQKCjxodHRwOi8vd3d3LnRyYW5zdGF0cy5idHMuZ292L0RMX1NlbGVjdEZpZWxkcy5hc3A/VGFibGVfSUQ9MjM2Pgo8IS0taHR0cHM6Ly90cmFuc3RhdHMuYnRzLmdvdi9QUkVaSVAvX09uX1RpbWVfUGVyZm9ybWFuY2VfMjAxN18xLnppcCAtLT4KClRoZSBkYXRhIGFyZSBhdmFpbGFibGUgaW4gb25lIGNvbXByZXNzZWQgQ1NWIGZpbGUgcGVyIG1vbnRoLgoKTG9hZGVkIGludG8gUiwgb25lIG1vbnRoIG9mIGRhdGEgcmVxdWlyZXMgYWJvdXQgMzAwIE1iIG9mIG1lbW9yeS4KCkRhdGEgZm9yIDE5ODctLTIwMDggd2FzIGNvbXBpbGVkIGZvciBhIERhdGFFeHBvIGNvbXBldGl0aW9uIG9mIHRoZQpfQW1lcmljYW4gU3RhdGlzdGljYWwgQXNzb2NpYXRpb25fIGhlbGQgaW4gMjAwOS4KClRoZSBkYXRhIGFyZSBhdmFpbGFibGUgb24gdGhlIExpbnV4IHN5c3RlbXMgaW4gdGhlIGRpcmVjdG9yeQpgL2dyb3VwL3N0YXRzb2Z0L2RhdGEvRGF0YUV4cG8wOWAgYXMgY29tcHJlc3NlZCBDU1YgZmlsZXMgYW5kIGFzIGEKU1FMaXRlIGRhdGEgYmFzZS4KCkxvYWRpbmcgdGhlIGRhdGEgaW50byBSIHdvdWxkIHJlcXVpcmUgYWJvdXQgNzAgR2Igb2YgbWVtb3J5LCB3aGljaCBpcwpub3QgcHJhY3RpY2FsLgoKVXNpbmcgdGhlIGRhdGEgYmFzZSBpcyBhIGdvb2QgYWx0ZXJuYXRpdmUuCgpTUUxpdGUgaXMgYSBzaW1wbGUgcmVsYXRpb25hbCBkYXRhIGJhc2UgdGhhdCB3b3JrcyB3aXRoIGRhdGEgc3RvcmVkIG9uCmEgbG9jYWwgZGlzay4KCk1vcmUgc29waGlzdGljYXRlZCBkYXRhIGJhc2VzIGFsbG93IGRhdGEgdG8gYmUgc3RvcmVkIG9uIG11bHRpcGxlCm1hY2hpbmVzIG9yIGluIHRoZSBjbG91ZAoKVGhlc2UgZGF0YSBiYXNlcyBhcmUgdXN1YWxseSBhY2Nlc3NlZCBieSBjb25uZWN0aW5nIHRvIGEgc2VydmVyLAp3aGljaCB1c3VhbGx5IGludm9sdmVzIGFuIGF1dGhlbnRpY2F0aW9uIHByb2Nlc3MuCgpGb3IgU1FMaXRlIHRoZSBjb25uZWN0aW9uIGlzIHNldCB1cCBieSBsaW5raW5nIHRvIGEgZmlsZS4KClRoZSBSU1FMaXRlIHBhY2thZ2UgcHJvdmlkZXMgYW4gaW50ZXJmYWNlIHRvIFNRTGl0ZS4KClRoZSBgREJJYCBwYWNrYWdlIHByb3ZpZGVzIGEgaGlnaCBsZXZlbCBpbnRlcmZhY2UgZm9yIHVzaW5nIGRhdGEgYmFzZXMuCgoqIGBkYkNvbm5lY3RgIGNyZWF0ZXMgYSBnZW5lcmljIGRhdGEgYmFzZSBjb25uZWN0aW9uIG9iamVjdC4KKiBgZGJMaXN0VGFibGVzYCByZXR1cm5zIGEgdmVjdG9yIHdpdGggdGhlIG5hbWVzIG9mIHRoZSBhdmFpbGFibGUgdGFibGVzLgoqIGBkYkRpc2Nvbm5lY3RgIGNsb3NlcyBhIGRhdGEgYmFzZSBjb25uZWN0aW9uLgoKQW5hbG9nb3VzIHBhY2thZ2VzIGV4aXN0IGZvciBQeXRob24gYW5kIFBlcmwuCgpUaGUgZmlyc3Qgc3RlcCBpcyB0byBjb25uZWN0IHRvIHRoZSBkYXRhIGJhc2UuCgo8IS0tICMgbm9saW50IHN0YXJ0IC0tPgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpsaWJyYXJ5KGRwbHlyKQpsaWJyYXJ5KERCSSkKbGlicmFyeShSU1FMaXRlKQpkYiA8LSBkYkNvbm5lY3QoU1FMaXRlKCksICIvZ3JvdXAvc3RhdHNvZnQvZGF0YS9EYXRhRXhwbzA5L29udGltZS5zcWxpdGUzIikKYGBgCjwhLS0gIyBub2xpbnQgZW5kIC0tPgoKVGhlcmUgaXMgb25seSBvbmUgdGFibGU6CmBgYHtyLCBldmFsID0gRkFMU0V9CmRiTGlzdFRhYmxlcyhkYikKIyMgWzFdICJvbnRpbWUiCmBgYAoKYHRibGAgY3JlYXRlcyBhIHRpYmJsZS1saWtlIG9iamVjdCBmb3IgYSBkYXRhIGJhc2UgdGFibGUuCgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpmbCA8LSB0YmwoZGIsICJvbnRpbWUiKQpgYGAKClRoaXMgb2JqZWN0IGlzIG5vdCBhIHJlZ3VsYXIgdGliYmxlLCBidXQgbWFueSB0aWJibGUgb3BlcmF0aW9ucyB3b3JrIG9uIGl0LgoKVG8gZmluZCB0aGUgdmFyaWFibGVzIGluIHRoZSB0YWJsZToKCmBgYHtyLCBldmFsID0gRkFMU0V9CnRibF92YXJzKGZsKQojIyAgWzFdICJZZWFyIiAgICAgICAgICAgICAgIk1vbnRoIiAgICAgICAgICAgICAiRGF5b2ZNb250aCIKIyMgIFs0XSAiRGF5T2ZXZWVrIiAgICAgICAgICJEZXBUaW1lIiAgICAgICAgICAgIkNSU0RlcFRpbWUiCiMjICBbN10gIkFyclRpbWUiICAgICAgICAgICAiQ1JTQXJyVGltZSIgICAgICAgICJVbmlxdWVDYXJyaWVyIgojIyBbMTBdICJGbGlnaHROdW0iICAgICAgICAgIlRhaWxOdW0iICAgICAgICAgICAiQWN0dWFsRWxhcHNlZFRpbWUiCiMjIFsxM10gIkNSU0VsYXBzZWRUaW1lIiAgICAiQWlyVGltZSIgICAgICAgICAgICJBcnJEZWxheSIKIyMgWzE2XSAiRGVwRGVsYXkiICAgICAgICAgICJPcmlnaW4iICAgICAgICAgICAgIkRlc3QiCiMjIFsxOV0gIkRpc3RhbmNlIiAgICAgICAgICAiVGF4aUluIiAgICAgICAgICAgICJUYXhpT3V0IgojIyBbMjJdICJDYW5jZWxsZWQiICAgICAgICAgIkNhbmNlbGxhdGlvbkNvZGUiICAiRGl2ZXJ0ZWQiCiMjIFsyNV0gIkNhcnJpZXJEZWxheSIgICAgICAiV2VhdGhlckRlbGF5IiAgICAgICJOQVNEZWxheSIKIyMgWzI4XSAiU2VjdXJpdHlEZWxheSIgICAgICJMYXRlQWlyY3JhZnREZWxheSIKYGBgCgpNYW55IGBkcGx5cmAgb3BlcmF0aW9ucyBhcmUgc3VwcG9ydGVkOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0Kc20gPC0gc3VtbWFyaXplKGZsLCBsYXN0X3llYXIgPSBtYXgoWWVhciwgbmEucm0gPSBUUlVFKSwgbiA9IG4oKSkKYGBgCgpUaGlzIGNvbXB1dGF0aW9uIGlzIGZhc3QgYmVjYXVzZSBpdCBkb2VzIG5vdCBhY3R1YWxseSBhY2Nlc3MKdGhlIGRhdGEgYmFzZS4KCkFuIG9wZXJhdGlvbiB0aGF0IG5lZWRzIHRoZSBkYXRhLCBzdWNoIGFzIHByaW50aW5nIHRoZSByZXN1bHQsIHdpbGwKYWNjZXNzIHRoZSBkYXRhIGJhc2UgYW5kIHRha2UgdGltZToKCmBgYHtyLCBldmFsID0gRkFMU0V9CnNtCiMjICMgU291cmNlOiAgIGxhenkgcXVlcnkgWz8/IHggMl0KIyMgIyBEYXRhYmFzZTogc3FsaXRlIDMuMzcuMgojIyAjICAgWy9tbnQvbmZzL2NsYXNuZXRhcHB2bS9ncm91cC9zdGF0c29mdC9kYXRhL0RhdGFFeHBvMDkvb250aW1lLnNxbGl0ZTNdCiMjICAgbGFzdF95ZWFyICAgICAgICAgbgojIyAgICAgICA8aW50PiAgICAgPGludD4KIyMgMSAgICAgIDIwMDggMTE4OTE0NDU4CmBgYAoKVGhlIGBzbWAgb2JqZWN0IGVzc2VudGlhbGx5IGNvbnRhaW5zIGEgU1FMIHF1ZXJ5LApbd2hpY2ggY2FuIGJlCmV4YW1pbmVkXShodHRwczovL2RicGx5ci50aWR5dmVyc2Uub3JnL2FydGljbGVzL3NxbC10cmFuc2xhdGlvbi5odG1sKQp1c2luZyBgc3FsX3JlbmRlcmAgZnJvbSB0aGUgYGRicGx5cmAgcGFja2FnZToKCmBgYHtyLCBldmFsID0gRkFMU0V9CmRicGx5cjo6c3FsX3JlbmRlcihzbSkKIyMgPFNRTD4gU0VMRUNUIE1BWChgWWVhcmApIEFTIGBsYXN0X3llYXJgLCBDT1VOVCgqKSBBUyBgbmAKIyMgRlJPTSBgb250aW1lYApgYGAKCkZsaWdodHMgdG8gQ0lEIGluIDIwMDg6CgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpmbF9jaWQgPC0gZmlsdGVyKGZsLCBZZWFyID09IDIwMDgsIERlc3QgPT0gIkNJRCIpCmRicGx5cjo6c3FsX3JlbmRlcihmbF9jaWQpCiMjIDxTUUw+IFNFTEVDVCAqCiMjIEZST00gYG9udGltZWAKIyMgV0hFUkUgKChgWWVhcmAgPSAyMDA4LjApIEFORCAoYERlc3RgID0gJ0NJRCcpKQpgYGAKCk51bWJlciBvZiBmbGlnaHRzIHRvIENJRCBpbiAyMDA4OgoKYGBge3IsIGV2YWwgPSBGQUxTRX0Kc20gPC0gc3VtbWFyaXplKGZsX2NpZCwgbiA9IG4oKSkKZGJwbHlyOjpzcWxfcmVuZGVyKHNtKQojIyA8U1FMPiBTRUxFQ1QgQ09VTlQoKikgQVMgYG5gCiMjIEZST00gYG9udGltZWAKIyMgV0hFUkUgKChgWWVhcmAgPSAyMDA4LjApIEFORCAoYERlc3RgID0gJ0NJRCcpKQpgYGAKCmBgYHtyLCBldmFsID0gRkFMU0V9CnNtCiMjICMgU291cmNlOiAgIGxhenkgcXVlcnkgWz8/IHggMV0KIyMgIyBEYXRhYmFzZTogc3FsaXRlIDMuMzcuMgojIyAjICAgWy9tbnQvbmZzL2NsYXNuZXRhcHB2bS9ncm91cC9zdGF0c29mdC9kYXRhL0RhdGFFeHBvMDkvb250aW1lLnNxbGl0ZTNdCiMjICAgICAgIG4KIyMgICA8aW50PgojIyAxICAyOTYxCmBgYAoKQnJlYWtkb3duIGJ5IHdoZXJlIHRoZSBmbGlnaHRzIG9yaWdpbmF0ZWQ6CgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpzbSA8LSBncm91cF9ieShmbF9jaWQsIE9yaWdpbikgfD4KICAgIHN1bW1hcml6ZShuID0gbigpKSB8PgogICAgdW5ncm91cCgpCmRicGx5cjo6c3FsX3JlbmRlcihzbSkKIyMgPFNRTD4gU0VMRUNUIGBPcmlnaW5gLCBDT1VOVCgqKSBBUyBgbmAKIyMgRlJPTSBgb250aW1lYAojIyBXSEVSRSAoKGBZZWFyYCA9IDIwMDguMCkgQU5EIChgRGVzdGAgPSAnQ0lEJykpCiMjIEdST1VQIEJZIGBPcmlnaW5gCmBgYAoKYGBge3IsIGV2YWwgPSBGQUxTRX0Kc20KIyMgIyBTb3VyY2U6ICAgbGF6eSBxdWVyeSBbPz8geCAyXQojIyAjIERhdGFiYXNlOiBzcWxpdGUgMy4zNy4yCiMjICMgICBbL21udC9uZnMvY2xhc25ldGFwcHZtL2dyb3VwL3N0YXRzb2Z0L2RhdGEvRGF0YUV4cG8wOS9vbnRpbWUuc3FsaXRlM10KIyMgICBPcmlnaW4gICAgIG4KIyMgICA8Y2hyPiAgPGludD4KIyMgMSBBVEwgICAgICAxMTkKIyMgMiBCTkEgICAgICAgIDEKIyMgMyBDVkcgICAgICAyOTIKIyMgNCBERU4gICAgICAyMzAKIyMgNSBERlcgICAgICA1MTkKIyMgNiBEVFcgICAgICAzNDUKIyMgNyBNU1AgICAgICAyNDEKIyMgOCBPUkQgICAgIDEyMTQKYGBgCgpNYW55IGJhc2UgUiBmdW5jdGlvbnMgY2FuIGJlIGNvbnZlcnRlZCB0byBTUUw6CgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpzbSA8LSBncm91cF9ieShmbF9jaWQsIE9yaWdpbikgfD4KICAgIHN1bW1hcml6ZShhdmVfYXJyX2RlbF9wb3MgPSBtZWFuKHBtYXgoQXJyRGVsYXksIDApLCBuYS5ybSA9IFRSVUUpKSB8PgogICAgdW5ncm91cCgpCmRicGx5cjo6c3FsX3JlbmRlcihzbSkKIyMgPFNRTD4gU0VMRUNUIGBPcmlnaW5gLCBBVkcoTUFYKGBBcnJEZWxheWAsIDAuMCkpIEFTIGBhdmVfYXJyX2RlbF9wb3NgCiMjIEZST00gYG9udGltZWAKIyMgV0hFUkUgKChgWWVhcmAgPSAyMDA4LjApIEFORCAoYERlc3RgID0gJ0NJRCcpKQojIyBHUk9VUCBCWSBgT3JpZ2luYApgYGAKCmBgYHtyLCBldmFsID0gRkFMU0V9CnNtX2NpZCA8LSBzbQpzbQojIyAjIFNvdXJjZTogICBsYXp5IHF1ZXJ5IFs/PyB4IDJdCiMjICMgRGF0YWJhc2U6IHNxbGl0ZSAzLjM3LjIKIyMgIyAgIFsvbW50L25mcy9jbGFzbmV0YXBwdm0vZ3JvdXAvc3RhdHNvZnQvZGF0YS9EYXRhRXhwbzA5L29udGltZS5zcWxpdGUzXQojIyAgIE9yaWdpbiBhdmVfYXJyX2RlbF9wb3MKIyMgICA8Y2hyPiAgICAgICAgICAgIDxkYmw+CiMjIDEgQVRMICAgICAgICAgICAgICAgMTcuNAojIyAyIEJOQSAgICAgICAgICAgICAgIDE4CiMjIDMgQ1ZHICAgICAgICAgICAgICAgMTIuNAojIyA0IERFTiAgICAgICAgICAgICAgIDEyLjEKIyMgNSBERlcgICAgICAgICAgICAgICAxMS4zCiMjIDYgRFRXICAgICAgICAgICAgICAgMTUuMAojIyA3IE1TUCAgICAgICAgICAgICAgIDIwLjEKIyMgOCBPUkQgICAgICAgICAgICAgICAyMS42CmBgYAoKYG11dGF0ZSgpYCBjYW4gYWxzbyBiZSB1c2VkOgoKPCEtLSAjIyBub2xpbnQgc3RhcnQ6IGxpbmVfbGVuZ3RoIC0tPgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpmbF9jaWRfZ2FpbiA8LSBtdXRhdGUoZmxfY2lkLCBHYWluID0gRGVwRGVsYXkgLSBBcnJEZWxheSkKZGJwbHlyOjpzcWxfcmVuZGVyKGZsX2NpZF9nYWluKQojIyA8U1FMPiBTRUxFQ1QgYFllYXJgLCBgTW9udGhgLCBgRGF5b2ZNb250aGAsIGBEYXlPZldlZWtgLCBgRGVwVGltZWAsIGBDUlNEZXBUaW1lYCwgYEFyclRpbWVgLCBgQ1JTQXJyVGltZWAsIGBVbmlxdWVDYXJyaWVyYCwgYEZsaWdodE51bWAsIGBUYWlsTnVtYCwgYEFjdHVhbEVsYXBzZWRUaW1lYCwgYENSU0VsYXBzZWRUaW1lYCwgYEFpclRpbWVgLCBgQXJyRGVsYXlgLCBgRGVwRGVsYXlgLCBgT3JpZ2luYCwgYERlc3RgLCBgRGlzdGFuY2VgLCBgVGF4aUluYCwgYFRheGlPdXRgLCBgQ2FuY2VsbGVkYCwgYENhbmNlbGxhdGlvbkNvZGVgLCBgRGl2ZXJ0ZWRgLCBgQ2FycmllckRlbGF5YCwgYFdlYXRoZXJEZWxheWAsIGBOQVNEZWxheWAsIGBTZWN1cml0eURlbGF5YCwgYExhdGVBaXJjcmFmdERlbGF5YCwgYERlcERlbGF5YCAtIGBBcnJEZWxheWAgQVMgYEdhaW5gCiMjIEZST00gYG9udGltZWAKIyMgV0hFUkUgKChgWWVhcmAgPSAyMDA4LjApIEFORCAoYERlc3RgID0gJ0NJRCcpKQpgYGAKPCEtLSAjIyBub2xpbnQgZW5kIC0tPgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ3JvdXBfYnkoZmxfY2lkX2dhaW4sIE9yaWdpbikgfD4KICAgIHN1bW1hcml6ZShtZWFuX2dhaW4gPSBtZWFuKEdhaW4pKSB8PgogICAgdW5ncm91cCgpCiMjICMgU291cmNlOiAgIGxhenkgcXVlcnkgWz8/IHggMl0KIyMgIyBEYXRhYmFzZTogc3FsaXRlIDMuMzcuMgojIyAjICAgWy9tbnQvbmZzL2NsYXNuZXRhcHB2bS9ncm91cC9zdGF0c29mdC9kYXRhL0RhdGFFeHBvMDkvb250aW1lLnNxbGl0ZTNdCiMjICAgT3JpZ2luIG1lYW5fZ2FpbgojIyAgIDxjaHI+ICAgICAgPGRibD4KIyMgMSBBVEwgICAgICAgIDguMjQKIyMgMiBCTkEgICAgICAtMjYKIyMgMyBDVkcgICAgICAgLTEuNjAKIyMgNCBERU4gICAgICAgIDAuMjc4CiMjIDUgREZXICAgICAgICA0LjEyCiMjIDYgRFRXICAgICAgICAzLjA0CiMjIDcgTVNQICAgICAgIC0wLjM1NwojIyA4IE9SRCAgICAgICAtMS4zNgpgYGAKCk9uY2UgYSBmaWx0ZXJlZCByZXN1bHQgaXMgc21hbGwgZW5vdWdoIGl0IGNhbiBiZSBicm91Z2h0IG92ZXIgYXMgYQpyZWd1bGFyIHRpYmJsZSB1c2luZwoKYGBge3IsIGV2YWwgPSBGQUxTRX0KdGIgPC0gY29sbGVjdChmbF9jaWQpCmBgYAoKVGhlcmUgYXJlIHNvbWUgbGltaXRhdGlvbnMgdG8gb3BlcmF0aW9ucyB0aGF0IGNhbiBiZSBkb25lIG9uIHRhYmxlcyBpbgpkYXRhIGJhc2VzLgoKRGVmaW5pbmcgYW5kIHVzaW5nIGEgZnVuY3Rpb24gd29ya3MgZm9yIGEgdGliYmxlOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KcG9zX21lYW4gPC0gZnVuY3Rpb24oeCkgbWVhbihwbWF4KHgsIDApKQpncm91cF9ieSh0YiwgT3JpZ2luKSB8PgogICAgc3VtbWFyaXplKGF2ZV9hcnJfZGVsX3BvcyA9IHBvc19tZWFuKEFyckRlbGF5KSkgfD4KICAgIHVuZ3JvdXAoKQpgYGAKCkJ1dCB0aGlzIGZhaWxzOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZ3JvdXBfYnkoZmxfY2lkLCBPcmlnaW4pIHw+CiAgICBzdW1tYXJpemUoYXZlX2Fycl9kZWxfcG9zID0gcG9zX21lYW4oQXJyRGVsYXkpKSB8PgogICAgdW5ncm91cCgpCiMjIEVycm9yOiBubyBzdWNoIGZ1bmN0aW9uOiBwb3NfbWVhbgpgYGAKClNvbWUgb3BlcmF0aW9ucyBtYXkgcmVxdWlyZSBzb21lIG1vZGlmaWNhdGlvbnMuCgpSZWNvbXB1dGluZyBgc21fY2lkYDoKCmBgYHtyLCBldmFsID0gRkFMU0V9CmZsIDwtIHRibChkYiwgIm9udGltZSIpCmZsX2NpZCA8LSBmaWx0ZXIoZmwsIFllYXIgPT0gMjAwOCwgRGVzdCA9PSAiQ0lEIikKc21fY2lkIDwtIGdyb3VwX2J5KGZsX2NpZCwgT3JpZ2luKSB8PgogICAgc3VtbWFyaXplKGF2ZV9hcnJfZGVsX3BvcyA9IG1lYW4ocG1heChBcnJEZWxheSwgMCksIG5hLnJtID0gVFJVRSkpIHw+CiAgICB1bmdyb3VwKCkKYGBgCgpKb2luaW5nIHdpdGggYSBsb2NhbCB0YWJsZSBkb2VzIG5vdCB3b3JrOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KYXAgPC0gc2VsZWN0KG55Y2ZsaWdodHMxMzo6YWlycG9ydHMsIGZhYSwgbmFtZSkKc21fY2lkIDwtIGxlZnRfam9pbihzbV9jaWQsIGFwLCBjKE9yaWdpbiA9ICJmYWEiKSkKIyMgRXJyb3IgaW4gYGF1dG9fY29weSgpYDoKIyMgISBgeGAgYW5kIGB5YCBtdXN0IHNoYXJlIHRoZSBzYW1lIHNyYy4KIyMg4oS5IHNldCBgY29weWAgPSBUUlVFIChtYXkgYmUgc2xvdykuCmBgYAoKQnV0IHRoaXMgZG9lcyB3b3JrOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0Kc21fY2lkIDwtIGxlZnRfam9pbihzbV9jaWQsIGFwLCBjKE9yaWdpbiA9ICJmYWEiKSwgY29weSA9IFRSVUUpCmBgYAoKVGhpcyBjb3BpZXMgdGhlIGBhcGAgdGFibGUgdG8gdGhlIGRhdGEgYmFzZToKCmBgYHtyLCBldmFsID0gRkFMU0V9CmRiTGlzdFRhYmxlcyhkYikKIyMgWzFdICJkYnBseXJfMDAxIiAgICJvbnRpbWUiICAgICAgICJzcWxpdGVfc3RhdDEiICJzcWxpdGVfc3RhdDQiCmBgYAoKRmluaXNoIGJ5IGRpc2Nvbm5lY3RpbmcgZnJvbSB0aGUgZGF0YSBiYXNlOgoKYGBge3IsIGV2YWwgPSBGQUxTRX0KZGJEaXNjb25uZWN0KGRiKQpgYGAKClRoaXMgaXMgbm90IG5lY2Vzc2FyeSBmb3IgU1FMaXRlIGJ1dCBpcyBmb3Igb3RoZXIgZGF0YSBiYXNlcywgc28gaXQgaXMKZ29vZCBwcmFjdGljZS4KClNvbWUgbm90ZXM6CgoqIEJ1aWxkaW5nIGRhdGEgYmFzZSBxdWVyaWVzIHdpdGggYGRwbHlyYCBpcyB2ZXJ5IGVmZmljaWVudC4KCiogUnVubmluZyB0aGUgcXVlcmllcyBjYW4gdGFrZSB0aW1lLgoKKiBUcmFuc2ZlcnJpbmcgKHBhcnRpYWwpIHJlc3VsdHMgdG8gUiBjYW4gdGFrZSB0aW1lLgoKKiBJbmFkdmVydGVudGx5IGFza2luZyB0byB0cmFuc2ZlciBhIHJlc3VsdCB0aGF0IGlzIHRvbyBsYXJnZQogIGNvdWxkIGNhdXNlIHByb2JsZW1zLgoKKiBPdGhlciBkYXRhIGJhc2VzIG1heSBhcnJhbmdlIHRvIGNhY2hlIHF1ZXJ5IHJlc3VsdHMuCgoqIE1vc3QgZGF0YSBiYXNlcywgaW5jbHVkaW5nIFNRTGl0ZSwgc3VwcG9ydCBjcmVhdGluZyBfaW5kaWNlc18gdGhhdAogIHNwZWVkIHVwIGNlcnRhaW4gcXVlcmllcyBhdCB0aGUgZXhwZW5zZSBvZiB1c2luZyBtb3JlIHN0b3JhZ2UuCgpUaGUgW2BkYnBseXJgIHdlYiBzaXRlXShodHRwczovL2RicGx5ci50aWR5dmVyc2Uub3JnL2luZGV4Lmh0bWwpCnByb3ZpZGVzIG1vcmUgaW5mb3JtYXRpb24gb24gdXNpbmcgZGF0YSBiYXNlcyB3aXRoIGBkcGx5cmAuCgpTb21lIG90aGVyIGFwcHJvYWNoZXMgZm9yIGhhbmRsaW5nIGxhcmdlci10aGFuLW1lbW9yeSBkYXRhOgoKLSBbQXBhY2hlIGFycm93XShodHRwczovL2Fycm93LmFwYWNoZS5vcmcvKSBhbmQgdGhlIFIKICBbYGFycm93YF0oaHR0cHM6Ly9hcnJvdy5hcGFjaGUub3JnL2RvY3Mvci9pbmRleC5odG1sKSBwYWNrYWdlLgoKLSBbUG9sYXJzXShodHRwczovL3BvbGEucnMvKSBhbmQgdGhlCiAgW2Bwb2xhcnNgXShodHRwczovL3Jwb2xhcnMuZ2l0aHViLmlvLykgYW5kCiAgW2B0aWR5cG9sYXJzYF0oaHR0cHM6Ly90aWR5cG9sYXJzLmV0aWVubmViYWNoZXIuY29tLykgUiBwYWNrYWdlcy4KCi0gW0R1Y2tEQl0oaHR0cHM6Ly9kdWNrZGIub3JnLykgd2l0aCBhCiAgW2d1aWRlXShodHRwczovL3JvYm90d2VhbHRoLmNvbS9hLWJlZ2lubmVycy1ndWlkZS10by11c2luZy1kdWNrZGItd2l0aC1zdG9jay1wcmljZS1kYXRhLWluLXIvKQogIGZvciB1c2luZyBmcm9tIFIuCgoKIyMgT3RoZXIgV3JhbmdsaW5nIFRvb2xzCgpTb21lIGFkZGl0aW9uYWwgdG9vbHMgcHJvdmlkZWQgYnkgYHRpZHlyYDoKCiogYHNlcGFyYXRlKClgOiBzcGxpdCBjaGFyYWN0ZXIgdmFyaWFibGUgaW50byBzZXZlcmFsIHZhcmlhYmxlczsKCiogYGZpbGwoKWA6IGZpbGwgaW4gbWlzc2luZyB2YWx1ZXMgaW4gYSB2YXJpYWJsZSAoZS5nLiBjYXJyeSBmb3J3YXJkKTsKCiogYGNvbXBsZXRlKClgOiBhZGQgbWlzc2luZyBjYXRlZ29yeSBjbGFzc2VzIChlLmcuIHplcm8gY291bnRzKS4KCldlIGhhdmUgYWxyZWFkeSBzZWVuIGBwaXZvdF9sb25nZXIoKWAgYW5kIGBwaXZvdF93aWRlcigpYC4KCkEgbmV3ZXIgYWx0ZXJuYXRpdmUgdG8gYHNlcGFyYXRlKClgIGlzIHRvIHVzZQpgc2VwYXJhdGVfd2lkZXJfcG9zaXRpb24oKWAgYW5kIGBzZXBhcmF0ZV93aWRlcl9kZWxpbSgpYAoKCiMjIyBTZXBhcmF0aW5nIFZhcmlhYmxlcwoKQSBkYXRhIHNldCBmcm9tIHRoZSBXSE8gb24gdHViZXJjdWxvc2lzIGNhc2VzOgoKYGBge3J9CmhlYWQod2hvLCAyKQpgYGAKClNvbWUgb2YgdGhlIHJlbWFpbmluZyB2YXJpYWJsZSBuYW1lczoKCmBgYHtyfQpuYW1lcyh3aG8pIHw+IHRhaWwoMjApCmBgYAoKVGhlIHZhcmlhYmxlIGBuZXdfc3BfbTI1MzRgLCBmb3IgZXhhbXBsZSwgcmVwcmVzZW50cyB0aGUgbnVtYmVyIG9mIGNhc2VzIGZvcgoKKiB0eXBlIGBzcGAgKHBvc2l0aXZlIHB1bG1vbmFyeSBzbWVhcik7Ciogc2V4IGBtYDsKKiBhZ2UgYmV0d2VlbiAyNSBhbmQgMzQuCgpJdCB3b3VsZCBiZSBiZXR0ZXIgKHRpZGllcikgdG8gaGF2ZSBzZXBhcmF0ZSB2YXJpYWJsZXMgZm9yCgoqIGNvdW50CiogdHlwZQoqIHNleAoqIGxvd2VyIGJvdW5kIG9uIGFnZSBicmFja2V0CiogdXBwZXIgYm91bmQgb24gYWdlIGJyYWNrZXQKCkEgcGlwZWxpbmUgdG8gY2xlYW4gdGhpcyB1cCBpbnZvbHZlcyBwaXZvdGluZyB0byBsb25nIGZvcm0gYW5kIHNldmVyYWwKYHNlcGFyYXRlYCBhbmQgYG11dGF0ZWAgc3RlcHMuCgpUaGUgb3JpZ2luYWwgZGF0YSBmcmFtZToKCmBgYHtyfQp3aG8KYGBgCgpGaXJzdCBwaXZvdCB0byBsb25nZXIgZm9ybToKCmBgYHtyfQp3aG9fY2xlYW4gPC0KICAgIHBpdm90X2xvbmdlcih3aG8sCiAgICAgICAgICAgICAgICAgbmV3X3NwX20wMTQgOiBuZXdyZWxfZjY1LAogICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gImtleSIsCiAgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gImNvdW50IikKYGBgCmBgYHtyfQp3aG9fY2xlYW4KYGBgCgpSZW1vdmUgYCJuZXciYCBvciBgIm5ld18iYCBwcmVmaXg6CgpgYGB7cn0Kd2hvX2NsZWFuIDwtCiAgICBwaXZvdF9sb25nZXIod2hvLAogICAgICAgICAgICAgICAgIG5ld19zcF9tMDE0IDogbmV3cmVsX2Y2NSwKICAgICAgICAgICAgICAgICBuYW1lc190byA9ICJrZXkiLAogICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJjb3VudCIpIHw+CiAgICBtdXRhdGUoa2V5ID0gc3ViKCJuZXdfPyIsICIiLCBrZXkpKQpgYGAKYGBge3J9Cndob19jbGVhbgpgYGAKClNlcGFyYXRlIGBrZXlgIGludG8gYHR5cGVgIGFuZCBgc2V4YWdlYDoKCmBgYHtyfQp3aG9fY2xlYW4gPC0KICAgIHBpdm90X2xvbmdlcih3aG8sCiAgICAgICAgICAgICAgICAgbmV3X3NwX20wMTQgOiBuZXdyZWxfZjY1LAogICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gImtleSIsCiAgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gImNvdW50IikgfD4KICAgIG11dGF0ZShrZXkgPSBzdWIoIm5ld18/IiwgIiIsIGtleSkpIHw+CiAgICBzZXBhcmF0ZShrZXksIGMoInR5cGUiLCAic2V4YWdlIikpCmBgYApgYGB7cn0Kd2hvX2NsZWFuCmBgYAoKU2VwYXJhdGUgYHNleGFnZWAgaW50byBgc2V4YCBhbmQgYGFnZWA6CgpgYGB7cn0Kd2hvX2NsZWFuIDwtCiAgICBwaXZvdF9sb25nZXIod2hvLAogICAgICAgICAgICAgICAgIG5ld19zcF9tMDE0IDogbmV3cmVsX2Y2NSwKICAgICAgICAgICAgICAgICBuYW1lc190byA9ICJrZXkiLAogICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJjb3VudCIpIHw+CiAgICBtdXRhdGUoa2V5ID0gc3ViKCJuZXdfPyIsICIiLCBrZXkpKSB8PgogICAgc2VwYXJhdGUoa2V5LCBjKCJ0eXBlIiwgInNleGFnZSIpKSB8PgogICAgc2VwYXJhdGUoc2V4YWdlLCBjKCJzZXgiLCAiYWdlIiksCiAgICAgICAgICAgICBzZXAgPSAxKQpgYGAKYGBge3J9Cndob19jbGVhbgpgYGAKCkZpeCB1cCBhZ2UgY2F0ZWdvcmllczoKCmBgYHtyfQp3aG9fY2xlYW4gPC0KICAgIHBpdm90X2xvbmdlcih3aG8sCiAgICAgICAgICAgICAgICAgbmV3X3NwX20wMTQgOiBuZXdyZWxfZjY1LAogICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gImtleSIsCiAgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gImNvdW50IikgfD4KICAgIG11dGF0ZShrZXkgPSBzdWIoIm5ld18/IiwgIiIsIGtleSkpIHw+CiAgICBzZXBhcmF0ZShrZXksIGMoInR5cGUiLCAic2V4YWdlIikpIHw+CiAgICBzZXBhcmF0ZShzZXhhZ2UsIGMoInNleCIsICJhZ2UiKSwKICAgICAgICAgICAgIHNlcCA9IDEpIHw+CiAgICBtdXRhdGUoYWdlID0gc3ViKCIwMTQiLCAiMDAxNCIsIGFnZSkpIHw+CiAgICBtdXRhdGUoYWdlID0gc3ViKCI2NSIsICI2NUluZiIsIGFnZSkpCmBgYApgYGB7cn0Kd2hvX2NsZWFuCmBgYAoKRmluYWxseSwgc3BsaXQgYGFnZWAgaW50byBgYWdlX2xvYCBhbmQgYGFnZS1oaWA6CgpgYGB7cn0Kd2hvX2NsZWFuIDwtCiAgICBwaXZvdF9sb25nZXIod2hvLAogICAgICAgICAgICAgICAgIG5ld19zcF9tMDE0IDogbmV3cmVsX2Y2NSwKICAgICAgICAgICAgICAgICBuYW1lc190byA9ICJrZXkiLAogICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJjb3VudCIpIHw+CiAgICBtdXRhdGUoa2V5ID0gc3ViKCJuZXdfPyIsICIiLCBrZXkpKSB8PgogICAgc2VwYXJhdGUoa2V5LCBjKCJ0eXBlIiwgInNleGFnZSIpKSB8PgogICAgc2VwYXJhdGUoc2V4YWdlLCBjKCJzZXgiLCAiYWdlIiksCiAgICAgICAgICAgICBzZXAgPSAxKSB8PgogICAgbXV0YXRlKGFnZSA9IHN1YigiMDE0IiwgIjAwMTQiLCBhZ2UpKSB8PgogICAgbXV0YXRlKGFnZSA9IHN1YigiNjUiLCAiNjVJbmYiLCBhZ2UpKSB8PgogICAgc2VwYXJhdGUoYWdlLCBjKCJhZ2VfbG8iLCAiYWdlX2hpIiksCiAgICAgICAgICAgICBzZXAgPSAyLAogICAgICAgICAgICAgY29udmVydCA9IFRSVUUpCmBgYApgYGB7cn0Kd2hvX2NsZWFuCmBgYAoKQSBwbG90IG9mIHRvdGFsIG51bWJlcnMgb2YgY2FzZXMgZm9yIGEgZmV3IGNvdW50cmllcyBvdmVyIHRpbWU6CgpgYGB7ciB3aG8tcGxvdCwgZXZhbCA9IEZBTFNFfQpmaWx0ZXIod2hvX2NsZWFuLAogICAgICAgaXNvMyAlaW4lIGMoIkNBTiIsICJERVUiLCAiR0JSIiwgIlVTQSIpKSB8PgogICAgZ3JvdXBfYnkoeWVhciwgaXNvMykgfD4KICAgIHN1bW1hcml6ZShjb3VudCA9IHN1bShjb3VudCwgbmEucm0gPSBUUlVFKSkgfD4KICAgIHVuZ3JvdXAoKSB8PgogICAgZ2dwbG90KCkgKwogICAgZ2VvbV9saW5lKGFlcyh4ID0geWVhciwKICAgICAgICAgICAgICAgICAgeSA9IGNvdW50LAogICAgICAgICAgICAgICAgICBjb2xvciA9IGlzbzMpKSArCiAgICB0aG0KYGBgCgpgYGB7ciB3aG8tcGxvdCwgZWNobyA9IEZBTFNFfQpgYGAKCgojIyMgRmlsbGluZyBJbiBPbWl0dGVkIFZhbHVlcwoKVGhlIFtSZWZ1Z2VlIFByb2Nlc3NpbmcgQ2VudGVyXShodHRwOi8vd3d3LndyYXBzbmV0Lm9yZy8pIHByb3ZpZGVzCmluZm9ybWF0aW9uIGZyb20gdGhlIFVuaXRlZCBTdGF0ZXMgUmVmdWdlZSBBZG1pc3Npb25zIFByb2dyYW0gKFVTUkFQKSwKaW5jbHVkaW5nClthcnJpdmFscyBieSBzdGF0ZSBhbmQgbmF0aW9uYWxpdHldKGh0dHA6Ly93d3cud3JhcHNuZXQub3JnL2FkbWlzc2lvbnMtYW5kLWFycml2YWxzLykuCgpUaHJlZSBmaWxlcyBhcmUgYXZhaWxhYmxlIGxvY2FsbHk6CgoqIGRhdGEgZnJvbSBbZWFybHkgSmFudWFyeQogIDIwMTddKGh0dHA6Ly9ob21lcGFnZS5zdGF0LnVpb3dhLmVkdS9+bHVrZS9kYXRhL0Fycml2YWxzLTIwMTctMDEtMDYueGxzKTsKCiogZGF0YSBmcm9tIFtlYXJseSBBcHJpbAogIDIwMTddKGh0dHA6Ly9ob21lcGFnZS5zdGF0LnVpb3dhLmVkdS9+bHVrZS9kYXRhL0Fycml2YWxzLTIwMTctMDQtMDUueGxzKS4KCiogZGF0YSBmcm9tIFtlYXJseSBNYXJjaAogIDIwMThdKGh0dHA6Ly9ob21lcGFnZS5zdGF0LnVpb3dhLmVkdS9+bHVrZS9kYXRhL0Fycml2YWxzLTIwMTgtMDMtMDUueGxzKS4KClRoZXNlIGFyZSBFeGNlbCBzcHJlYWQgc2hlZXRzLgoKUmVhZGluZyBpbiB0aGUgdGhpcmQgZmlsZSwgd2l0aCBkYXRhIGZvciBPY3RvYmVyIDIwMTcgdGhyb3VnaCBGZWJydWFyeQoyMDE4OgoKYGBge3IsIG1lc3NhZ2UgPSBGQUxTRX0KcmVmX3VybCA8LQogICAgImh0dHA6Ly9ob21lcGFnZS5zdGF0LnVpb3dhLmVkdS9+bHVrZS9kYXRhL0Fycml2YWxzLTIwMTgtMDMtMDUueGxzIgpyZWZfZmlsZSA8LSAiQXJyaXZhbHMtMjAxOC0wMy0wNS54bHMiCmlmICghIGZpbGUuZXhpc3RzKHJlZl9maWxlKSkKICAgIGRvd25sb2FkLmZpbGUocmVmX3VybCwgcmVmX2ZpbGUpCgpsaWJyYXJ5KHJlYWR4bCkKcmVmIDwtIHJlYWRfZXhjZWwocmVmX2ZpbGUsIHNraXAgPSAxNikgIyMgc2tpcCBoZWFkZXIKcmVmIDwtIGhlYWQocmVmLCAtMikgIyMgZHJvcCBsYXN0IHR3byByb3dzCm5hbWVzKHJlZilbMV0gPC0gIkRlc3RpbmF0aW9uIgpgYGAKClRoZSBgRGVzdGluYXRpb25gIGNvbHVtbiBuZWVkcyBmaWxsaW5nIGluICh0aGlzIGlzIGNvbW1vbiBpbgpzcHJlYWQgc2hlZXQgZGF0YSk6CgpgYGB7cn0KcmVmCmBgYAoKRmlsbCBkb3duIHRoZSBEZXN0aW5hdGlvbjoKCmBgYHtyfQpyZWZfY2xlYW4gPC0KICAgIGZpbGwocmVmLCBEZXN0aW5hdGlvbikKYGBgCgpgYGB7cn0KcmVmX2NsZWFuCmBgYAoKYGBge3IsIGluY2x1ZGUgPSBGQUxTRX0KIyMgY2hlY2sgdGhlIHRvdGFscwpyZWYudG90IDwtIGZpbHRlcihyZWZfY2xlYW4sIGlzLm5hKE5hdGlvbmFsaXR5KSkgfD4KICAgIHNlbGVjdChEZXN0aW5hdGlvbiwgT2N0IDogU2VwKQpydCA8LSBmaWx0ZXIocmVmX2NsZWFuLCAhIGlzLm5hKE5hdGlvbmFsaXR5KSkgfD4KICAgIHNlbGVjdChEZXN0aW5hdGlvbiwgTmF0aW9uYWxpdHksIE9jdCA6IFNlcCkgfD4KICAgIGdyb3VwX2J5KERlc3RpbmF0aW9uKSB8PgogICAgc3VtbWFyaXplX2lmKGlzLm51bWVyaWMsIHN1bSkKc3RvcGlmbm90KGlkZW50aWNhbChydCwgcmVmLnRvdCkpCmBgYAoKRHJvcCB0aGUgdG90YWxzOgoKYGBge3J9CnJlZl9jbGVhbiA8LQogICAgZmlsbChyZWYsIERlc3RpbmF0aW9uKSB8PgogICAgZmlsdGVyKCEgaXMubmEoTmF0aW9uYWxpdHkpKQpgYGAKCmBgYHtyfQpyZWZfY2xlYW4KYGBgCgpQaXZvdCB0byBhIHRpZGllciBmb3JtIHdpdGggb25lIHJvdyBwZXIgbW9udGg6CgpgYGB7cn0KcmVmX2NsZWFuIDwtCiAgICBmaWxsKHJlZiwgRGVzdGluYXRpb24pIHw+CiAgICBmaWx0ZXIoISBpcy5uYShOYXRpb25hbGl0eSkpIHw+CiAgICBwaXZvdF9sb25nZXIoCiAgICAgICAgT2N0IDogU2VwLAogICAgICAgIG5hbWVzX3RvID0gIm1vbnRoIiwKICAgICAgICB2YWx1ZXNfdG8gPSAiY291bnQiKQpgYGAKCmBgYHtyfQpyZWZfY2xlYW4KYGBgCgpUb3AgNSBkZXN0aW5hdGlvbiBzdGF0ZXM6CgpgYGB7cn0KZ3JvdXBfYnkocmVmX2NsZWFuLCBEZXN0aW5hdGlvbikgfD4KICAgIHN1bW1hcml6ZShjb3VudCA9IHN1bShjb3VudCkpIHw+CiAgICB1bmdyb3VwKCkgfD4KICAgIHNsaWNlX21heChjb3VudCwgbiA9IDUpCmBgYAoKVG9wIDUgb3JpZ2luIG5hdGlvbmFsaXRpZXM6CgpgYGB7cn0KZ3JvdXBfYnkocmVmX2NsZWFuLCBOYXRpb25hbGl0eSkgfD4KICAgIHN1bW1hcml6ZShjb3VudCA9IHN1bShjb3VudCkpIHw+CiAgICB1bmdyb3VwKCkgfD4KICAgIHNsaWNlX21heChjb3VudCwgbiA9IDUpCmBgYAoKCiMjIyBDb21wbGV0aW5nIE1pc3NpbmcgRmFjdG9yIExldmVsIENvbWJpbmF0aW9ucwoKVGhpcyBleGFtcGxlIGlzIGJhc2VkIG9uIGEgW2Jsb2cKcG9zdF0oaHR0cHM6Ly93aW4tdmVjdG9yLmNvbS8yMDE3LzAyLzIxL3RoZS16ZXJvLWJ1Zy8pIHVzaW5nIGRhdGEKZnJvbSB0aGUgW1NpZ25pZmljYW50IEVhcnRocXVha2UKRGF0YWJhc2VdKGh0dHBzOi8vd3d3Lm5jZWkubm9hYS5nb3YvYWNjZXNzL21ldGFkYXRhL2xhbmRpbmctcGFnZS9iaW4vaXNvP2lkPWdvdi5ub2FhLm5nZGMubWdnLmhhemFyZHM6RzAxMjE1MykuCjwhLS1odHRwczovL3d3dy5uZ2RjLm5vYWEuZ292L2hhemVsL3ZpZXcvaGF6YXJkcy9lYXJ0aHF1YWtlL3NlYXJjaC0tPgoKVGhlIGRhdGEgdXNlZCBpcyBhIHNlbGVjdGlvbiBmb3IgTm9ydGggQW1lcmljYSBhbmQgSGF3YWlpLgoKUmVhZGluZyBpbiB0aGUgZGF0YToKCmBgYHtyfQppZiAoISBmaWxlLmV4aXN0cygiZWFydGhxdWFrZXMudHN2IikpCiAgICBkb3dubG9hZC5maWxlKCJodHRwOi8vaG9tZXBhZ2Uuc3RhdC51aW93YS5lZHUvfmx1a2UvZGF0YS9lYXJ0aHF1YWtlcy50c3YiLAogICAgICAgICAgICAgICAgICAiZWFydGhxdWFrZXMudHN2IikKZXEgPC0gcmVhZC50YWJsZSgiZWFydGhxdWFrZXMudHN2IiwKICAgICAgICAgICAgICAgICBoZWFkZXIgPSBUUlVFLAogICAgICAgICAgICAgICAgIHNlcCA9ICJcdCIpCmVxIDwtIGFzX3RpYmJsZShlcSkKYGBgCgpgYGB7cn0Kc2VsZWN0KGVxLCBTVEFURSwgZXZlcnl0aGluZygpKQpgYGAKClNvbWUgYFNUQVRFYCBlbnRyaWVzIGFyZSBibGFuaywgYnV0IHRoZSBsb2NhdGlvbiBuYW1lIGlkZW50aWZpZXMgdGhlIHN0YXRlOgoKYGBge3J9CmZpbHRlcihlcSwgU1RBVEUgPT0gIiIpIHw+CiAgICBzZWxlY3QoU1RBVEUsIExPQ0FUSU9OX05BTUUpCmBgYAoKT25lIHdheSB0byBmaXggdGhpczoKCmBgYHtyfQpiYWRpZHggPC0gd2hpY2goZXEkU1RBVEUgPT0gIiIpCmJhZHN0YXRlIDwtIHN1YigiOi4qIiwgIiIsCiAgICAgICAgICAgICAgICBlcSRMT0NBVElPTl9OQU1FW2JhZGlkeF0pCmVxJFNUQVRFW2JhZGlkeF0gPC0KICAgIHN0YXRlLmFiYlttYXRjaCh0b2xvd2VyKGJhZHN0YXRlKSwKICAgICAgICAgICAgICAgICAgICB0b2xvd2VyKHN0YXRlLm5hbWUpKV0KYGBgCgpBZGQgZnVsbCBzdGF0ZSBuYW1lcyBhcyBhIGZhY3RvcjoKCmBgYHtyfQpzdGF0ZXMgPC0gZGF0YS5mcmFtZShTVEFURSA9IHN0YXRlLmFiYiwKICAgICAgICAgICAgICAgICAgICAgU1RBVEVfTkFNRSA9IHN0YXRlLm5hbWUpCmVxIDwtIGxlZnRfam9pbihlcSwgc3RhdGVzLCAiU1RBVEUiKQplcSA8LSBtdXRhdGUoZXEsCiAgICAgICAgICAgICBTVEFURV9OQU1FID0gZmFjdG9yKFNUQVRFX05BTUUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXRlLm5hbWUpKQpgYGAKCkxvb2sgYXQgdG90YWwgbnVtYmVyIG9mIGVhcnRocXVha2VzIGJ5IHN0YXRlOgoKYGBge3J9CnRibCA8LSBjb3VudChlcSwgU1RBVEVfTkFNRSkKaGVhZCh0YmwsIDYpCmBgYAoKYGBge3IsIGZpZy5oZWlnaHQgPSA1LCBmaWcud2lkdGggPSA4LCBjbGFzcy5zb3VyY2UgPSAiZm9sZC1oaWRlIn0KbGlicmFyeShmb3JjYXRzKQpwYiA8LSBnZ3Bsb3QodGJsLCBhZXMoeSA9IGZjdF9yZXYoU1RBVEVfTkFNRSksIHggPSBuKSkgKwogICAgZ2VvbV9jb2woKSArCiAgICB0aG0gKwogICAgdGhlbWUoYXhpcy50aXRsZS55ID0gZWxlbWVudF9ibGFuaygpKQpwcyA8LSBnZ3Bsb3QodGJsLCBhZXMoeSA9IGZjdF9yZXYoU1RBVEVfTkFNRSksIHggPSBuKSkgKwogICAgZ2VvbV9wb2ludCgpICsKICAgIGdlb21fc2VnbWVudChhZXMoeGVuZCA9IDAsIHllbmQgPSBmY3RfcmV2KFNUQVRFX05BTUUpKSkgKwogICAgdGhtICsKICAgIHRoZW1lKGF4aXMudGl0bGUueSA9IGVsZW1lbnRfYmxhbmsoKSkKcGIgfCBwcwpgYGAKCk1hbnkgc3RhdGVzLCBpbmNsdWRpbmcgSW93YSwgYXJlIG1pc3NpbmcuCgpUaGVpciBjb3VudHMgc2hvdWxkIGJlIHplcm8uCgpUaGlzIG9mdGVuIGltcGFjdHMgdmlzdWFsaXphdGlvbnMuCgpgY29tcGxldGUoKWAgY2FuIGJlIHVzZWQgdG8gZmlsbCBpbiB0aGUgemVybyB2YWx1ZXMuCgpgYGB7cn0KdGJsIDwtCiAgICBjb3VudChlcSwgU1RBVEVfTkFNRSkgfD4KICAgIGNvbXBsZXRlKFNUQVRFX05BTUUsIGZpbGwgPSBsaXN0KG4gPSAwKSkKdGJsCmBgYAoKYGBge3IsIGZpZy5oZWlnaHQgPSA3LCBmaWcud2lkdGggPSA4LCBjbGFzcy5zb3VyY2UgPSAiZm9sZC1oaWRlIn0KdGhtX3l0eHRfc20gPC0gdGhlbWUoYXhpcy50ZXh0LnkgPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEwKSkKKHBiICUrJSB0YmwgKyB0aG1feXR4dF9zbSkgfCAocHMgJSslIHRibCArIHRobV95dHh0X3NtKQpgYGAKCjwhLS0KCiMgbm9saW50IHN0YXJ0ClBsYXkgYXJvdW5kIHdpdGggYHRlZWAKYGBge3IsIGV2YWwgPSBGQUxTRX0KdGVlIDwtIGZ1bmN0aW9uKHgsIGZ1bikgeyBmdW4oeCk7IHggfQp0ZWUoLiB8PiBoZWFkKDIpIHw+IHByaW50KSB8Pgp0ZWUoYnJvd3NlcikgIyMgYnJvd3NlclRleHQgZ2V0cyB0aGUgYXJnCiMgbm9saW50IGVuZApgYGAKCk1lbnRpb24gYGZpbGxgIGFuZCBzdWNoLgotLT4KCjwhLS0KCiAgICBvcHRpb25zKHRpYmJsZS5wcmludF9tYXggPSBuLCB0aWJibGUucHJpbnRfbWluID0gbSk6IGlmIHRoZXJlIGFyZQogICAgbW9yZSB0aGFuIG4gcm93cywgcHJpbnQgb25seSB0aGUgZmlyc3QgbSByb3dzLiBVc2UKICAgIG9wdGlvbnModGliYmxlLnByaW50X21heCA9IEluZikgdG8gYWx3YXlzIHNob3cgYWxsIHJvd3MuCgogICAgb3B0aW9ucyh0aWJibGUud2lkdGggPSBJbmYpIHdpbGwgYWx3YXlzIHByaW50IGFsbCBjb2x1bW5zLAogICAgcmVnYXJkbGVzcyBvZiB0aGUgd2lkdGggb2YgdGhlIHNjcmVlbi4KCgojIyAqKioqIHRyeSB0byByZWFkIGZ1bGwgdGFibGUga2lsbHMgdGhlIHByb2Nlc3MKCiMjICoqKiogY2hlY2sgaWYgbW9yZSByZWFzb25hYmxlIG9uIFItZGV2ZWwKCi0tPgoKCiMjIFJlYWRpbmcKCkNoYXB0ZXIgW19EYXRhIHRyYW5zZm9ybWF0aW9uX10oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei90cmFuc2Zvcm0uaHRtbCkgb2YKW1IgZm9yIERhdGEgU2NpZW5jZV0oaHR0cHM6Ly9yNGRzLmhhZGxleS5uei8pLgoKQ2hhcHRlciBbX1RpZHkgZGF0YV9dKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovZGF0YS10aWR5Lmh0bWwpIG9mCltSIGZvciBEYXRhIFNjaWVuY2VdKGh0dHBzOi8vcjRkcy5oYWRsZXkubnovKS4KCgojIyBJbnRlcmFjdGl2ZSBUdXRvcmlhbAoKQW4gaW50ZXJhY3RpdmUgW2BsZWFybnJgXShodHRwczovL3JzdHVkaW8uZ2l0aHViLmlvL2xlYXJuci8pIHR1dG9yaWFsCmZvciB0aGVzZSBub3RlcyBpcyBbYXZhaWxhYmxlXShgciBXTE5LKCJ0dXRvcmlhbHMvZHBseXIuUm1kIilgKS4KCllvdSBjYW4gcnVuIHRoZSB0dXRvcmlhbCB3aXRoCgpgYGB7ciwgZXZhbCA9IEZBTFNFfQpTVEFUNDU4MDo6cnVuVHV0b3JpYWwoImRwbHlyIikKYGBgCgpZb3UgY2FuIGluc3RhbGwgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgYFNUQVQ0NTgwYCBwYWNrYWdlIHdpdGgKCmBgYHtyLCBldmFsID0gRkFMU0V9CnJlbW90ZXM6Omluc3RhbGxfZ2l0bGFiKCJsdWtlLXRpZXJuZXkvU1RBVDQ1ODAiKQpgYGAKCllvdSBtYXkgbmVlZCB0byBpbnN0YWxsIHRoZSBgcmVtb3Rlc2AgcGFja2FnZSBmcm9tIENSQU4gZmlyc3QuCgoKIyMgRXhlcmNpc2VzCgoxLiBUbyBicmluZyBpbiBgZHBseXJgIGFuZCB0aGUgYG1wZ2AgZGF0YSwgc3RhcnQgYnkgZXZhbHVhdGluZwoKICAgIGBgYHIKICAgIGxpYnJhcnkoZHBseXIpCiAgICBsaWJyYXJ5KGdncGxvdDIpCiAgICBgYGAKICAgIFRoZSBgc2VsZWN0YCBmdW5jdGlvbiBhbGxvd3MgdmFyaWFibGVzIHRvIGJlIHNwZWNpZmllZCBpbiBhIHZhcmlldHkgb2YKICAgIHdheXMuIFdoaWNoIG9mIHRoZSBmb2xsb3dpbmcgZG9lcyAqKm5vdCoqIHByb2R1Y2UgYSBkYXRhIGZyYW1lIHdpdGggb25seQogICAgdGhlIHZhcmlhYmxlcyBgbWFudWZhY3R1cmVyYCwgYG1vZGVsYCwgYGN0eWAsIGBod3lgPwoKICAgIGEuIGBzZWxlY3QobXBnLCAxOjIsIDc6OClgCiAgICBiLiBgc2VsZWN0KG1wZywgc3RhcnRzX3dpdGgoIm0iKSwgZW5kc193aXRoKCJ5IikpYAogICAgYy4gYHNlbGVjdChtcGcsIDE6MiwgY3R5IDogaHd5KWAKICAgIGQuIGBzZWxlY3QobXBnLCAtKGRpc3BsIDogZHJ2KSwgLShmbCA6IGNsYXNzKSlgCgoyLiBDb25zaWRlciB0aGUgY29kZQoKICAgIGBgYHIKICAgIGxpYnJhcnkoZHBseXIpCiAgICBsaWJyYXJ5KGdncGxvdDIpCiAgICBmaWx0ZXIobXBnLCAtLS0pIHw+IG5yb3coKQogICAgYGBgCgogICAgV2hpY2ggb2YgdGhlIHJlcGxhY2VtZW50cyBmb3IgYC0tLWAgY29tcHV0ZXMgdGhlIG51bWJlciBvZiBGb3JkCiAgICB2ZWhpY2xlcyB3aXRoIG1vcmUgdGhhbiA2IGN5bGluZGVycyBpbiB0aGUgYG1wZ2AgdGFibGU/CgogICAgYS4gYG1vZGVsID09ICJmb3JkIiwgY3lsIDw9IDZgCiAgICBiLiBgbWFudWZhY3R1cmVyID09ICJmb3JkIiB8IGN5bCA+IDZgCiAgICBjLiBgbWFudWZhY3R1cmVyID09ICJmb3JkIiwgY3lsID4gNmAKICAgIGQuIGBtYW51ZmFjdHVyZXIgPT0gImZvcmQiLCBjeWxpbmRlcnMgPiA2YAoKMy4gSW4gdGhlIDIwMTMgTllDIGZsaWdodHMgZGF0YSBwcm92aWRlZCBieSB0aGUgYG55Y2ZsaWdodHMxM2AgcGFja2FnZQogICBob3cgbWFueSBmbGlnaHRzIHdlcmUgdGhlcmUgdG8gRGVzIE1vaW5lcyAoRkFBIGNvZGUgRFNNKSBmcm9tIE5ZQwogICBpbiB0aGUgZmlyc3QgdGhyZWUgbW9udGhzIG9mIDIwMTM/CgogICAgYS4gMTMKICAgIGIuIDk4CiAgICBjLiA2NAogICAgZC4gNzcKCjQuIFRvIGJyaW5nIGluIGBkcGx5cmAgYW5kIHRoZSBgbXBnYCBkYXRhLCBzdGFydCBieSBldmFsdWF0aW5nCgogICAgYGBgcgogICAgbGlicmFyeShkcGx5cikKICAgIGxpYnJhcnkoZ2dwbG90MikKICAgIGBgYAoKICAgIFdoaWNoIG9mIHRoZSBmb2xsb3dpbmcgc29ydHMgdGhlIHJvd3MgZm9yIGBtcGdgIGJ5IGluY3JlYXNpbmcgYGN5bGAKICAgIHZhbHVlIGFuZCwgd2l0aGluIGVhY2ggYGN5bGAgdmFsdWUgc29ydHMgdGhlIHJvd3MgZnJvbSBsYXJnZXN0IHRvCiAgICBzbWFsbGVzdCBgaHd5YCB2YWx1ZS4KCiAgICBhLiBgYXJyYW5nZShtcGcsIGRlc2MoaHd5KSwgY3lsKWAKICAgIGIuIGBhcnJhbmdlKG1wZywgY3lsLCBkZXNjKGh3eSkpYAogICAgYy4gYGFycmFuZ2UobXBnLCBkZXNjKGN5bCksIGh3eSlgCiAgICBkLiBgYXJyYW5nZShtcGcsIGN5bCwgaHd5KWAKCjUuIFRvIGJyaW5nIGluIGBkcGx5cmAgYW5kIHRoZSBgZmxpZ2h0c2AgZGF0YSwgc3RhcnQgYnkgZXZhbHVhdGluZwoKICAgIGBgYHIKICAgIGxpYnJhcnkoZHBseXIpCiAgICBsaWJyYXJ5KG55Y2ZsaWdodHMxMykKICAgIGBgYAoKICAgIFRoZSBgZGVwX3RpbWVgIHZhcmlhYmxlIGluIHRoZSBgZmxpZ2h0c2AgZGF0YSBzZXQgZnJvbSB0aGUKICAgIGBueWNmbGlnaHRzMTNgIHBhY2thZ2UgaXMgY29udmVuaWVudCB0byBsb29rIGF0ICg1MjkgbWVhbnMgNToyOSBBTQogICAgYW5kIDIyOjEyIG1lYW5zIDEwOjEyIFBNKSwgYnV0IGhhcmQgdG8gY29tcHV0ZSB3aXRoLiBXaGljaCBvZiB0aGUKICAgIGZvbGxvd2luZyBhZGRzIHZhcmlhYmxlcyBgZGVwX2hvdXJgIGFuZCBgZGVwX21pbmAgY29udGFpbmluZyBob3VyCiAgICBhbmQgbWludXRlIG9mIHRoZSBkZXBhcnR1cmUgdGltZT8KCiAgICBhLiBgbXV0YXRlKGZsaWdodHMsIGRlcF9ob3VyID0gZGVwX3RpbWUgJS8lIDYwLCBkZXBfbWluID0gZGVwX3RpbWUgJSUgNjApYAogICAgYi4gYG11dGF0ZShmbGlnaHRzLCBkZXBfaG91ciA9IGRlcF90aW1lICUvJSAxMDAsIGRlcF9taW4gPSBkZXBfdGltZSAlJSAxMDApYAogICAgYy4gYG11dGF0ZShmbGlnaHRzLCBkZXBfaG91ciA9IGRlcF90aW1lIC8gMTAwLCBkZXBfbWluID0gZGVwX3RpbWUgLSBkZXBfaG91cilgCiAgICBhLiBgbXV0YXRlKGZsaWdodHMsIGRlcF9ob3VyID0gZGVwX3RpbWUgJS8lIDYwLCBkZXBfbWluID0gZGVwX2hvdXIgJSUgNjApYAoKNi4gVXNpbmcgdGhlIGBnYXBtaW5kZXJgIGRhdGEgc2V0LCB0aGUgZm9sbG93aW5nIGNvZGUgY29tcHV0ZXMKICAgIHBvcHVsYXRpb24gdG90YWxzIGZvciBlYWNoIGNvbnRpbmVudCBhbmQgZWFjaCB5ZWFyOgoKICAgIGBgYHtyLCBtZXNzYWdlID0gRkFMU0V9CiAgICBsaWJyYXJ5KGRwbHlyKQogICAgbGlicmFyeShnYXBtaW5kZXIpCiAgICBjcG9wcyA8LSBncm91cF9ieShnYXBtaW5kZXIsIGNvbnRpbmVudCwgeWVhcikgfD4KICAgICAgICBzdW1tYXJpemUocG9wID0gc3VtKHBvcCkpIHw+CiAgICAgICAgdW5ncm91cCgpCiAgICBgYGAKCiAgICBUbyBwcm9kdWNlIGEgcGxvdCB0byBjb21wYXJlIHBvcHVsYXRpb24gZ3Jvd3RoIG92ZXIgdGhlIHllYXJzIGZvcgogICAgdGhlIGNvbnRpbmVudHMgaXQgaXMgdXNlZnVsIHRvIHN0YW5kYXJkaXplIHRoZSBwb3B1bGF0aW9uIGRhdGEgZm9yCiAgICBlYWNoIGNvbnRpbmVudCwgZm9yIGV4YW1wbGUgYnkgZGl2aWRpbmcgdGhlIHBvcHVsYXRpb24gdmFsdWVzIGJ5CiAgICB0aGUgYXZlcmFnZSBwb3B1bGF0aW9uIHNpemUgZm9yIGVhY2ggY29udGluZW50LiBPbmUgd2F5IHRvIGRvIHRoaXMKICAgIGlzIHdpdGggYSBncm91cGVkIGBtdXRhdGVgLiBUaGUgZmlyc3QgbGluZSBvZiB5b3VyIHJlc3VsdCBzaG91bGQgYmUKCiAgICBgYGB7ciwgaW5jbHVkZSA9IEZBTFNFfQogICAgY3BvcHNfc3RkIDwtIGdyb3VwX2J5KGNwb3BzLCBjb250aW5lbnQpIHw+CiAgICAgICAgbXV0YXRlKHN0ZHBvcCA9IHBvcCAvIG1lYW4ocG9wKSkgfD4KICAgICAgICB1bmdyb3VwKCkKICAgIGBgYAogICAgYGBge3J9CiAgICBoZWFkKGNwb3BzX3N0ZCwgMSkKICAgIGBgYAoKICAgIGBgYHtyLCBmaWcuY2FwID0gIiJ9CiAgICBsaWJyYXJ5KGdncGxvdDIpCiAgICBnZ3Bsb3QoY3BvcHNfc3RkLCBhZXMoeCA9IHllYXIsIHkgPSBzdGRwb3AsIGNvbG9yID0gY29udGluZW50KSkgKwogICAgICAgIGdlb21fbGluZSgpCiAgICBgYGAKCiAgICBXaGljaCBvZiB0aGUgZm9sbG93aW5nIHByb2R1Y2VzIHRoZSBjb3JyZWN0IHJlc3VsdDoKCiAgICBhLiBgY3BvcHNfc3RkIDwtIGdyb3VwX2J5KGNwb3BzLCBjb250aW5lbnQpIHw+IG11dGF0ZShzdGRwb3AgPSBwb3AgLyBtZWFuKHllYXIpKSB8PiB1bmdyb3VwKClgCiAgICBhLiBgY3BvcHNfc3RkIDwtIGdyb3VwX2J5KGNwb3BzLCB5ZWFyKSB8PiBtdXRhdGUoc3RkcG9wID0gcG9wIC8gbWVhbihwb3ApKSB8PiB1bmdyb3VwKClgCiAgICBjLiBgY3BvcHNfc3RkIDwtIGdyb3VwX2J5KGNwb3BzLCBjb250aW5lbnQpIHw+IG11dGF0ZShzdGRwb3AgPSBwb3AgLyBtZWFuKHBvcCkpIHw+IHVuZ3JvdXAoKWAKICAgIGQuIGBjcG9wc19zdGQgPC0gZ3JvdXBfYnkoY3BvcHMsIHllYXIpIHw+IG11dGF0ZShzdGRwb3AgPSBwb3AgLyBtZWFuKHllYXIpKSB8PiB1bmdyb3VwKClgCgo3LiBBbm90aGVyIGFwcHJvYWNoIHRvIHRoZSBwcmV2aW91cyBleGVyY2lzZSBmaXJzdCBjcmVhdGVzIGEgdGFibGUgb2YKICAgcG9wdWxhdGlvbiBhdmVyYWdlcyB3aXRoCgogICAgYGBgcgogICAgY3BvcHNfYXZnIDwtIGdyb3VwX2J5KGNwb3BzLCBjb250aW5lbnQpIHw+CiAgICAgICAgc3VtbWFyaXplKGF2Z19wb3AgPSBtZWFuKHBvcCkpCiAgICBgYGAKCiAgICBUaGVuIHVzZSBhIGxlZnQgam9pbiB0byBhZGQgYGF2Z19wb3BgIHRvIHRoZSBgY3BvcHNgIHRhYmxlLAogICAgZm9sbG93ZWQgYnkgYW4gb3JkaW5hcnkgbXV0YXRlIHN0ZXA6CgogICAgYGBgcgogICAgbGVmdF9qb2luKC0tLSkgfD4KICAgICAgICBtdXRhdGUoc3RkcG9wID0gcG9wIC8gYXZnX3BvcCkKICAgIGBgYAoKICAgIFdoaWNoIGlzIHRoZSBjb3JyZWN0IHJlcGxhY2VtZW50IGZvciBgLS0tYD8KCiAgICBhLiBgY3BvcHNfYXZnLCBjcG9wcywgImNvbnRpbmVudCJgCiAgICBiLiBgY3BvcHMsIGNwb3BzX2F2ZywgImNvbnRpbmVudCJgCiAgICBjLiBgY3BvcHMsIGNwb3BzX2F2ZywgInllYXIiYAogICAgZC4gYGNwb3BzX2F2ZywgY3BvcHMsICJ5ZWFyImAK