Some Other Topics
Some topics we did not have time to look at:
Visualizing Uncertainty: Hurricanes
All estimates from data are associated with some degree of uncertainty.
Effectively communicating that uncertainty in visualizations is challenging and an active area of research.
The cone of uncertainty: (From Cairo (2019); images from a blog post by the author.)
The NHC forecast cone is designed so that two-thirds of historical official forecast errors over a 5-year sample fall within the cone for a particular time point..
When published in the media these visualizations are routinely misinterpreted something like this:
A more effective representation might be something like this, showing an ensemble of possible tracks:
An animated version may be more effective, if the presentation medium permits.
Developing better visualizations for hurricane forecasting, especially targeting the public, is an active area of research.
Visualizing Uncertainty: Chocolate Bars
Expert ratings, on a scale from 0 to 5, for chocolate bars manufactured in several countries:
The standard deviations of the data distributions are comparable, but the lengths of confidence intervals for the mean vary because of the different sample sizes:
The same plot with a reduced horizontal range:
A more elaborate display with confidence intervals at several levels:
Confidence densities, or confidence distributions, as proposed in
Adrian W. Bowman. Graphics for Uncertainty. J. R. Statist. Soc. A 182:1-16, 2018. Link
One drawback of all of these methods:
The least precise measurement draws the most attention.
These examples from Wilke’s book use the ungeviz
package available on GitHub.
Another package providing some tools for uncertainty visualization is ggdist
package.
Visualizing Uncertainty: Old Cars
Using the very old mtcars
data set to illustrate estimating a smooth relationship:
A default geom_smooth
shows an estimate along with a point-wise confidence band.
This may not give the best sense of the joint uncertainty: if the curve is higher on some places it may need to be lower in others.
Showing an ensemble of curves that all are plausible can be a better choice.
This approach was shown earlier for visualizing possible hurricane paths.
This ensemble is generated using a case-based bootstrap.
These plots are called ensemble plots (also spaghetti plots, for obvious reasons).
If animation is available, an alternative is to show the curves one at a time in an animation.
Again, a bootstrap is used to produce the estimates.
This is an example of a hypothetical outcomes plot, or HOP, as introduced in
Hullman, Jessica, Paul Resnick, and Eytan Adar. “Hypothetical outcome plots outperform error bars and violin plots for inferences about reliability of variable ordering.” PLOS ONE 10, no. 11 (2015).
Data Quality and Integrity
A visualization can accurately reflect data but still be misleading if the data are faulty.
A NY Times article from May 2021 shows a choropleth map of the estimated share of adults who would “definitely” or “probably” get the COVID-19 vaccine.
Cutoffs: 49 60 65 70 75 80 91 %
The map may accurately reflect the estimates, but the estimates have obvious problems.
The data used for the map are available here.
Discussions on social media suggest that the state level data may be more reasonable:
Data Science Ethics
Some issues:
Some references:
Plot Annotation, Plot Ensembles, and Dashboards
Plot annotations can create popout and help focus the viewer’s attention.
They may be increasingly important as images are shared on line without context.
Here is an examples for the mpg
data:
Plot Ensembles: Coffee
It is often useful to use several graphics to present an analysis.
Collections of related graphs are sometimes called ensemble graphics.
On line presentations of analyses involving multiple visualizations and, typically, some interactive features are also called dashboards.
To aid the viewer it is usually best to design these visualizations together, with common axis choices and color mappings.
Fig 12.1 in Unwin (2015) provides a simple example:
library(ggplot2)
library(GGally)
library(gridExtra)
coffee_thm <- theme(text = element_text(size = 14))
data(coffee, package = "pgmm")
coffee <- within(coffee, Type <- ifelse(Variety == 1,
"Arabica", "Robusta"))
names(coffee) <- abbreviate(names(coffee), 8)
a <- ggplot(coffee, aes(x = Type)) + geom_bar(aes(fill = Type)) +
scale_fill_manual(values = c("grey70", "red")) +
guides(fill = "none") + ylab("") +
coffee_thm
b <- ggplot(coffee, aes(x = Fat, y = Caffine, colour = Type)) +
geom_point(size = 3) +
scale_colour_manual(values = c("grey70", "red")) +
coffee_thm
c <- ggparcoord(coffee[order(coffee$Type), ], columns = 3 : 14,
groupColumn = "Type", scale = "uniminmax") +
xlab("") + ylab("") +
theme(legend.position = "none") +
scale_colour_manual(values = c("grey", "red")) +
theme(axis.ticks.y = element_blank(),
axis.text.y = element_blank()) +
coffee_thm
grid.arrange(arrangeGrob(a, b, ncol = 2, widths = c(1, 2)),
c, nrow = 2)
Data on the chemical composition of coffee samples collected from around the world, comprising 43 samples from 29 countries. Each sample is either of the Arabica or Robusta variety. Twelve of the thirteen chemical constituents reported in the study are given. The omitted variable is total chlorogenic acid; it is generally the sum of the chlorogenic, neochlorogenic and isochlorogenic acid values.
Streuli, H. (1973). Der heutige stand der kaffeechemie. In Association Scientifique International du Cafe, 6th International Colloquium on Coffee Chemisrty, Bogata, Columbia, pp. 61-72.
Making a Point and Telling a Story
In a report, make sure each plot has a point and makes its point.
Make sure to think about:
It is often good to make sure a figure can stand on its own without asking the reader to search the text for explanations.
Communicating with data is like telling a story, with a starting point, a journey, and an end.
Sometimes a single visualization can capture the full story.
More often, several visualizations will be needed.
Often it is good to:
start with a high level overview;
show how to look at some particular cases, e.g. with a single plot;
build up to a more complete analysis, e.g. with a multi-panel plot.
With multiple visualizations it is good make sure that:
There is a chapter of Wilke, 2019 with more advice on this.
A recent book length treatment is
Deborah Nolan and Sara Stoudt (2021) Communicating with Data, Oxford Univerity Press.
Wrapping Up
Some of the areas we covered:
Visualization
Many different types of graphs.
- Strengths, weaknesses.
- Pitfalls.
- Scalability.
- Creating these graphs in R.
Perception
- Channels and mappings; relative effectiveness.
- Using to assess, design visualizations.
- Effective use of color.
A little on interaction, animation.
Emphasis on techniques useful for exploration, scientific reporting.
Data Technologies
Reading different data formats.
Scraping data from the web.
Cleaning data.
Rearranging data for analysis.
Merging data from several sources.
Learning More
Class notes will remain available, in some form, at the class web site.
Some books to look at:
Some blogs to check out:
Keep a critical eye out for good (and not so good) uses of data visualization in the media.
LS0tCnRpdGxlOiAiRmluYWwgTm90ZXMiCm91dHB1dDoKICBodG1sX2RvY3VtZW50OgogICAgdG9jOiB5ZXMKICAgIGNvZGVfZm9sZGluZzogc2hvdwogICAgY29kZV9kb3dubG9hZDogdHJ1ZQotLS0KCmBgYHtyIHNldHVwLCBpbmNsdWRlID0gRkFMU0V9CnNvdXJjZShoZXJlOjpoZXJlKCJzZXR1cC5SIikpCm9wdGlvbnMoaHRtbHRvb2xzLmRpci52ZXJzaW9uID0gRkFMU0UpCmxpYnJhcnkoZ2dwbG90MikKa25pdHI6Om9wdHNfY2h1bmskc2V0KGNvbGxhcHNlID0gVFJVRSwgY2xhc3Muc291cmNlID0gImZvbGQtaGlkZSIsCiAgICAgICAgICAgICAgICAgICAgICBtZXNzYWdlID0gRkFMU0UsIGZpZy5hbGlnbiA9ICJjZW50ZXIiKQpsaWJyYXJ5KHRpZHl2ZXJzZSkKdGhlbWVfc2V0KHRoZW1lX21pbmltYWwoKSArCiAgICAgICAgICB0aGVtZSh0ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxNikpICsKICAgICAgICAgIHRoZW1lKHBhbmVsLmJvcmRlciA9IGVsZW1lbnRfcmVjdChjb2xvciA9ICJncmV5MzAiLCBmaWxsID0gTkEpKSkKc2V0LnNlZWQoMTIzNDUpCmBgYAoKCiMjIFNvbWUgT3RoZXIgVG9waWNzCgpTb21lIHRvcGljcyB3ZSBkaWQgbm90IGhhdmUgdGltZSB0byBsb29rIGF0OgoKKiBXb3JraW5nIHdpdGggbW9kZWxzIChbQ2hhcHRlciA2IGluIEhlYWx5LAogIDIwMThdKGh0dHBzOi8vc29jdml6LmNvL21vZGVsaW5nLmh0bWwpOyBbVGlkeSBNb2RlbGluZyB3aXRoIFJdKGh0dHBzOi8vd3d3LnRtd3Iub3JnLykpLgoKKiBbVmlzdWFsaXppbmcgbWlzc2luZwogIHZhbHVlc10oaHR0cDovL25hbmlhci5uanRpZXJuZXkuY29tL2FydGljbGVzL25hbmlhci12aXN1YWxpc2F0aW9uLmh0bWwpLgoKKiBWaXN1YWxpemluZyB1bmNlcnRhaW50eSAoW0NoYXB0ZXIgMTYgb2YgV2lsa2UsCiAgMjAxOV0oaHR0cHM6Ly9jbGF1c3dpbGtlLmNvbS9kYXRhdml6L3Zpc3VhbGl6aW5nLXVuY2VydGFpbnR5Lmh0bWwpCiAgYW5kIFtiZWxvd10oI3Zpc3VhbGl6aW5nLXVuY2VydGFpbnR5KSkKCiogUGxvdCBhbm5vdGF0aW9uLCBwbG90IGVuc2VtYmxlcywgYW5kIGRhc2hib2FyZHMuIChbUGFydCBJSSBvZiBXaWxrZSwKICAyMDE5XShodHRwczovL2NsYXVzd2lsa2UuY29tL2RhdGF2aXovcHJvcG9ydGlvbmFsLWluay5odG1sKTsKICBbQ2hhcHRlciA1IG9mIEhlYWx5LCAyMDE4XShodHRwczovL3NvY3Zpei5jby93b3JrZ2VvbXMuaHRtbCk7CiAgW2JlbG93XSgjcGxvdC1hbm5vdGF0aW9uLXBsb3QtZW5zZW1ibGVzLWFuZC1kYXNoYm9hcmRzKSkuCgoqIERhdGEgU2NpZW5jZSBFdGhpY3MgKFtiZWxvd10oI2RhdGEtc2NpZW5jZS1ldGhpY3MpKS4KCgojIyBWaXN1YWxpemluZyBVbmNlcnRhaW50eTogSHVycmljYW5lcwoKQWxsIGVzdGltYXRlcyBmcm9tIGRhdGEgYXJlIGFzc29jaWF0ZWQgd2l0aCBzb21lIGRlZ3JlZSBvZiB1bmNlcnRhaW50eS4KCkVmZmVjdGl2ZWx5IGNvbW11bmljYXRpbmcgdGhhdCB1bmNlcnRhaW50eSBpbiB2aXN1YWxpemF0aW9ucyBpcwpjaGFsbGVuZ2luZyBhbmQgYW4gYWN0aXZlIGFyZWEgb2YKW3Jlc2VhcmNoXShodHRwOi8vc3BhY2UudWNtZXJjZWQuZWR1L2NoYXB0ZXIpLgoKVGhlIF9jb25lIG9mIHVuY2VydGFpbnR5XzogKEZyb20gQ2Fpcm8gKDIwMTkpOyBpbWFnZXMgZnJvbSBhIFtibG9nCnBvc3RdKGh0dHA6Ly93d3cudGhlZnVuY3Rpb25hbGFydC5jb20vMjAyMC8wMS9hbGwtZ3JhcGhpY3MtZnJvbS1ob3ctY2hhcnRzLWxpZS1mcmVlbHkuaHRtbCkKYnkgdGhlIGF1dGhvci4pCgo8IS0tCmh0dHBzOi8vd3d3LmRyb3Bib3guY29tL3NoL2Qxa2IwamRyaGtiNDNqOS9BQURUQmZSdkFoLW14bVN4QlJOWnBMSmphLzUuQ0hBUFRFUjU/ZGw9MCZwcmV2aWV3PVBERjEwLlRyb3BpY2Fsc3Rvcm0ucGRmJnN1YmZvbGRlcl9uYXZfdHJhY2tpbmc9MQotLT4KYGBge3IsIGVjaG8gPSBGQUxTRX0Ka25pdHI6OmluY2x1ZGVfZ3JhcGhpY3MoSU1HKCJQREYxMC5Ucm9waWNhbHN0b3JtLnBuZyIpKQpgYGAKClRoZSBbTkhDIGZvcmVjYXN0IGNvbmVdKGh0dHBzOi8vd3d3Lm5oYy5ub2FhLmdvdi9hYm91dGNvbmUuc2h0bWwpIGlzCmRlc2lnbmVkIHNvIHRoYXQgdHdvLXRoaXJkcyBvZiBoaXN0b3JpY2FsIG9mZmljaWFsIGZvcmVjYXN0IGVycm9ycyBvdmVyCmEgNS15ZWFyIHNhbXBsZSBmYWxsIHdpdGhpbiB0aGUgY29uZSBmb3IgYSBwYXJ0aWN1bGFyIHRpbWUgcG9pbnQuLgoKV2hlbiBwdWJsaXNoZWQgaW4gdGhlIG1lZGlhIHRoZXNlIHZpc3VhbGl6YXRpb25zIGFyZSByb3V0aW5lbHkKbWlzaW50ZXJwcmV0ZWQgc29tZXRoaW5nIGxpa2UgdGhpczoKCjwhLS0KaHR0cHM6Ly93d3cuZHJvcGJveC5jb20vc2gvZDFrYjBqZHJoa2I0M2o5L0FBRFRCZlJ2QWgtbXhtU3hCUk5acExKamEvNS5DSEFQVEVSNT9kbD0wJnByZXZpZXc9UERGMTEuU3Rvcm1XUk9OR1NpemUucGRmJnN1YmZvbGRlcl9uYXZfdHJhY2tpbmc9MQotLT4KCmBgYHtyLCBlY2hvID0gRkFMU0V9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKElNRygiUERGMTEuU3Rvcm1XUk9OR1NpemUucG5nIikpCmBgYAoKQSBtb3JlIGVmZmVjdGl2ZSByZXByZXNlbnRhdGlvbiBtaWdodCBiZSBzb21ldGhpbmcgbGlrZSB0aGlzLCBzaG93aW5nCmFuIF9lbnNlbWJsZV8gb2YgcG9zc2libGUgdHJhY2tzOgoKPCEtLQpodHRwczovL3d3dy5kcm9wYm94LmNvbS9zaC9kMWtiMGpkcmhrYjQzajkvQUFEVEJmUnZBaC1teG1TeEJSTlpwTEpqYS81LkNIQVBURVI1P2RsPTAmcHJldmlldz1QREYxMy5TdG9ybUxpbmVzLnBkZiZzdWJmb2xkZXJfbmF2X3RyYWNraW5nPTEKLS0+CgpgYGB7ciwgZWNobyA9IEZBTFNFfQprbml0cjo6aW5jbHVkZV9ncmFwaGljcyhJTUcoIlBERjEzLlN0b3JtTGluZXMucG5nIikpCmBgYAoKQW4gYW5pbWF0ZWQgdmVyc2lvbiBtYXkgYmUgbW9yZSBlZmZlY3RpdmUsIGlmIHRoZSBwcmVzZW50YXRpb24gbWVkaXVtCnBlcm1pdHMuCgpEZXZlbG9waW5nIGJldHRlciB2aXN1YWxpemF0aW9ucyBmb3IgaHVycmljYW5lIGZvcmVjYXN0aW5nLCBlc3BlY2lhbGx5CnRhcmdldGluZyB0aGUgcHVibGljLCBpcyBhbiBhY3RpdmUgYXJlYSBvZiByZXNlYXJjaC4KCgojIyBWaXN1YWxpemluZyBVbmNlcnRhaW50eTogQ2hvY29sYXRlIEJhcnMKCltFeHBlcnQgcmF0aW5nc10oaHR0cDovL2ZsYXZvcnNvZmNhY2FvLmNvbSksIG9uIGEgc2NhbGUgZnJvbSAwIHRvIDUsCmZvciBjaG9jb2xhdGUgYmFycyBtYW51ZmFjdHVyZWQgaW4gc2V2ZXJhbCBjb3VudHJpZXM6CgpgYGB7ciwgZWNobyA9IEZBTFNFfQpkYXRhKGNhY2FvLCBwYWNrYWdlID0gImR2aXouc3VwcCIpCmxpYnJhcnkoY29sb3JzcGFjZSkKY291bnRyaWVzIDwtIGMoIlUuUy5BLiIsICJBdXN0cmlhIiwgIkJlbGdpdW0iLCAiQ2FuYWRhIiwgIlBlcnUiLCAiU3dpdHplcmxhbmQiKQoKY29sODAgPC0gZGVzYXR1cmF0ZShkYXJrZW4oIiMwMDcyQjIiLCAuMiksIC4zKQpjb2w5NSA8LSBkZXNhdHVyYXRlKGxpZ2h0ZW4oIiMwMDcyQjIiLCAuMiksIC4zKQpjb2w5OSA8LSBkZXNhdHVyYXRlKGxpZ2h0ZW4oIiMwMDcyQjIiLCAuNCksIC4zKQpjb2xQIDwtIGNvbDk1CmNvbE0gPC0gIiNENTVFMDAiCgpjMSA8LSBmaWx0ZXIoY2FjYW8sIGxvY2F0aW9uICVpbiUgY291bnRyaWVzKQpjMXN1bXMgPC0gZ3JvdXBfYnkoYzEsIGxvY2F0aW9uKSB8PgogICAgc3VtbWFyaXplKG0gPSBtZWFuKHJhdGluZyksCiAgICAgICAgICAgICAgcyA9IHNkKHJhdGluZyksCiAgICAgICAgICAgICAgbiA9IG4oKSkgfD4KICAgIHVuZ3JvdXAoKQoKYzFDSSA8LSBtdXRhdGUoZGF0YS5mcmFtZShsZXZlbCA9IGMoMC44LCAwLjk1LCAwLjk5KSksCiAgICAgICAgICAgICAgIGRmID0gbGFwcGx5KGxldmVsLAogICAgICAgICAgICAgICAgICAgICAgICAgICBmdW5jdGlvbihsZXYpCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB3aXRoKGMxc3VtcywgewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGggPC0gcyAqIHF0KDEgLSAoMSAtIGxldikgLyAyLCBuIC0gMSkgLwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzcXJ0KG4pCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY2JpbmQoYzFzdW1zLCBkYXRhLmZyYW1lKHhtaW4gPSBtIC0gaCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgeG1heCA9IG0gKyBoKSkKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pKSkgfD4KICAgIHVubmVzdCgiZGYiKQoKZ2dwbG90KGMxLCBhZXMocmF0aW5nLCByZW9yZGVyKGxvY2F0aW9uLCByYXRpbmcpKSkgKwogICAgZ2VvbV9wb2ludChwb3NpdGlvbiA9IHBvc2l0aW9uX2ppdHRlcihoZWlnaHQgPSAwLjMsIHdpZHRoID0gMC4wNSksCiAgICAgICAgICAgICAgIHNpemUgPSAwLjUsIGNvbG9yID0gY29sUCkgKwogICAgIyNnZW9tX3BvaW50KGFlcyhtLCBsb2NhdGlvbiksIGRhdGEgPSBjMXN1bXMsIHNpemUgPSAyLjUsIGNvbG9yID0gY29sTSkgKwogICAgZ2VvbV9zZWdtZW50KGFlcyh4ID0gbSwgeGVuZCA9IG0sCiAgICAgICAgICAgICAgICAgICAgIHkgPSBhcy5pbnRlZ2VyKHJlb3JkZXIobG9jYXRpb24sIG0pKSAtIDAuMywKICAgICAgICAgICAgICAgICAgICAgeWVuZCA9IGFzLmludGVnZXIocmVvcmRlcihsb2NhdGlvbiwgbSkpICsgMC4zKSwKICAgICAgICAgICAgICAgICBsaW5ld2lkdGggPSAyLCBjb2xvciA9IGNvbE0sIGRhdGEgPSBjMXN1bXMpICsKICAgIHlsYWIoIiIpICsKICAgIGdndGl0bGUoIlJhdGluZ3MgZm9yIENob2NvbGF0ZSBCYXJzIiwgIkJhcnMgYXJlIHNhbXBsZSBtZWFucy4iKQpgYGAKClRoZSBzdGFuZGFyZCBkZXZpYXRpb25zIG9mIHRoZSBkYXRhIGRpc3RyaWJ1dGlvbnMgYXJlIGNvbXBhcmFibGUsIGJ1dAp0aGUgbGVuZ3RocyBvZiBjb25maWRlbmNlIGludGVydmFscyBmb3IgdGhlIG1lYW4gdmFyeSBiZWNhdXNlIG9mIHRoZQpkaWZmZXJlbnQgc2FtcGxlIHNpemVzOgoKYGBge3IsIGVjaG8gPSBGQUxTRX0KcCA8LSBnZ3Bsb3QoZmlsdGVyKGMxQ0ksIGxldmVsID09IDAuOTUpLAogICAgICAgICAgICBhZXMobSwgcmVvcmRlcihsb2NhdGlvbiwgbSksIHhtaW4gPSB4bWluLCB4bWF4ID0geG1heCkpICsKICAgIGdlb21fZXJyb3JiYXJoKGhlaWdodCA9IDApICsKICAgIGdlb21fcG9pbnQoc2l6ZSA9IDIuNSwgY29sb3IgPSBjb2xNKSArCiAgICB5bGFiKCIiKSArCiAgICBnZ3RpdGxlKCJDb25maWRlbmNlIEludGVydmFscyBmb3IgdGhlIE1lYW4iLCAiQ29uZmlkZW5jZSBsZXZlbCA5NSUiKQoKcCArIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKDEsIDQpLCBuYW1lID0gIm1lYW4gcmF0aW5nIikKYGBgCgpUaGUgc2FtZSBwbG90IHdpdGggYSByZWR1Y2VkIGhvcml6b250YWwgcmFuZ2U6CgpgYGB7ciwgZWNobyA9IEZBTFNFfQpwICsgc2NhbGVfeF9jb250aW51b3VzKGxpbWl0cyA9IGMoMi41LCAzLjgpLCBuYW1lID0gIm1lYW4gcmF0aW5nIikKYGBgCgpBIG1vcmUgZWxhYm9yYXRlIGRpc3BsYXkgd2l0aCBjb25maWRlbmNlIGludGVydmFscyBhdCBzZXZlcmFsIGxldmVsczoKCmBgYHtyLCBlY2hvID0gRkFMU0V9CiN8IHdhcm5pbmc6IGZhbHNlCiMjIGJhc2VkIG9uIGNvZGUgZm9yIFdpbGtlJ3MgRmlnLiAxNi43CmFycmFuZ2UoYzFDSSwgZGVzYyhsZXZlbCkpIHw+CiAgICBtdXRhdGUobGV2ZWwgPSBwYXN0ZTAoMTAwICogbGV2ZWwsICIlIiksCiAgICAgICAgICAgbG9jYXRpb24gPSByZW9yZGVyKGxvY2F0aW9uLCBtKSkgfD4KICAgIGdncGxvdChhZXMobSwgbG9jYXRpb24sIHhtaW4gPSB4bWluLCB4bWF4ID0geG1heCkpICsKICAgIGdlb21fZXJyb3JiYXJoKGFlcyhzaXplID0gbGV2ZWwsIGNvbG9yID0gbGV2ZWwpLCBoZWlnaHQgPSAwKSArCiAgICBnZW9tX2Vycm9yYmFyaChhZXMoY29sb3IgPSBsZXZlbCksIGhlaWdodCA9IDAuMSkgKwogICAgZ2VvbV9wb2ludChzaXplID0gMi41LCBjb2xvciA9IGNvbE0pICsKICAgIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKDIuNSwgMy44KSwgbmFtZSA9ICJtZWFuIHJhdGluZyIpICsKICAgIHNjYWxlX3NpemVfbWFudWFsKG5hbWUgPSAiY29uZmlkZW5jZSBsZXZlbCIsCiAgICAgICAgICAgICAgICAgICAgICB2YWx1ZXMgPSBjKGA4MCVgID0gMi4yNSwgYDk1JWAgPSAxLjUsIGA5OSVgID0gMC43NSksCiAgICAgICAgICAgICAgICAgICAgICBndWlkZSA9IGd1aWRlX2xlZ2VuZChkaXJlY3Rpb24gPSAiaG9yaXpvbnRhbCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aXRsZS5wb3NpdGlvbiA9ICJ0b3AiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGFiZWwucG9zaXRpb24gPSAiYm90dG9tIikpICsKICAgIHNjYWxlX2NvbG9yX21hbnVhbChuYW1lID0gImNvbmZpZGVuY2UgbGV2ZWwiLAogICAgICAgICAgICAgICAgICAgICAgIHZhbHVlcyA9IGMoYDgwJWAgPSBjb2w4MCwgYDk1JWAgPSBjb2w5NSwgYDk5JWAgPSBjb2w5OSksCiAgICAgICAgICAgICAgICAgICAgICAgZ3VpZGUgPSBndWlkZV9sZWdlbmQoZGlyZWN0aW9uID0gImhvcml6b250YWwiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRpdGxlLnBvc2l0aW9uID0gInRvcCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGFiZWwucG9zaXRpb24gPSAiYm90dG9tIikpICsKCiAgICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSBjKDEsIDAuMDEpLCBsZWdlbmQuanVzdGlmaWNhdGlvbiA9IGMoMSwgMCkpICsKICAgIHlsYWIoIiIpICsKICAgIGdndGl0bGUoIkNvbmZpZGVuY2UgSW50ZXJ2YWxzIGZvciB0aGUgTWVhbiIpCmBgYAoKQ29uZmlkZW5jZSBkZW5zaXRpZXMsIG9yIGNvbmZpZGVuY2UgZGlzdHJpYnV0aW9ucywgYXMgcHJvcG9zZWQgaW4KCj4gQWRyaWFuIFcuIEJvd21hbi4gR3JhcGhpY3MgZm9yIFVuY2VydGFpbnR5LiBKLiBSLiBTdGF0aXN0LiBTb2MuIEEKPiAxODI6MS0xNiwgMjAxOC4gW0xpbmtdKGh0dHBzOi8vcnNzLm9ubGluZWxpYnJhcnkud2lsZXkuY29tL2RvaS9mdWxsLzEwLjExMTEvcnNzYS4xMjM3OSkKCmBgYHtyLCBlY2hvID0gRkFMU0V9CiMjIGJhc2VkIG9uIGNvZGUgZm9yIFdpbGtlJ3MgRmlnLiAxNi45IChlKQpsaWJyYXJ5KHVuZ2V2aXopCmdncGxvdChmaWx0ZXIoYzFDSSwgbGV2ZWwgPT0gMC45NSksCiAgICAgICBhZXMoeCA9IG0sIHkgPSByZW9yZGVyKGxvY2F0aW9uLCBtKSkpICsKICAgIHN0YXRfY29uZmlkZW5jZV9kZW5zaXR5KGFlcyhtb2UgPSB4bWF4IC0gbSwgZmlsbCA9IGFmdGVyX3N0YXQobmRlbnNpdHkpKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGhlaWdodCA9IDAuNywgY29uZmlkZW5jZSA9IDAuOTUsIGFscGhhID0gTkEsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYS5ybSA9IFRSVUUpICsKICAgIGdlb21fc2VnbWVudChhZXMoeCA9IG0sIHhlbmQgPSBtLAogICAgICAgICAgICAgICAgICAgICB5ID0gYXMuaW50ZWdlcihyZW9yZGVyKGxvY2F0aW9uLCBtKSkgLSAwLjM1LAogICAgICAgICAgICAgICAgICAgICB5ZW5kID0gYXMuaW50ZWdlcihyZW9yZGVyKGxvY2F0aW9uLCBtKSkgKyAwLjM1KSwKICAgICAgICAgICAgICAgICBzaXplID0gMiwgY29sb3IgPSBjb2xNKSArCiAgICBzY2FsZV9maWxsX2dyYWRpZW50KGxvdyA9ICIjODFBN0Q2MDAiLCBoaWdoID0gIiMzNDVBN0ZEMCIpICsKICAgIHNjYWxlX3hfY29udGludW91cyhsaW1pdHMgPSBjKDIuNSwgMy44KSwgbmFtZSA9ICJtZWFuIHJhdGluZyIpICsKICAgIHlsYWIoIiIpCmBgYAoKT25lIGRyYXdiYWNrIG9mIGFsbCBvZiB0aGVzZSBtZXRob2RzOgoKPiBUaGUgbGVhc3QgcHJlY2lzZSBtZWFzdXJlbWVudCBkcmF3cyB0aGUgbW9zdCBhdHRlbnRpb24uCgpUaGVzZSBleGFtcGxlcyBmcm9tIFdpbGtlJ3MgYm9vayB1c2UgdGhlIFtgdW5nZXZpemAKcGFja2FnZV0oaHR0cHM6Ly9naXRodWIuY29tL3dpbGtlbGFiL3VuZ2V2aXopIGF2YWlsYWJsZSBvbiBHaXRIdWIuCgpBbm90aGVyIHBhY2thZ2UgcHJvdmlkaW5nIHNvbWUgdG9vbHMgZm9yIHVuY2VydGFpbnR5IHZpc3VhbGl6YXRpb24gaXMKW2BnZ2Rpc3RgIHBhY2thZ2VdKGh0dHBzOi8vbWpza2F5LmdpdGh1Yi5pby9nZ2Rpc3QvKS4KCgojIyBWaXN1YWxpemluZyBVbmNlcnRhaW50eTogT2xkIENhcnMKClVzaW5nIHRoZSB2ZXJ5IG9sZCBgbXRjYXJzYCBkYXRhIHNldCB0byBpbGx1c3RyYXRlIGVzdGltYXRpbmcgYSBzbW9vdGgKcmVsYXRpb25zaGlwOgoKYGBge3IsIG1lc3NhZ2UgPSBGQUxTRSwgZWNobyA9IEZBTFNFfQpwIDwtIGdncGxvdChtdGNhcnMsIGFlcyhkaXNwLCBtcGcpKSArCiAgICBnZW9tX3BvaW50KCkKCnAgKyBnZW9tX3Ntb290aCgpCmBgYAoKQSBkZWZhdWx0IGBnZW9tX3Ntb290aGAgc2hvd3MgYW4gZXN0aW1hdGUgYWxvbmcgd2l0aCBhIHBvaW50LXdpc2UKY29uZmlkZW5jZSBiYW5kLgoKVGhpcyBtYXkgbm90IGdpdmUgdGhlIGJlc3Qgc2Vuc2Ugb2YgdGhlIGpvaW50IHVuY2VydGFpbnR5OiBpZiB0aGUgY3VydmUKaXMgaGlnaGVyIG9uIHNvbWUgcGxhY2VzIGl0IG1heSBuZWVkIHRvIGJlIGxvd2VyIGluIG90aGVycy4KClNob3dpbmcgYW4gX2Vuc2VtYmxlXyBvZiBjdXJ2ZXMgdGhhdCBhbGwgYXJlIHBsYXVzaWJsZSBjYW4gYmUgYSBiZXR0ZXIKY2hvaWNlLgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgbWVzc2FnZSA9IEZBTFNFfQptdHMgPC0gbGFwcGx5KHNlcV9sZW4oMTApLAogICAgICAgICAgICAgIGZ1bmN0aW9uKGkpIG11dGF0ZShzYW1wbGVfZnJhYyhtdGNhcnMsIDEsIHJlcGxhY2UgPSBUUlVFKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2FtcGxlID0gaSkpIHw+CiAgICBiaW5kX3Jvd3MoKQoKcDIgPC0gcCArCiAgICBnZW9tX3Ntb290aChjb2xvciA9IE5BKSArCiAgICBnZW9tX3Ntb290aChhZXMoZ3JvdXAgPSBzYW1wbGUpLAogICAgICAgICAgICAgICAgc2UgPSBGQUxTRSwgc2l6ZSA9IDAuMywgY29sb3IgPSAiIzMzNjZGRiIsIGRhdGEgPSBtdHMpCnAyCmBgYAoKVGhpcyBhcHByb2FjaCB3YXMgc2hvd24gZWFybGllciBmb3IgdmlzdWFsaXppbmcgcG9zc2libGUgaHVycmljYW5lIHBhdGhzLgoKVGhpcyBlbnNlbWJsZSBpcyBnZW5lcmF0ZWQgdXNpbmcgYSBfY2FzZS1iYXNlZCBib290c3RyYXBfLgoKVGhlc2UgcGxvdHMgYXJlIGNhbGxlZCBfZW5zZW1ibGUgcGxvdHNfIChhbHNvIHNwYWdoZXR0aSBwbG90cywgZm9yCm9idmlvdXMgcmVhc29ucykuCgpJZiBhbmltYXRpb24gaXMgYXZhaWxhYmxlLCBhbiBhbHRlcm5hdGl2ZSBpcyB0byBzaG93IHRoZSBjdXJ2ZXMgb25lIGF0CmEgdGltZSBpbiBhbiBhbmltYXRpb24uCgpgYGB7ciwgY2FjaGUgPSBUUlVFLCBtZXNzYWdlID0gRkFMU0UsIGVjaG8gPSBGQUxTRX0KbGlicmFyeShnZ2FuaW1hdGUpCmFuaW1hdGUocDIgKwogICAgICAgIHRyYW5zaXRpb25fc3RhdGVzKHNhbXBsZSwgdHJhbnNpdGlvbl9sZW5ndGggPSAyLCBzdGF0ZV9sZW5ndGggPSAxKSkKYGBgCgpBZ2FpbiwgYSBib290c3RyYXAgaXMgdXNlZCB0byBwcm9kdWNlIHRoZSBlc3RpbWF0ZXMuCgpUaGlzIGlzIGFuIGV4YW1wbGUgb2YgYSBfaHlwb3RoZXRpY2FsIG91dGNvbWVzIHBsb3RfLCBvciBfSE9QXywgYXMKaW50cm9kdWNlZCBpbgoKPiBIdWxsbWFuLCBKZXNzaWNhLCBQYXVsIFJlc25pY2ssIGFuZCBFeXRhbiBBZGFyLiAiSHlwb3RoZXRpY2FsCj4gb3V0Y29tZSBwbG90cyBvdXRwZXJmb3JtIGVycm9yIGJhcnMgYW5kIHZpb2xpbiBwbG90cyBmb3IgaW5mZXJlbmNlcwo+IGFib3V0IHJlbGlhYmlsaXR5IG9mIHZhcmlhYmxlIG9yZGVyaW5nLiIgUExPUyBPTkUgMTAsIG5vLiAxMSAoMjAxNSkuCgoKIyMgRGF0YSBRdWFsaXR5IGFuZCBJbnRlZ3JpdHkKCkEgdmlzdWFsaXphdGlvbiBjYW4gYWNjdXJhdGVseSByZWZsZWN0IGRhdGEgYnV0IHN0aWxsIGJlIG1pc2xlYWRpbmcgaWYKdGhlIGRhdGEgYXJlIGZhdWx0eS4KCkEgW05ZIFRpbWVzCmFydGljbGVdKGh0dHBzOi8vd3d3Lm55dGltZXMuY29tLzIwMjEvMDUvMDMvaGVhbHRoL2NvdmlkLWhlcmQtaW1tdW5pdHktdmFjY2luZS5odG1sKQpmcm9tIE1heSAyMDIxIHNob3dzIGEgY2hvcm9wbGV0aCBtYXAgb2YgdGhlIGVzdGltYXRlZCBzaGFyZSBvZiBhZHVsdHMKd2hvIHdvdWxkICJkZWZpbml0ZWx5IiBvciAicHJvYmFibHkiIGdldCB0aGUgQ09WSUQtMTkgdmFjY2luZS4KCkN1dG9mZnM6IDQ5ICA2MCAgIDY1ICA3MCAgNzUgIDgwICA5MSAlCgpgYGB7ciwgZWNobyA9IEZBTFNFLCBvdXQud2lkdGggPSA1MDB9CmtuaXRyOjppbmNsdWRlX2dyYXBoaWNzKElNRygibWFwLTEwNTAucG5nIikpCmBgYAoKVGhlIG1hcCBtYXkgYWNjdXJhdGVseSByZWZsZWN0IHRoZSBlc3RpbWF0ZXMsIGJ1dCB0aGUgZXN0aW1hdGVzIGhhdmUKb2J2aW91cyBwcm9ibGVtcy4KClRoZSBkYXRhIHVzZWQgZm9yIHRoZSBtYXAgYXJlIGF2YWlsYWJsZQpbaGVyZV0oaHR0cHM6Ly9hc3BlLmhocy5nb3YvcmVwb3J0cy92YWNjaW5lLWhlc2l0YW5jeS1jb3ZpZC0xOS1zdGF0ZS1jb3VudHktbG9jYWwtZXN0aW1hdGVzKS4KCkRpc2N1c3Npb25zIG9uIHNvY2lhbCBtZWRpYSBzdWdnZXN0IHRoYXQgdGhlIHN0YXRlIGxldmVsIGRhdGEgbWF5IGJlCm1vcmUgcmVhc29uYWJsZToKCjwhLS0gIyMgbm9saW50IHN0YXJ0OiBsaW5lX2xlbmd0aCAtLT4KPCEtLSBodHRwczovL3R3aXR0ZXIuY29tL2N0X2JlcmdzdHJvbS9zdGF0dXMvMTM5MDUwOTI5ODM4ODY2MDIzMT9zPTExIC0tPgoKYGBge3IsIGVjaG8gPSBGQUxTRSwgZmlnLndpZHRoID0gMTAsIGZpZy5oZWlnaHQgPSAxMH0KbGlicmFyeSh0aWR5dmVyc2UpCmRhdGEgPC0gcmVhZC5jc3YoImh0dHA6Ly93d3cuc3RhdC51aW93YS5lZHUvfmx1a2UvZGF0YS9WYWNjaW5lSGVzaXRhbmN5LTIwMjEtMDQtMDYuY3N2IikgfD4KICAgIHNldE5hbWVzKGMoImZpcHMiLCAic3RhdGUiLCAiSGVzaXRhbnQiLCAiU3Ryb25nbHlIZXNpdGFudCIpKSB8PgogICAgbXV0YXRlKGFjcm9zcygzOjQsIHBhcnNlX251bWJlcikpIHw+CiAgICBtdXRhdGUoV2lsbGluZyA9IDEwMCAtIEhlc2l0YW50IC0gU3Ryb25nbHlIZXNpdGFudCkKCm1hcCA8LSB1c21hcDo6dXNfbWFwKCkgfD4KICAgIG11dGF0ZShmaXBzID0gYXMubnVtZXJpYyhmaXBzKSkKCm1hcF9kYXRhIDwtIGxlZnRfam9pbihtYXAsIGRhdGEsICJmaXBzIikKCmdncGxvdChtYXBfZGF0YSwgYWVzKGZpbGwgPSBXaWxsaW5nKSkgKwogICAgZ2VvbV9zZigpICsKICAgIGdndGhlbWVzOjp0aGVtZV9tYXAoKSArCiAgICBzY2FsZV9maWxsX2Rpc3RpbGxlcihwYWxldHRlID0gIlB1cnBsZXMiLAogICAgICAgICAgICAgICAgICAgICAgICAgZGlyZWN0aW9uID0gMSwKICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVscyA9IHNjYWxlczo6bGFiZWxfcGVyY2VudChzY2FsZSA9IDEpLAogICAgICAgICAgICAgICAgICAgICAgICAgZ3VpZGUgPSBndWlkZV9jb2xvcmJhcih0aXRsZS5oanVzdCA9IDAuNSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGl0bGUucG9zaXRpb24gPSAidG9wIikpICsKICAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJ0b3AiLAogICAgICAgICAgbGVnZW5kLmp1c3RpZmljYXRpb24gPSAiY2VudGVyIiwKICAgICAgICAgIGxlZ2VuZC50aXRsZSA9IGVsZW1lbnRfdGV4dChoanVzdCA9IDAuNSksCiAgICAgICAgICBwbG90LnRpdGxlID0gZWxlbWVudF90ZXh0KGhqdXN0ID0gMC41KSwKICAgICAgICAgIHBsb3Quc3VidGl0bGUgPSBlbGVtZW50X3RleHQoaGp1c3QgPSAwLjUpKSArCiAgICBsYWJzKHRpdGxlID0gIlVuZXZlbiBXaWxsaW5nbmVzcyB0byBHZXQgVmFjY2luYXRlZCBDb3VsZCBBZmZlY3QgSGVyZCBJbW11bml0eSIsCiAgICAgICAgIHN1YnRpdGxlID0gIkluIHNvbWUgcGFydHMgb2YgdGhlIFVuaXRlZCBTdGF0ZXMsIGlub2N1bGF0aW9uIHJhdGVzIG1heSBub3QgcmVhY2ggdGhlIHRocmVzaG9sZCBuZWVkZWRcbnRvIHByZXZlbnQgdGhlIGNvcm9uYXZpcnVzIGZyb20gc3ByZWFkaW5nIGVhc2lseS4iLAogICAgICAgICBjYXB0aW9uID0gIkRhdGE6IGh0dHBzOi8vYXNwZS5oaHMuZ292L3BkZi1yZXBvcnQvdmFjY2luZS1oZXNpdGFuY3kiLAogICAgICAgICBmaWxsID0gIkVzdGltYXRlZCBzaGFyZSBvZiBhZHVsdHMgd2hvIHdvdWxkXG5cImRlZmluaXRlbHlcIiBvciBcInByb2JhYmx5XCIgZ2V0IHRoZSB2YWNjaW5lIikKYGBgCjwhLS0gIyMgbm9saW50IGVuZCAtLT4KCjwhLS0gbm90IGJlaW5nIGFibGUgdG8gY2VudGVyIGEgbG9uZyBsZWdlbmQgdGlsdGUgc2VlbXMgdG8gYmUgYQpjdXJyZW50IGdncGxvdCBidWcgLS0+Cgo8ZGl2IGlkPSJkYXRhLXNjaWVuY2UtZXRoaWNzIj48L2Rpdj4KCiMjIERhdGEgU2NpZW5jZSBFdGhpY3MKClNvbWUgaXNzdWVzOgoKKiBEYXRhIG1pc3JlcHJlc2VudGF0aW9uCgoqIERhdGEgZmFsc2lmaWNhdGlvbgoKKiBEYXRhIHByaXZhY3kKCiogRGF0YSBzY3JhcGluZyBhbmQgdGVybXMgb2YgdXNlCgoqIEFsZ29yaXRobWljIGJpYXMKClNvbWUgcmVmZXJlbmNlczoKCjwhLS0gaHR0cHM6Ly9hcnhpdi5vcmcvYWJzLzE5MDguMDYxNjYgLS0+CgoqIFtEYXRhIHNjaWVuY2UKICBldGhpY3NdKGh0dHBzOi8vbWRzci1ib29rLmdpdGh1Yi5pby9tZHNyMmUvY2gtZXRoaWNzLmh0bWwpIGNoYXB0ZXIKICBpbjogQmVuamFtaW4gUy4gQmF1bWVyLCBEYW5pZWwgVC4gS2FwbGFuLCBhbmQgTmljaG9sYXMgSi4gSG9ydG9uCiAgKDIwMjEpICAKICBbX01vZGVybiBEYXRhIFNjaWVuY2Ugd2l0aCBSLCAybmQgZWRpdGlvbl9dKGh0dHBzOi8vbWRzci1ib29rLmdpdGh1Yi5pby9tZHNyMmUvKS4KCiogW0RhdGEgc2NpZW5jZSBldGhpY3NdKGh0dHBzOi8vZGF0YXNjaWVuY2Vib3gub3JnLzAyLWV0aGljcy5odG1sKSBzZWN0aW9uIG9mCiAgdGhlIG9ubGluZSBib29rCiAgW0RhdGEgU2NpZW5jZSBpbiBhIEJveF0oaHR0cHM6Ly9kYXRhc2NpZW5jZWJveC5vcmcvaW5kZXguaHRtbCkKICBieSBNaW5lIMOHZXRpbmtheWEtUnVuZGVsLi4KCiogQWxiZXJ0byBDYWlybyAoMjAxOSkgX0hvdyBDaGFydHMgTGllOiBHZXR0aW5nIFNtYXJ0ZXIgYWJvdXQgVmlzdWFsCiAgSW5mb3JtYXRpb25fLCBXLiBXLiBOb3J0b24gJiBDb21wYW55LgoKPGRpdiBpZD0icGxvdC1hbm5vdGF0aW9uLXBsb3QtZW5zZW1ibGVzLWFuZC1kYXNoYm9hcmRzIj48L2Rpdj4KCgojIyBQbG90IEFubm90YXRpb24sIFBsb3QgRW5zZW1ibGVzLCBhbmQgRGFzaGJvYXJkcwoKUGxvdCBhbm5vdGF0aW9ucyBjYW4gY3JlYXRlIHBvcG91dCBhbmQgaGVscCBmb2N1cyB0aGUgdmlld2VyJ3MKYXR0ZW50aW9uLgoKVGhleSBtYXkgYmUgaW5jcmVhc2luZ2x5IGltcG9ydGFudCBhcyBpbWFnZXMgYXJlIHNoYXJlZCBvbiBsaW5lCndpdGhvdXQgY29udGV4dC4KCkhlcmUgaXMgYW4gZXhhbXBsZXMgZm9yIHRoZSBgbXBnYCBkYXRhOgoKPCEtLSAjIyBub2xpbnQgc3RhcnQ6IGxpbmVfbGVuZ3RoIC0tPgpgYGB7ciwgZWNobyA9IEZBTFNFfQojfCB3YXJuaW5nOiBmYWxzZQpsaWJyYXJ5KGdnZm9yY2UpCmdncGxvdChtcGcsIGFlcyhkaXNwbCwgaHd5KSkgKwogICAgZ2VvbV9wb2ludCgpICsKICAgIGdlb21fbWFya19odWxsKGFlcyhmaWx0ZXIgPSBjbGFzcyA9PSAiMnNlYXRlciIpLAogICAgICAgICAgICAgICAgICAgZmlsbCA9ICJibHVlIiwKICAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uID0gIjItU2VhdGVycyBoYXZlIGhpZ2ggZGlzcGxhY2VtZW50IHZhbHVlcywgYnV0IGFsc28gaGlnaCBmdWVsIGVmZmljaWVuY3kgZm9yIHRoZWlyIGRpc3BsYWNlbWVudC4iKSArCiAgICBnZW9tX21hcmtfcmVjdChhZXMoZmlsdGVyID0gaHd5ID4gNDApLAogICAgICAgICAgICAgICAgICAgZmlsbCA9ICJncmVlbiIsCiAgICAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbiA9ICJUaGVzZSBhcmUgVm9sa3N3YWdlbnMiKSArCiAgICBnZW9tX21hcmtfY2lyY2xlKGFlcyhmaWx0ZXIgPSBod3kgPT0gMTIpLAogICAgICAgICAgICAgICAgICAgICBmaWxsID0gInJlZCIsCiAgICAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uID0gIlRocmVlIHBpY2t1cHMgYW5kIGFuIHN1di4iKQpgYGAKPCEtLSAjIyBub2xpbnQgZW5kIC0tPgoKCiMjIFBsb3QgRW5zZW1ibGVzOiBDb2ZmZWUKCkl0IGlzIG9mdGVuIHVzZWZ1bCB0byB1c2Ugc2V2ZXJhbCBncmFwaGljcyB0byBwcmVzZW50IGFuIGFuYWx5c2lzLgoKQ29sbGVjdGlvbnMgb2YgcmVsYXRlZCBncmFwaHMgYXJlIHNvbWV0aW1lcyBjYWxsZWQgX2Vuc2VtYmxlIGdyYXBoaWNzXy4KCk9uIGxpbmUgcHJlc2VudGF0aW9ucyBvZiBhbmFseXNlcyBpbnZvbHZpbmcgbXVsdGlwbGUgdmlzdWFsaXphdGlvbnMKYW5kLCB0eXBpY2FsbHksIHNvbWUgaW50ZXJhY3RpdmUgZmVhdHVyZXMgYXJlIGFsc28gY2FsbGVkCl9kYXNoYm9hcmRzXy4KClRvIGFpZCB0aGUgdmlld2VyIGl0IGlzIHVzdWFsbHkgYmVzdCB0byBkZXNpZ24gdGhlc2UgdmlzdWFsaXphdGlvbnMKdG9nZXRoZXIsIHdpdGggY29tbW9uIGF4aXMgY2hvaWNlcyBhbmQgY29sb3IgbWFwcGluZ3MuCgpGaWcgMTIuMSBpbiBVbndpbiAoMjAxNSkgcHJvdmlkZXMgYSBzaW1wbGUgZXhhbXBsZToKCjwhLS0gIyMgbm9saW50IHN0YXJ0OiBsaW5lX2xlbmd0aCAtLT4KYGBge3IsIG91dC53aWR0aD0gIjY1JSIsIG91dC5leHRyYT0nc3R5bGU9ImZsb2F0OnJpZ2h0OyBwYWRkaW5nOjEwcHgiJ30KI3wgZmlnLndpZHRoOiA5CiN8IGZpZy5oZWlnaHQ6IDYuNQojfCBmaWctcmV0aW5hOiB0cnVlCiN8IGZpZy1hbHQ6ICJBIGRhc2hib2FyZCB3aXRoIHRocmVlIHBsb3RzLiBBIGJhciBjaGFydCBzaG93cyB0aGVyZSBhcmUgYWJvdXQgNCB0aW1lcyBhcyBtYW55IEFyYWJpY2Egc2FtcGxlcyBhZCBSdWJ1c3RhIHNhbXBsZXMuIEEgc2NhdHRlcnBsb3Qgb2YgQ2FmZmVpbmUgYWdhaW5zdCBGYXQgY29udGVudCBzaG93cyBjbGVhciBzZXBhcmF0aW9uIG9mIHRoZSB0d28gZ3JvdXBzLiBBIHBhcmFsbGVsIGNvb3JkaW5hdGVzIHBsb3Qgc2hvd3MgdGhlIDEyIHZhbHVlcyBtZWFzdXJlZCBvbiBlYWNoIGdyb3VwLiIKCmxpYnJhcnkoZ2dwbG90MikKbGlicmFyeShHR2FsbHkpCmxpYnJhcnkoZ3JpZEV4dHJhKQoKY29mZmVlX3RobSA8LSB0aGVtZSh0ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxNCkpCgpkYXRhKGNvZmZlZSwgcGFja2FnZSA9ICJwZ21tIikKY29mZmVlIDwtIHdpdGhpbihjb2ZmZWUsIFR5cGUgPC0gaWZlbHNlKFZhcmlldHkgPT0gMSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJBcmFiaWNhIiwgIlJvYnVzdGEiKSkKbmFtZXMoY29mZmVlKSA8LSBhYmJyZXZpYXRlKG5hbWVzKGNvZmZlZSksIDgpCmEgPC0gZ2dwbG90KGNvZmZlZSwgYWVzKHggPSBUeXBlKSkgKyBnZW9tX2JhcihhZXMoZmlsbCA9IFR5cGUpKSArCiAgICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCJncmV5NzAiLCAicmVkIikpICsKICAgIGd1aWRlcyhmaWxsID0gIm5vbmUiKSArIHlsYWIoIiIpICsKICAgIGNvZmZlZV90aG0KYiA8LSBnZ3Bsb3QoY29mZmVlLCBhZXMoeCA9IEZhdCwgeSA9IENhZmZpbmUsIGNvbG91ciA9IFR5cGUpKSArCiAgICBnZW9tX3BvaW50KHNpemUgPSAzKSArCiAgICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoImdyZXk3MCIsICJyZWQiKSkgKwogICAgY29mZmVlX3RobQpjIDwtIGdncGFyY29vcmQoY29mZmVlW29yZGVyKGNvZmZlZSRUeXBlKSwgXSwgY29sdW1ucyA9IDMgOiAxNCwKICAgICAgICAgICAgICAgIGdyb3VwQ29sdW1uID0gIlR5cGUiLCBzY2FsZSA9ICJ1bmltaW5tYXgiKSArCiAgICB4bGFiKCIiKSArIHlsYWIoIiIpICsKICAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikgKwogICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKCJncmV5IiwgInJlZCIpKSArCiAgICB0aGVtZShheGlzLnRpY2tzLnkgPSBlbGVtZW50X2JsYW5rKCksCiAgICAgICAgICBheGlzLnRleHQueSA9IGVsZW1lbnRfYmxhbmsoKSkgKwogICAgY29mZmVlX3RobQpncmlkLmFycmFuZ2UoYXJyYW5nZUdyb2IoYSwgYiwgbmNvbCA9IDIsIHdpZHRocyA9IGMoMSwgMikpLAogICAgICAgICAgICAgYywgbnJvdyA9IDIpCmBgYAo8IS0tICMjIG5vbGludCBlbmQgLS0+CgpEYXRhIG9uIHRoZSBjaGVtaWNhbCBjb21wb3NpdGlvbiBvZiBjb2ZmZWUgc2FtcGxlcyBjb2xsZWN0ZWQgZnJvbQphcm91bmQgdGhlIHdvcmxkLCBjb21wcmlzaW5nIDQzIHNhbXBsZXMgZnJvbSAyOSBjb3VudHJpZXMuIEVhY2ggc2FtcGxlCmlzIGVpdGhlciBvZiB0aGUgQXJhYmljYSBvciBSb2J1c3RhIHZhcmlldHkuIFR3ZWx2ZSBvZiB0aGUgdGhpcnRlZW4KY2hlbWljYWwgY29uc3RpdHVlbnRzIHJlcG9ydGVkIGluIHRoZSBzdHVkeSBhcmUgZ2l2ZW4uICBUaGUgb21pdHRlZAp2YXJpYWJsZSBpcyB0b3RhbCBjaGxvcm9nZW5pYyBhY2lkOyBpdCBpcyBnZW5lcmFsbHkgdGhlIHN1bSBvZiB0aGUKY2hsb3JvZ2VuaWMsIG5lb2NobG9yb2dlbmljIGFuZCBpc29jaGxvcm9nZW5pYyBhY2lkIHZhbHVlcy4KClN0cmV1bGksIEguICgxOTczKS4gRGVyIGhldXRpZ2Ugc3RhbmQgZGVyIGthZmZlZWNoZW1pZS4gSW4KX0Fzc29jaWF0aW9uIFNjaWVudGlmaXF1ZSBJbnRlcm5hdGlvbmFsIGR1IENhZmUsIDZ0aCBJbnRlcm5hdGlvbmFsCkNvbGxvcXVpdW0gb24gQ29mZmVlIENoZW1pc3J0eV8sIEJvZ2F0YSwgQ29sdW1iaWEsIHBwLiAgNjEtNzIuCgoKIyMgTWFraW5nIGEgUG9pbnQgYW5kIFRlbGxpbmcgYSBTdG9yeQoKSW4gYSByZXBvcnQsIG1ha2Ugc3VyZSBlYWNoIHBsb3QgaGFzIGEgcG9pbnQgYW5kIG1ha2VzIGl0cyBwb2ludC4KCk1ha2Ugc3VyZSB0byB0aGluayBhYm91dDoKCiogYXhpcyBsYWJlbHM7CgoqIHRpdGxlcyBhbmQgc3VidGl0bGVzOwoKKiBjYXB0aW9uczsKCiogaGlnaGxpZ2h0aW5nIGtleSBmZWF0dXJlczsgPCEtLSBnZ2hpZ2hsaWdodCAtLT4KCiogYWNjZXNzaWJpbGl0eSAoZS5nLiBjb2xvciBjaG9pY2U7IGFsdC10ZXh0KS4KCkl0IGlzIG9mdGVuIGdvb2QgdG8gbWFrZSBzdXJlIGEgZmlndXJlIGNhbiBzdGFuZCBvbiBpdHMgb3duCndpdGhvdXQgYXNraW5nIHRoZSByZWFkZXIgdG8gc2VhcmNoIHRoZSB0ZXh0IGZvciBleHBsYW5hdGlvbnMuCgpDb21tdW5pY2F0aW5nIHdpdGggZGF0YSBpcyBsaWtlIHRlbGxpbmcgYSBzdG9yeSwgd2l0aCBhIHN0YXJ0aW5nCnBvaW50LCBhIGpvdXJuZXksIGFuZCBhbiBlbmQuCgpTb21ldGltZXMgYSBzaW5nbGUgdmlzdWFsaXphdGlvbiBjYW4gY2FwdHVyZSB0aGUgZnVsbCBzdG9yeS4KCk1vcmUgb2Z0ZW4sIHNldmVyYWwgdmlzdWFsaXphdGlvbnMgd2lsbCBiZSBuZWVkZWQuCgpPZnRlbiBpdCBpcyBnb29kIHRvOgoKKiBzdGFydCB3aXRoIGEgaGlnaCBsZXZlbCBvdmVydmlldzsKCiogc2hvdyBob3cgdG8gbG9vayBhdCBzb21lIHBhcnRpY3VsYXIgY2FzZXMsIGUuZy4gd2l0aCBhIHNpbmdsZSBwbG90OwoKKiBidWlsZCB1cCB0byBhIG1vcmUgY29tcGxldGUgYW5hbHlzaXMsIGUuZy4gd2l0aCBhIG11bHRpLXBhbmVsIHBsb3QuCgpXaXRoIG11bHRpcGxlIHZpc3VhbGl6YXRpb25zIGl0IGlzIGdvb2QgbWFrZSBzdXJlIHRoYXQ6CgoqIGVhY2ggb25lIHdvcmtzIHdlbGwgb24gaXRzIG93bjsKCiogdGhleSB3b3JrIHdlbGwgdG9nZXRoZXIgKGUuZy4gdXNlIGNvbnNpc3RlbnQgc3R5bGluZywgY29sb3JzKS4KClRoZXJlIGlzIGEgW2NoYXB0ZXIgb2YgV2lsa2UsCjIwMTldKGh0dHBzOi8vY2xhdXN3aWxrZS5jb20vZGF0YXZpei90ZWxsaW5nLWEtc3RvcnkuaHRtbCkgd2l0aCBtb3JlCmFkdmljZSBvbiB0aGlzLgoKQSByZWNlbnQgYm9vayBsZW5ndGggdHJlYXRtZW50IGlzCgo+IERlYm9yYWggTm9sYW4gYW5kIFNhcmEgU3RvdWR0ICgyMDIxKSBfQ29tbXVuaWNhdGluZyB3aXRoIERhdGFfLAo+IE94Zm9yZCBVbml2ZXJpdHkgUHJlc3MuCgoKIyMgV3JhcHBpbmcgVXAKClNvbWUgb2YgdGhlIGFyZWFzIHdlIGNvdmVyZWQ6CgoKIyMjIFZpc3VhbGl6YXRpb24KCk1hbnkgZGlmZmVyZW50IHR5cGVzIG9mIGdyYXBocy4KCiogU3RyZW5ndGhzLCB3ZWFrbmVzc2VzLgoqIFBpdGZhbGxzLgoqIFNjYWxhYmlsaXR5LgoqIENyZWF0aW5nIHRoZXNlIGdyYXBocyBpbiBSLgoKUGVyY2VwdGlvbgoKKiBDaGFubmVscyBhbmQgbWFwcGluZ3M7IHJlbGF0aXZlIGVmZmVjdGl2ZW5lc3MuCiogVXNpbmcgdG8gYXNzZXNzLCBkZXNpZ24gdmlzdWFsaXphdGlvbnMuCiogRWZmZWN0aXZlIHVzZSBvZiBjb2xvci4KCkEgbGl0dGxlIG9uIGludGVyYWN0aW9uLCBhbmltYXRpb24uCgpFbXBoYXNpcyBvbiB0ZWNobmlxdWVzIHVzZWZ1bCBmb3IgZXhwbG9yYXRpb24sIHNjaWVudGlmaWMgcmVwb3J0aW5nLgoKCiMjIyBEYXRhIFRlY2hub2xvZ2llcwoKUmVhZGluZyBkaWZmZXJlbnQgZGF0YSBmb3JtYXRzLgoKU2NyYXBpbmcgZGF0YSBmcm9tIHRoZSB3ZWIuCgpDbGVhbmluZyBkYXRhLgoKUmVhcnJhbmdpbmcgZGF0YSBmb3IgYW5hbHlzaXMuCgpNZXJnaW5nIGRhdGEgZnJvbSBzZXZlcmFsIHNvdXJjZXMuCgoKIyMjIFJlcHJvZHVjaWJsZSByZXNlYXJjaCB0b29scwoKYHJtYXJrZG93bmAgZm9yIGludGVncmF0aW5nIGNvZGUgYW5kIHJlcG9ydGluZy4KClZlcnNpb24gY29udHJvbCwgYGdpdGAsIGBHaXRMYWJgLgoKCiMjIExlYXJuaW5nIE1vcmUKCkNsYXNzIG5vdGVzIHdpbGwgcmVtYWluIGF2YWlsYWJsZSwgaW4gc29tZSBmb3JtLCBhdCB0aGUgY2xhc3Mgd2ViIHNpdGUuCgpTb21lIGJvb2tzIHRvIGxvb2sgYXQ6CgoqIEFsYmVydG8gQ2Fpcm8gKDIwMTkpIF9Ib3cgQ2hhcnRzIExpZTogR2V0dGluZyBTbWFydGVyIGFib3V0IFZpc3VhbAogIEluZm9ybWF0aW9uXywgVy4gVy4gTm9ydG9uICYgQ29tcGFueS4KCiogQ2xhdXMgTy4gV2lsa2UgKDIwMTkpIFtfRnVuZGFtZW50YWxzIG9mIERhdGEKICBWaXN1YWxpemF0aW9uX10oaHR0cHM6Ly9jbGF1c3dpbGtlLmNvbS9kYXRhdml6LyksIE/igJlSZWlsbHksCiAgSW5jLiAoW0Jvb2sgc291cmNlIG9uCiAgR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vY2xhdXN3aWxrZS9kYXRhdml6KTsgW3N1cHBvcnRpbmcKICBtYXRlcmlhbHMgb24gR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vY2xhdXN3aWxrZS9kdml6LnN1cHApKQoKKiBLaWVyYW4gSGVhbHkgKDIwMTgpIFtfRGF0YSBWaXN1YWxpemF0aW9uOiBBIHByYWN0aWNhbAogIGludHJvZHVjdGlvbl9dKGh0dHBzOi8vc29jdml6LmNvLyksIFByaW5jZXRvbgoKKiBXaW5zdG9uIENoYW5nICgyMDE4KSBbX1IgR3JhcGhpY3MgQ29va2Jvb2tfLCAybmQKICBlZGl0aW9uXShodHRwczovL3ItZ3JhcGhpY3Mub3JnLyksIE/igJlSZWlsbHkuIChbQm9vayBzb3VyY2Ugb24KICBHaXRIdWJdKGh0dHBzOi8vZ2l0aHViLmNvbS93Y2gvcmdjb29rYm9vaykpCgpTb21lIGJsb2dzIHRvIGNoZWNrIG91dDoKCiogW0p1bmsgQ2hhcnRzXShodHRwczovL2p1bmtjaGFydHMudHlwZXBhZC5jb20vKQoKKiBbVGhlIEZ1bmN0aW9uYWwgQXJ0IEJsb2ddKGh0dHA6Ly93d3cudGhlZnVuY3Rpb25hbGFydC5jb20vKQoKKiBbRmxvd2luZyBEYXRhXShodHRwczovL2Zsb3dpbmdkYXRhLmNvbS8pCgpLZWVwIGEgY3JpdGljYWwgZXllIG91dCBmb3IgZ29vZCAoYW5kIG5vdCBzbyBnb29kKSB1c2VzIG9mIGRhdGEKdmlzdWFsaXphdGlvbiBpbiB0aGUgbWVkaWEuCg==