1 Analiza opisowa

Jednym ze sposobów zrozumienia, jak działa rząd miasta, jest spojrzenie na to, kogo zatrudnia i jak jego pracownicy są wynagradzani. Dane te zawierają nazwiska, nazwę stanowiska i wynagrodzenie pracowników miasta San Francisco w ujęciu rocznym od 2011 do 2014 roku.

Oto kilka pomysłów na eksplorację danych:

  • Jak zmieniały się wynagrodzenia w czasie między różnymi grupami ludzi?

  • Jak płaca podstawowa, wynagrodzenie za nadgodziny i świadczenia są rozdzielane pomiędzy różne grupy?

  • Czy w tym zestawie danych istnieją dowody na dyskryminację płacową ze względu na płeć?

  • Jak przydzielany jest budżet w zależności od grupy i zakresu obowiązków?

# wymiary ramki:
dim(salaries)
## [1] 148654     13
# nazwy kolumn:
names(salaries)
##  [1] "Id"               "EmployeeName"     "JobTitle"         "BasePay"         
##  [5] "OvertimePay"      "OtherPay"         "Benefits"         "TotalPay"        
##  [9] "TotalPayBenefits" "Year"             "Notes"            "Agency"          
## [13] "Status"

1.1 Histogramy

hist(salaries$TotalPay,main="Total Pay", xlab="Pay (in dollars)")
abline(v = mean(salaries$TotalPay),lty="dashed")
abline(v = median(salaries$TotalPay))
legend("topright", legend=c("Mediana","Średnia"),lty=c("solid","dashed"))

par(mfrow=c(2,2))
hist(salaries$TotalPay,main="Total Pay, default breaks", xlab="Pay (in dollars)")
hist(salaries$TotalPay,main="Total Pay, breaks=100", xlab="Pay (in dollars)", breaks=100)
hist(salaries$TotalPay,main="Total Pay, breaks=1000", xlab="Pay (in dollars)",breaks=1000)

hist(salaries$TotalPay,main="Total Pay, Zoomed-in", xlab="Pay (in dollars)", xlim=c(0,1e5), breaks=1000)

salaries2 <- subset(salaries, JobTitle=="Firefighter" & Status=="FT")
dim(salaries2)
## [1] 738  13
par(mfrow=c(2,2))
hist(salaries2$TotalPay,main="Firefighters, default breaks", xlab="Pay (in dollars)")
hist(salaries2$TotalPay,main="Firefighters, breaks=30", xlab="Pay (in dollars)", breaks=30)
hist(salaries2$TotalPay,main="Firefighters, breaks=100", xlab="Pay (in dollars)", breaks=100)
hist(salaries2$TotalPay,main="Firefighters, breaks=1000", xlab="Pay (in dollars)",breaks=1000)

1.2 Wykresy pudełkowe

par(mfrow=c(1,1))
boxplot(salaries$TotalPay,main="Total Pay, breaks=1000", ylab="Pay (in dollars)")

2 Estymacja funkcji gęstości

Pierwszy raport dotyczy nieparametrycznej estymacji gęstości. Klasycznym nieparametrycznym estymatorem gęstości jest histogram, który dostarcza nieciągłe i stałe oszacowania. W tym raporcie skupiono się na niektórych alternatywach, które zapewniają ciągłe lub nawet gładkie oszacowania zamiast.

Metody kernelowe stanowią ważną klasę gładkich estymatorów gęstości i zaimplementowane są przez funkcję R density(). Estymatory te są w zasadzie tylko lokalnie ważonymi średnimi, a ich obliczenie jest stosunkowo proste w teorii. W praktyce, różne wybory sposobu implementacji obliczeń mogą jednak mieć duży wpływ na rzeczywisty czas obliczeń, a implementację kernelowych estymatorów gęstości zilustruje trzy punkty:

  • jeśli to możliwe, wybierz wektoryzowane implementacje w R,
  • jeśli niewielka strata w dokładności jest do zaakceptowania, przybliżone rozwiązanie może być o rzędy wielkości szybsze niż implementacja literalna,
  • czas potrzebny do numerycznej oceny różnych funkcje elementarne może bardzo zależeć od funkcji i sposobu implementacji obliczeń.

Metody kernelowe opierają się na jednym lub więcej parametrach regularności, które muszą być dobrane tak, aby osiągnąć właściwą równowagę w dostosowaniu do danych bez zbytniego dostosowywania się do losowej zmienności w danych.

Wybór odpowiedniej ilości regularności jest równie ważny jak wybór metody do użycia w pierwszej kolejności. W rzeczywistości może być ważniejszy. Tak naprawdę nie mamy kompletnej implementacji nieparametrycznego estymatora dopóki nie zaimplementujemy automatycznego, opartego na danych sposobu wyboru ilości regulacji.

Implementacja tylko obliczeń dla oceny estymatora jądra, powiedzmy, i pozostawiając to całkowicie użytkownikowi wyboru szerokości pasma jest pracą w połowie wykonaną. Metody i implementacje do wyboru szerokości pasma są więc w tym raporcie omówione dość szczegółowo.

W ostatniej części przeprowadzona jest analiza prawdopodobieństwa. Robi się to w celu dalszego wyjaśnienia, dlaczego potrzebne są estymatory z regularyzacją w celu uniknięcia nadmiernego dopasowania do danych, oraz dlaczego nie istnieje w ogóle nieparametryczny maksymalnego prawdopodobieństwa estymatora gęstości. Regularyzację prawdopodobieństwamożna osiągnąć poprzez ograniczenie szacunków gęstości do rodziny coraz bardziej elastycznych gęstości parametrycznych, które są dopasowane do danych. Jest to znane jako metoda sit. Inne podejście opiera się na rozszerzeniach bazowych, ale w obu przypadkach automatyczny wybór wielkości regularności jest tak samo ważny jak w przypadku metod jądrowych.

Aby utworzyć wykres gęstości jądra, musisz oszacować gęstość jądra. W tym celu można użyć funkcji density, a następnie przekazać obiekt density do funkcji plot.

# dane
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data)

# Kernel density plot
plot(d, lwd = 2, main = "Default kernel density plot")

Argument jądra funkcji gęstości domyślnie używa jądra gaussowskiego (kernel = “gaussian”), ale dostępnych jest więcej typów jądra, takich jak “prostokątne”, “trójkątne”, “epanechnikov”, “biweight”, “cosine” i “optcosine”. Wybór będzie zależał od twoich danych, ale w większości scenariuszy wartość domyślna jest najbardziej zalecana.

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             kernel = "rectangular")

# Kernel density plot
plot(d, lwd = 2, main = "Rectangular kernel")

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             kernel = "triangular")

# Kernel density plot
plot(d, lwd = 2, main = "Triangular kernel")

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             kernel = "epanechnikov")

# Kernel density plot
plot(d, lwd = 2, main = "Epanechnikov kernel")

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             kernel = "biweight")

# Kernel density plot
plot(d, lwd = 2, main = "Biweight kernel")

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             kernel = "cosine")

# Kernel density plot
plot(d, lwd = 2, main = "Cosine kernel")

2.1 Selekcja pasma

Argument bw funkcji gęstości pozwala na zmianę używanego pasma wygładzania. Możesz przekazać wartość lub ciąg znaków podający regułę wyboru lub funkcję. Domyślną wartością jest “nrd0” (lub bw.nrd0(.)), która implementuje podejście oparte na zasadzie reguły kciuka :-) Inne dostępne opcje to:

2.1.1 Reguła Scotta (1992)

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             bw = "nrd")

# Kernel density plot
plot(d, lwd = 2, main = "nrd bandwidth")

2.1.2 Nieobciążona cross-walidacja

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             bw = "ucv")

# Kernel density plot
plot(d, lwd = 2, main = "ucv bandwidth")

2.1.3 Obciążona cross-walidacja

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             bw = "bcv")

# Kernel density plot
plot(d, lwd = 2, main = "bcv bandwidth") 

2.1.4 Metoda Sheather & Jones (1991)

# Data
set.seed(14012021)
data <- rnorm(200, mean = 4)

# Kernel density estimation
d <- density(data,
             bw = "SJ")

# Kernel density plot
plot(d, lwd = 2, main = "SJ bandwidth")

Ostrzeżenie!

Szerokość pasma musi być bardzo starannie dobrana! Mała szerokość pasma spowoduje powstanie nadmiernie dopasowanej krzywej, natomiast zbyt duża szerokość pasma spowoduje powstanie krzywej nadmiernie wygładzonej.

3 Ćwiczenie 1.

Uruchom demo estymatora funkcji gęstości kernel. Zmieniaj zarówno dane wejściowe, jak i opcje estymatora - szerokość pasma oraz rodzaj funkcji jądrowej. Czy widzisz istotne różnice w oszacowaniu?

#install.packages("remotes") #tylko raz! potem #
#install_github("hericks/KDE") #tylko raz! potem #
#install.packages("htmltools")
library(ggplot2)

Rzeczywiście można zauważyć wyraźne różnice. Zwiększenie wartości odchylenia standardowego powoduje, że rozkład staje się bardziej płaski i rozciągnięty, co oznacza większe rozproszenie danych wokół średniej. Z kolei zmniejszenie odchylenia standardowego sprawia, że rozkład staje się węższy i bardziej skupiony. Zmiana wartości średniej powoduje przesunięcie środka rozkładu na osi poziomej (oś x). Innymi słowy, cały rozkład przesuwa się w prawo lub w lewo w zależności od tego, czy średnia została zwiększona, czy zmniejszona, ale sam kształt rozkładu pozostaje taki sam. Natomiast modyfikacja szerokości pasma (bandwidth) wpływa na stopień, w jakim krzywa dopasowuje się do danych. Im mniejsza wartość pasma, tym bardziej krzywa staje się szczegółowa i lepiej odwzorowuje lokalne zmiany w danych. Jednak większa szerokość pasma powoduje wygładzenie krzywej, co prowadzi do bardziej ogólnego dopasowania i pominięcia drobnych szczegółów.

4 Ćwiczenie 2.

Wykorzystując dowolną funkcję R do estymacji funkcji gęstości oszacuj jej przebieg dla wynagrodzeń (zbiór danych salaries) strażaków w San Francisco. Wykorzystaj metody graficzne dostępne w pakiecie ggplot2. Mile widziane przekroje oraz odpowiedzi na pytania badawcze zadane na wstępie.

salariesFFSanF <- subset(salaries, JobTitle=="Firefighter" & Agency=="San Francisco")
dim(salariesFFSanF)
## [1] 2359   13

Histogram wynagrodzeń dla strażaków z San Francisco.

sr_mediana_linie <- data.frame(nazwa = c("Średnia",
                                     "Mediana"),
                       wartość     = c(mean(salariesFFSanF$TotalPay),
                                  median(salariesFFSanF$TotalPay)))

ggplot(salariesFFSanF,aes(x = TotalPay)) +
  geom_histogram(binwidth=10000, color="purple",fill="yellow") +
  labs(x = "Pay (in dollars)", y = "Frequency", title = "Histogram wynagrodzeń strażaków w San Francisco") +
  geom_vline(data=sr_mediana_linie,aes(xintercept = wartość,
                               linetype = nazwa,
                               col = nazwa),linewidth=1,color="blue")+
  theme_minimal()

##Poniżej histogram zmian wynagrodzeń strażaków z SF na przestrzeni lat

sr_mediana_linie_2012 <- data.frame(nazwa = c("Średnia z 2012",
                                     "Mediana z 2012"), wartość = c(mean(salariesFFSanF$TotalPay[salariesFFSanF$Year==2012]),
 median(salariesFFSanF$TotalPay[salariesFFSanF$Year==2012])))

ggplot(salariesFFSanF,aes(x = TotalPay)) +
  geom_histogram(binwidth=25000, color="purple",fill="yellow") +
  labs(x = "Pay (in dollars)", y = "Frequency", title = "Histogram wynagrodzeń strażaków w San Francisco") +
  geom_vline(data=sr_mediana_linie,aes(xintercept = wartość,
                               linetype = nazwa,
                               col = nazwa),linewidth=1,color="blue")+
  theme_minimal()+
  facet_grid(. ~ Year)

Możemy zauważyć, że w porównaniu do 2012 roku w 2013 roku liczba strażaków, którzy zarabiali kwotę odpowiadającą średniej i medianie z 2012 roku, wzrosła. Jednak w 2014 roku wartości te powróciły do poziomów zbliżonych do tych z 2012 roku. Świadczy o tym zmiana średniego wynagrodzenia w analizowanych latach – po wzroście w 2013 roku osiągnęło ono najwyższy poziom, by następnie w 2014 roku spaść poniżej wartości odnotowanej w 2012 roku.

sr_mediany_lata <- data.frame(Rok = c(2012,2013,2014), 
                                    średnie = c(mean(salariesFFSanF$TotalPay[salariesFFSanF$Year==2012]),
mean(salariesFFSanF$TotalPay[salariesFFSanF$Year==2013]),
mean(salariesFFSanF$TotalPay[salariesFFSanF$Year==2014])),
                                    mediany = c(median(salariesFFSanF$TotalPay[salariesFFSanF$Year==2012]),
median(salariesFFSanF$TotalPay[salariesFFSanF$Year==2013]),
median(salariesFFSanF$TotalPay[salariesFFSanF$Year==2014])))

##Wykres pokazujący zmiany średniej i mediany płac dla strażaków w San Francisco w latach 2012-2014

ggplot(sr_mediany_lata, aes(Rok, średnie)) + 
    geom_ribbon(aes(ymin = średnie, ymax = mediany),fill = "purple")+
  geom_line(aes(x=Rok,y=mediany,color="Mediany"),lwd=1)+
  geom_line(aes(x=Rok,y=średnie,color="Średnie"),lwd=1)+
  labs(x = "Year", y = "Total Pay (in Dollars)", title = "Zmiany różnicy między średnią i medianą na przestrzeni trzech lat wśród strażaków San Francisco")

Możemy zaobserwować zmiany w średniej i medianie wynagrodzeń strażaków w analizowanych latach, jak wspomniano wcześniej. Co więcej, różnica między medianą a średnią staje się coraz bardziej widoczna, przy czym średnia wypada na niższym poziomie w stosunku do mediany. To wskazuje, że większość strażaków otrzymuje wynagrodzenie wyższe niż przeciętna płaca w tej grupie zawodowej.

  • Jak płaca podstawowa, wynagrodzenie za nadgodziny i świadczenia są rozdzielane pomiędzy różne grupy?
salariesFFSanF$Benefits<-as.numeric(salariesFFSanF$Benefits)
salariesFFSanF$OvertimePay<-as.numeric(salariesFFSanF$OvertimePay)
salariesFFSanF$BasePay<-as.numeric(salariesFFSanF$BasePay)

##Płaca podstawowa

sr_mediana_linie_basepay <- data.frame(nazwa = c("Średnia dla FT",
                                     "Mediana dla FT"), 
                                     wartość=c(mean(salariesFFSanF$BasePay[salariesFFSanF$Status=='FT']),
 median(salariesFFSanF$BasePay[salariesFFSanF$Status=='FT'])))

ggplot(salariesFFSanF,aes(x = BasePay)) +
  geom_histogram(binwidth=25000, color="green",fill="blue") +
  labs(x = "Base Pay", y = "Frequency", title = "Histogram płacy podstawowej dla strażaków w San Francisco") +
  geom_vline(data=sr_mediana_linie_basepay,aes(xintercept = wartość,
                               linetype = nazwa,
                               col = nazwa),linewidth=1,color="black")+
  theme_minimal()+
  facet_grid(. ~ Status)
## Warning: Removed 11 rows containing non-finite outside the scale range
## (`stat_bin()`).

##Wynagrodzenie za nadgodziny

sr_mediana_linie_overtime <- data.frame(nazwa = c("Średnia",
                                     "Mediana"), wartość = c(mean(salariesFFSanF$OvertimePay),
 median(salariesFFSanF$OvertimePay)))

ggplot(salariesFFSanF,aes(x = OvertimePay)) +
  geom_histogram(binwidth=25000, color="green",fill="blue") +
  labs(x = "Overtime Pay", y = "Frequency", title = "Histogram płacy za nadgodziny dla strażaków w San Francisco") +
  geom_vline(data=sr_mediana_linie_overtime,aes(xintercept = wartość,
                               linetype = nazwa,
                               col = nazwa),linewidth=1,color="black")+
  theme_minimal()+
  facet_grid(. ~ Status)

##Benefity

sr_mediana_linie_benefits <- data.frame(nazwa = c("Średnia",
                                     "Mediana"), wartość = c(mean(salariesFFSanF$Benefits),
 median(salariesFFSanF$Benefits)))

ggplot(salariesFFSanF,aes(x = Benefits)) +
  geom_histogram(binwidth=10000, color="red",fill="blue") +
  labs(x = "Benefits", y = "Frequency", title = "Histogram świadczeń/Benefitów dla strażaków w San Francisco") +
  geom_vline(data=sr_mediana_linie_benefits,aes(xintercept = wartość,
                               linetype = nazwa,
                               col = nazwa),linewidth=1,color="black")+
  theme_minimal()+
  facet_grid(. ~ Status)

Możemy zauważyć, że budżet przeznaczony na wynagrodzenie podstawowe, płatności za nadgodziny oraz świadczenia różni się w zależności od tego, czy strażacy są zatrudnieni w pełnym wymiarze godzin (FT), czy na pół etatu (ST). Zgodnie z przewidywaniami, wszystkie te wydatki są niższe w przypadku strażaków pracujących w niepełnym wymiarze godzin.

Oszacowanie gęstości rozkładu wynagrodzeń strażaków w San Francisco:

# Kernel density estimation
d <- density(salariesFFSanF$TotalPay,
             kernel = "gaussian",
             bw = 50000)
# Kernel density plot
plot(d, lwd = 2, main = "Gaussian kernel")

Mamy tutaj przykład rozkładu bimodalnego, który pomaga zrozumieć wcześniej zauważone zjawisko, że mediana przewyższa średnią. Wynika to z obecności znacznej grupy strażaków (stanowiącej drugie lokalne maksimum), którzy zarabiają niewielkie kwoty w przedziale <0; 50 000 USD>. Ich zarobki znacząco obniżają średnią płac, ale ich liczba nie jest na tyle duża, aby w istotny sposób wpłynąć na obniżenie mediany.

LS0tDQp0aXRsZTogIktlcm5lbCINCmF1dGhvcjogIkFsZWtzYW5kcmEgQnVrb3dza2EiDQpkYXRlOiAiYHIgU3lzLkRhdGUoKWAiDQpvdXRwdXQ6DQogIGh0bWxfZG9jdW1lbnQ6DQogICAgdGhlbWU6IGNlcnVsZWFuDQogICAgaGlnaGxpZ2h0OiB0ZXh0bWF0ZQ0KICAgIGZvbnRzaXplOiA4cHQNCiAgICB0b2M6IHRydWUNCiAgICBudW1iZXJfc2VjdGlvbnM6IHRydWUNCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlDQogICAgdG9jX2Zsb2F0Og0KICAgICAgY29sbGFwc2VkOiBmYWxzZQ0KZWRpdG9yX29wdGlvbnM6IA0KICBtYXJrZG93bjogDQogICAgd3JhcDogNzINCi0tLQ0KDQpgYGB7ciBzZXR1cCwgaW5jbHVkZT1GQUxTRX0NCmtuaXRyOjpvcHRzX2NodW5rJHNldChlY2hvID0gVFJVRSkNCm9wdGlvbnMoc2NpcGVuPTk5OSwgZGlnaXRzPTMpDQpzYWxhcmllcyA8LSByZWFkLmNzdigiaHR0cHM6Ly9naXRodWIuY29tL2tmbGlzaWtvd3NraS9kcy9yYXcvbWFzdGVyL1NhbGFyaWVzLmNzdiIpDQpgYGANCg0KIyBBbmFsaXphIG9waXNvd2ENCg0KSmVkbnltIHplIHNwb3NvYsOzdyB6cm96dW1pZW5pYSwgamFrIGR6aWHFgmEgcnrEhWQgbWlhc3RhLCBqZXN0IHNwb2pyemVuaWUgbmEgdG8sIGtvZ28gemF0cnVkbmlhIGkgamFrIGplZ28gcHJhY293bmljeSBzxIUgd3luYWdyYWR6YW5pLiBEYW5lIHRlIHphd2llcmFqxIUgbmF6d2lza2EsIG5henfEmSBzdGFub3dpc2thIGkgd3luYWdyb2R6ZW5pZSBwcmFjb3duaWvDs3cgbWlhc3RhIFNhbiBGcmFuY2lzY28gdyB1asSZY2l1IHJvY3pueW0gb2QgMjAxMSBkbyAyMDE0IHJva3UuDQoNCk90byBraWxrYSBwb215c8WCw7N3IG5hIGVrc3Bsb3JhY2rEmSBkYW55Y2g6DQoNCi0gSmFrIHptaWVuaWHFgnkgc2nEmSB3eW5hZ3JvZHplbmlhIHcgY3phc2llIG1pxJlkenkgcsOzxbxueW1pIGdydXBhbWkgbHVkemk/DQoNCi0gSmFrIHDFgmFjYSBwb2RzdGF3b3dhLCB3eW5hZ3JvZHplbmllIHphIG5hZGdvZHppbnkgaSDFm3dpYWRjemVuaWEgc8SFIHJvemR6aWVsYW5lIHBvbWnEmWR6eSByw7PFvG5lIGdydXB5Pw0KDQotIEN6eSB3IHR5bSB6ZXN0YXdpZSBkYW55Y2ggaXN0bmllasSFIGRvd29keSBuYSBkeXNrcnltaW5hY2rEmSBwxYJhY293xIUgemUgd3pnbMSZZHUgbmEgcMWCZcSHPw0KDQotIEphayBwcnp5ZHppZWxhbnkgamVzdCBidWTFvGV0IHcgemFsZcW8bm/Fm2NpIG9kIGdydXB5IGkgemFrcmVzdSBvYm93acSFemvDs3c/DQoNCmBgYHtyIH0NCiMgd3ltaWFyeSByYW1raToNCmRpbShzYWxhcmllcykNCiMgbmF6d3kga29sdW1uOg0KbmFtZXMoc2FsYXJpZXMpDQpgYGANCg0KIyMgSGlzdG9ncmFteQ0KDQpgYGB7ciB9DQpoaXN0KHNhbGFyaWVzJFRvdGFsUGF5LG1haW49IlRvdGFsIFBheSIsIHhsYWI9IlBheSAoaW4gZG9sbGFycykiKQ0KYWJsaW5lKHYgPSBtZWFuKHNhbGFyaWVzJFRvdGFsUGF5KSxsdHk9ImRhc2hlZCIpDQphYmxpbmUodiA9IG1lZGlhbihzYWxhcmllcyRUb3RhbFBheSkpDQpsZWdlbmQoInRvcHJpZ2h0IiwgbGVnZW5kPWMoIk1lZGlhbmEiLCLFmnJlZG5pYSIpLGx0eT1jKCJzb2xpZCIsImRhc2hlZCIpKQ0KYGBgDQoNCmBgYHtyIH0NCnBhcihtZnJvdz1jKDIsMikpDQpoaXN0KHNhbGFyaWVzJFRvdGFsUGF5LG1haW49IlRvdGFsIFBheSwgZGVmYXVsdCBicmVha3MiLCB4bGFiPSJQYXkgKGluIGRvbGxhcnMpIikNCmhpc3Qoc2FsYXJpZXMkVG90YWxQYXksbWFpbj0iVG90YWwgUGF5LCBicmVha3M9MTAwIiwgeGxhYj0iUGF5IChpbiBkb2xsYXJzKSIsIGJyZWFrcz0xMDApDQpoaXN0KHNhbGFyaWVzJFRvdGFsUGF5LG1haW49IlRvdGFsIFBheSwgYnJlYWtzPTEwMDAiLCB4bGFiPSJQYXkgKGluIGRvbGxhcnMpIixicmVha3M9MTAwMCkNCmBgYA0KDQpgYGB7ciB9DQpoaXN0KHNhbGFyaWVzJFRvdGFsUGF5LG1haW49IlRvdGFsIFBheSwgWm9vbWVkLWluIiwgeGxhYj0iUGF5IChpbiBkb2xsYXJzKSIsIHhsaW09YygwLDFlNSksIGJyZWFrcz0xMDAwKQ0KYGBgDQoNCmBgYHtyIH0NCnNhbGFyaWVzMiA8LSBzdWJzZXQoc2FsYXJpZXMsIEpvYlRpdGxlPT0iRmlyZWZpZ2h0ZXIiICYgU3RhdHVzPT0iRlQiKQ0KZGltKHNhbGFyaWVzMikNCmBgYA0KDQpgYGB7ciB9DQpwYXIobWZyb3c9YygyLDIpKQ0KaGlzdChzYWxhcmllczIkVG90YWxQYXksbWFpbj0iRmlyZWZpZ2h0ZXJzLCBkZWZhdWx0IGJyZWFrcyIsIHhsYWI9IlBheSAoaW4gZG9sbGFycykiKQ0KaGlzdChzYWxhcmllczIkVG90YWxQYXksbWFpbj0iRmlyZWZpZ2h0ZXJzLCBicmVha3M9MzAiLCB4bGFiPSJQYXkgKGluIGRvbGxhcnMpIiwgYnJlYWtzPTMwKQ0KaGlzdChzYWxhcmllczIkVG90YWxQYXksbWFpbj0iRmlyZWZpZ2h0ZXJzLCBicmVha3M9MTAwIiwgeGxhYj0iUGF5IChpbiBkb2xsYXJzKSIsIGJyZWFrcz0xMDApDQpoaXN0KHNhbGFyaWVzMiRUb3RhbFBheSxtYWluPSJGaXJlZmlnaHRlcnMsIGJyZWFrcz0xMDAwIiwgeGxhYj0iUGF5IChpbiBkb2xsYXJzKSIsYnJlYWtzPTEwMDApDQpgYGANCg0KIyMgV3lrcmVzeSBwdWRlxYJrb3dlDQoNCmBgYHtyIH0NCnBhcihtZnJvdz1jKDEsMSkpDQpib3hwbG90KHNhbGFyaWVzJFRvdGFsUGF5LG1haW49IlRvdGFsIFBheSwgYnJlYWtzPTEwMDAiLCB5bGFiPSJQYXkgKGluIGRvbGxhcnMpIikNCmBgYA0KDQojIEVzdHltYWNqYSBmdW5rY2ppIGfEmXN0b8WbY2kNCg0KUGllcndzenkgcmFwb3J0IGRvdHljenkgbmllcGFyYW1ldHJ5Y3puZWogZXN0eW1hY2ppIGfEmXN0b8WbY2kuIEtsYXN5Y3pueW0gbmllcGFyYW1ldHJ5Y3pueW0gZXN0eW1hdG9yZW0gZ8SZc3RvxZtjaSBqZXN0IGhpc3RvZ3JhbSwga3TDs3J5IGRvc3RhcmN6YSBuaWVjacSFZ8WCZSBpIHN0YcWCZSBvc3phY293YW5pYS4gVyB0eW0gcmFwb3JjaWUgc2t1cGlvbm8gc2nEmSBuYSBuaWVrdMOzcnljaA0KYWx0ZXJuYXR5d2FjaCwga3TDs3JlIHphcGV3bmlhasSFIGNpxIVnxYJlIGx1YiBuYXdldCBnxYJhZGtpZSBvc3phY293YW5pYSB6YW1pYXN0Lg0KDQoqTWV0b2R5IGtlcm5lbG93ZSogc3Rhbm93acSFIHdhxbxuxIUga2xhc8SZIGfFgmFka2ljaCBlc3R5bWF0b3LDs3cgZ8SZc3RvxZtjaSBpIHphaW1wbGVtZW50b3dhbmUgc8SFIHByemV6IGZ1bmtjasSZIFIgYGRlbnNpdHkoKWAuIEVzdHltYXRvcnkgdGUgc8SFIHcgemFzYWR6aWUgdHlsa28gbG9rYWxuaWUgd2HFvG9ueW1pIMWbcmVkbmltaSwgYSBpY2ggb2JsaWN6ZW5pZSBqZXN0IHN0b3N1bmtvd28gcHJvc3RlIHcgdGVvcmlpLiBXIHByYWt0eWNlLCByw7PFvG5lIHd5Ym9yeSBzcG9zb2J1IGltcGxlbWVudGFjamkgb2JsaWN6ZcWEIG1vZ8SFIGplZG5hayBtaWXEhyBkdcW8eSB3cMWCeXcgbmEgcnplY3p5d2lzdHkgY3phcw0Kb2JsaWN6ZcWELCBhIGltcGxlbWVudGFjasSZIGtlcm5lbG93eWNoIGVzdHltYXRvcsOzdyBnxJlzdG/Fm2NpIHppbHVzdHJ1amUgdHJ6eSBwdW5rdHk6DQoNCi0gICBqZcWbbGkgdG8gbW/FvGxpd2UsIHd5Ymllcnogd2VrdG9yeXpvd2FuZSBpbXBsZW1lbnRhY2plIHcgUiwNCi0gICBqZcWbbGkgbmlld2llbGthIHN0cmF0YSB3IGRva8WCYWRub8WbY2kgamVzdCBkbyB6YWFrY2VwdG93YW5pYSwgcHJ6eWJsacW8b25lIHJvendpxIV6YW5pZSBtb8W8ZSBiecSHIG8gcnrEmWR5IHdpZWxrb8WbY2kgc3p5YnN6ZSBuacW8IGltcGxlbWVudGFjamEgbGl0ZXJhbG5hLA0KLSAgIGN6YXMgcG90cnplYm55IGRvIG51bWVyeWN6bmVqIG9jZW55IHLDs8W8bnljaCBbZnVua2NqZSBlbGVtZW50YXJuZV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvRWxlbWVudGFyeV9mdW5jdGlvbikgbW/FvGUgYmFyZHpvIHphbGXFvGXEhyBvZCBmdW5rY2ppIGkgc3Bvc29idSBpbXBsZW1lbnRhY2ppIG9ibGljemXFhC4NCg0KTWV0b2R5IGtlcm5lbG93ZSBvcGllcmFqxIUgc2nEmSBuYSBqZWRueW0gbHViIHdpxJljZWogKnBhcmFtZXRyYWNoIHJlZ3VsYXJub8WbY2kqLCBrdMOzcmUgbXVzesSFIGJ5xIcgZG9icmFuZSB0YWssIGFieSBvc2nEhWduxIXEhyB3xYJhxZtjaXfEhSByw7N3bm93YWfEmSB3IGRvc3Rvc293YW5pdSBkbyBkYW55Y2ggYmV6IHpieXRuaWVnbyBkb3N0b3Nvd3l3YW5pYSBzacSZIGRvIGxvc293ZWogem1pZW5ub8WbY2kgdyBkYW55Y2guDQoNCld5YsOzciBvZHBvd2llZG5pZWogaWxvxZtjaSByZWd1bGFybm/Fm2NpIGplc3QgcsOzd25pZSB3YcW8bnkgamFrIHd5YsOzciBtZXRvZHkgZG8gdcW8eWNpYSB3IHBpZXJ3c3plaiBrb2xlam5vxZtjaS4gVyAgcnplY3p5d2lzdG/Fm2NpIG1vxbxlIGJ5xIcgd2HFvG5pZWpzenkuIFRhayBuYXByYXdkxJkgbmllIG1hbXkga29tcGxldG5laiBpbXBsZW1lbnRhY2ppIG5pZXBhcmFtZXRyeWN6bmVnbyBlc3R5bWF0b3JhIGRvcMOza2kgbmllIHphaW1wbGVtZW50dWplbXkgYXV0b21hdHljem5lZ28sIG9wYXJ0ZWdvIG5hIGRhbnljaCBzcG9zb2J1IHd5Ym9ydSBpbG/Fm2NpIHJlZ3VsYWNqaS4NCg0KSW1wbGVtZW50YWNqYSB0eWxrbyBvYmxpY3plxYQgZGxhIG9jZW55IGVzdHltYXRvcmEgasSFZHJhLCBwb3dpZWR6bXksIGkgcG96b3N0YXdpYWrEhWMgdG8gY2HFgmtvd2ljaWUgdcW8eXRrb3duaWtvd2kgd3lib3J1IHN6ZXJva2/Fm2NpIHBhc21hIGplc3QgcHJhY8SFIHcgcG/Fgm93aWUgd3lrb25hbsSFLiBNZXRvZHkgaSBpbXBsZW1lbnRhY2plIGRvIHd5Ym9ydSBzemVyb2tvxZtjaSBwYXNtYSBzxIUgd2nEmWMgdyB0eW0gcmFwb3JjaWUgb23Ds3dpb25lIGRvxZvEhyBzemN6ZWfDs8WCb3dvLg0KDQpXIG9zdGF0bmllaiBjesSZxZtjaSBwcnplcHJvd2Fkem9uYSBqZXN0IGFuYWxpemEgcHJhd2RvcG9kb2JpZcWEc3R3YS4gUm9iaSBzacSZIHRvIHcgY2VsdSBkYWxzemVnbyB3eWphxZtuaWVuaWEsIGRsYWN6ZWdvIHBvdHJ6ZWJuZSBzxIUgZXN0eW1hdG9yeSB6IHJlZ3VsYXJ5emFjasSFIHcgY2VsdSB1bmlrbmnEmWNpYSBuYWRtaWVybmVnbyBkb3Bhc293YW5pYSBkbyBkYW55Y2gsIG9yYXogZGxhY3plZ28gbmllIGlzdG5pZWplIHcgb2fDs2xlIG5pZXBhcmFtZXRyeWN6bnkgbWFrc3ltYWxuZWdvIHByYXdkb3BvZG9iaWXFhHN0d2EgZXN0eW1hdG9yYSBnxJlzdG/Fm2NpLiBSZWd1bGFyeXphY2rEmSBwcmF3ZG9wb2RvYmllxYRzdHdhbW/FvG5hIG9zacSFZ27EhcSHIHBvcHJ6ZXogb2dyYW5pY3plbmllIHN6YWN1bmvDs3cgZ8SZc3RvxZtjaSBkbyByb2R6aW55IGNvcmF6IGJhcmR6aWVqIGVsYXN0eWN6bnljaCBnxJlzdG/Fm2NpIHBhcmFtZXRyeWN6bnljaCwga3TDs3JlIHPEhSBkb3Bhc293YW5lIGRvIGRhbnljaC4gSmVzdCB0byB6bmFuZSBqYWtvICptZXRvZGEgc2l0Ki4gSW5uZSBwb2RlasWbY2llIG9waWVyYSBzacSZIG5hIHJvenN6ZXJ6ZW5pYWNoIGJhem93eWNoLCBhbGUgdyBvYnUgcHJ6eXBhZGthY2ggYXV0b21hdHljem55IHd5YsOzciB3aWVsa2/Fm2NpIHJlZ3VsYXJub8WbY2kgamVzdCB0YWsgc2FtbyB3YcW8bnkgamFrIHcgcHJ6eXBhZGt1IG1ldG9kIGrEhWRyb3d5Y2guDQoNCkFieSB1dHdvcnp5xIcgd3lrcmVzIGfEmXN0b8WbY2kgasSFZHJhLCBtdXNpc3ogb3N6YWNvd2HEhyBnxJlzdG/Fm8SHIGrEhWRyYS4gVyB0eW0gY2VsdSBtb8W8bmEgdcW8ecSHIGZ1bmtjamkgZGVuc2l0eSwgYSBuYXN0xJlwbmllIHByemVrYXphxIcgb2JpZWt0IGRlbnNpdHkgZG8gZnVua2NqaSBwbG90Lg0KDQpgYGB7cn0NCiMgZGFuZQ0Kc2V0LnNlZWQoMTQwMTIwMjEpDQpkYXRhIDwtIHJub3JtKDIwMCwgbWVhbiA9IDQpDQoNCiMgS2VybmVsIGRlbnNpdHkgZXN0aW1hdGlvbg0KZCA8LSBkZW5zaXR5KGRhdGEpDQoNCiMgS2VybmVsIGRlbnNpdHkgcGxvdA0KcGxvdChkLCBsd2QgPSAyLCBtYWluID0gIkRlZmF1bHQga2VybmVsIGRlbnNpdHkgcGxvdCIpDQoNCmBgYA0KDQpBcmd1bWVudCBqxIVkcmEgZnVua2NqaSBnxJlzdG/Fm2NpIGRvbXnFm2xuaWUgdcW8eXdhIGrEhWRyYSBnYXVzc293c2tpZWdvIChrZXJuZWwgPSAiZ2F1c3NpYW4iKSwgYWxlIGRvc3TEmXBueWNoIGplc3Qgd2nEmWNlaiB0eXDDs3cgasSFZHJhLCB0YWtpY2ggamFrICJwcm9zdG9rxIV0bmUiLCAidHLDs2prxIV0bmUiLCAiZXBhbmVjaG5pa292IiwgImJpd2VpZ2h0IiwgImNvc2luZSIgaSAib3B0Y29zaW5lIi4gV3liw7NyIGLEmWR6aWUgemFsZcW8YcWCIG9kIHR3b2ljaCBkYW55Y2gsIGFsZSB3IHdpxJlrc3pvxZtjaSBzY2VuYXJpdXN6eSB3YXJ0b8WbxIcgZG9tecWbbG5hIGplc3QgbmFqYmFyZHppZWogemFsZWNhbmEuDQoNCmBgYHtyfQ0KIyBEYXRhDQpzZXQuc2VlZCgxNDAxMjAyMSkNCmRhdGEgPC0gcm5vcm0oMjAwLCBtZWFuID0gNCkNCg0KIyBLZXJuZWwgZGVuc2l0eSBlc3RpbWF0aW9uDQpkIDwtIGRlbnNpdHkoZGF0YSwNCiAgICAgICAgICAgICBrZXJuZWwgPSAicmVjdGFuZ3VsYXIiKQ0KDQojIEtlcm5lbCBkZW5zaXR5IHBsb3QNCnBsb3QoZCwgbHdkID0gMiwgbWFpbiA9ICJSZWN0YW5ndWxhciBrZXJuZWwiKQ0KYGBgDQoNCmBgYHtyfQ0KIyBEYXRhDQpzZXQuc2VlZCgxNDAxMjAyMSkNCmRhdGEgPC0gcm5vcm0oMjAwLCBtZWFuID0gNCkNCg0KIyBLZXJuZWwgZGVuc2l0eSBlc3RpbWF0aW9uDQpkIDwtIGRlbnNpdHkoZGF0YSwNCiAgICAgICAgICAgICBrZXJuZWwgPSAidHJpYW5ndWxhciIpDQoNCiMgS2VybmVsIGRlbnNpdHkgcGxvdA0KcGxvdChkLCBsd2QgPSAyLCBtYWluID0gIlRyaWFuZ3VsYXIga2VybmVsIikNCmBgYA0KDQpgYGB7cn0NCiMgRGF0YQ0Kc2V0LnNlZWQoMTQwMTIwMjEpDQpkYXRhIDwtIHJub3JtKDIwMCwgbWVhbiA9IDQpDQoNCiMgS2VybmVsIGRlbnNpdHkgZXN0aW1hdGlvbg0KZCA8LSBkZW5zaXR5KGRhdGEsDQogICAgICAgICAgICAga2VybmVsID0gImVwYW5lY2huaWtvdiIpDQoNCiMgS2VybmVsIGRlbnNpdHkgcGxvdA0KcGxvdChkLCBsd2QgPSAyLCBtYWluID0gIkVwYW5lY2huaWtvdiBrZXJuZWwiKQ0KYGBgDQoNCmBgYHtyfQ0KIyBEYXRhDQpzZXQuc2VlZCgxNDAxMjAyMSkNCmRhdGEgPC0gcm5vcm0oMjAwLCBtZWFuID0gNCkNCg0KIyBLZXJuZWwgZGVuc2l0eSBlc3RpbWF0aW9uDQpkIDwtIGRlbnNpdHkoZGF0YSwNCiAgICAgICAgICAgICBrZXJuZWwgPSAiYml3ZWlnaHQiKQ0KDQojIEtlcm5lbCBkZW5zaXR5IHBsb3QNCnBsb3QoZCwgbHdkID0gMiwgbWFpbiA9ICJCaXdlaWdodCBrZXJuZWwiKQ0KYGBgDQoNCmBgYHtyfQ0KIyBEYXRhDQpzZXQuc2VlZCgxNDAxMjAyMSkNCmRhdGEgPC0gcm5vcm0oMjAwLCBtZWFuID0gNCkNCg0KIyBLZXJuZWwgZGVuc2l0eSBlc3RpbWF0aW9uDQpkIDwtIGRlbnNpdHkoZGF0YSwNCiAgICAgICAgICAgICBrZXJuZWwgPSAiY29zaW5lIikNCg0KIyBLZXJuZWwgZGVuc2l0eSBwbG90DQpwbG90KGQsIGx3ZCA9IDIsIG1haW4gPSAiQ29zaW5lIGtlcm5lbCIpDQpgYGANCg0KIyMgU2VsZWtjamEgcGFzbWENCg0KQXJndW1lbnQgYncgZnVua2NqaSBnxJlzdG/Fm2NpIHBvendhbGEgbmEgem1pYW7EmSB1xbx5d2FuZWdvIHBhc21hIHd5Z8WCYWR6YW5pYS4gTW/FvGVzeiBwcnpla2F6YcSHIHdhcnRvxZvEhyBsdWIgY2nEhWcgem5ha8OzdyBwb2RhasSFY3kgcmVndcWCxJkgd3lib3J1IGx1YiBmdW5rY2rEmS4gRG9tecWbbG7EhSB3YXJ0b8WbY2nEhSBqZXN0ICJucmQwIiAobHViIGJ3Lm5yZDAoLikpLCBrdMOzcmEgaW1wbGVtZW50dWplIHBvZGVqxZtjaWUgb3BhcnRlIG5hIHphc2FkemllIHJlZ3XFgnkga2NpdWthIDotKSBJbm5lIGRvc3TEmXBuZSBvcGNqZSB0bzoNCg0KIyMjIFJlZ3XFgmEgU2NvdHRhICgxOTkyKQ0KDQpgYGB7cn0NCiMgRGF0YQ0Kc2V0LnNlZWQoMTQwMTIwMjEpDQpkYXRhIDwtIHJub3JtKDIwMCwgbWVhbiA9IDQpDQoNCiMgS2VybmVsIGRlbnNpdHkgZXN0aW1hdGlvbg0KZCA8LSBkZW5zaXR5KGRhdGEsDQogICAgICAgICAgICAgYncgPSAibnJkIikNCg0KIyBLZXJuZWwgZGVuc2l0eSBwbG90DQpwbG90KGQsIGx3ZCA9IDIsIG1haW4gPSAibnJkIGJhbmR3aWR0aCIpDQpgYGANCg0KIyMjIE5pZW9iY2nEhcW8b25hIGNyb3NzLXdhbGlkYWNqYQ0KDQpgYGB7ciBtZXNzYWdlPUZBTFNFLCB3YXJuaW5nPUZBTFNFfQ0KIyBEYXRhDQpzZXQuc2VlZCgxNDAxMjAyMSkNCmRhdGEgPC0gcm5vcm0oMjAwLCBtZWFuID0gNCkNCg0KIyBLZXJuZWwgZGVuc2l0eSBlc3RpbWF0aW9uDQpkIDwtIGRlbnNpdHkoZGF0YSwNCiAgICAgICAgICAgICBidyA9ICJ1Y3YiKQ0KDQojIEtlcm5lbCBkZW5zaXR5IHBsb3QNCnBsb3QoZCwgbHdkID0gMiwgbWFpbiA9ICJ1Y3YgYmFuZHdpZHRoIikNCmBgYA0KDQojIyMgT2JjacSFxbxvbmEgY3Jvc3Mtd2FsaWRhY2phDQoNCmBgYHtyIG1lc3NhZ2U9RkFMU0UsIHdhcm5pbmc9RkFMU0V9DQojIERhdGENCnNldC5zZWVkKDE0MDEyMDIxKQ0KZGF0YSA8LSBybm9ybSgyMDAsIG1lYW4gPSA0KQ0KDQojIEtlcm5lbCBkZW5zaXR5IGVzdGltYXRpb24NCmQgPC0gZGVuc2l0eShkYXRhLA0KICAgICAgICAgICAgIGJ3ID0gImJjdiIpDQoNCiMgS2VybmVsIGRlbnNpdHkgcGxvdA0KcGxvdChkLCBsd2QgPSAyLCBtYWluID0gImJjdiBiYW5kd2lkdGgiKSANCg0KYGBgDQoNCiMjIyBNZXRvZGEgU2hlYXRoZXIgJiBKb25lcyAoMTk5MSkNCg0KYGBge3J9DQojIERhdGENCnNldC5zZWVkKDE0MDEyMDIxKQ0KZGF0YSA8LSBybm9ybSgyMDAsIG1lYW4gPSA0KQ0KDQojIEtlcm5lbCBkZW5zaXR5IGVzdGltYXRpb24NCmQgPC0gZGVuc2l0eShkYXRhLA0KICAgICAgICAgICAgIGJ3ID0gIlNKIikNCg0KIyBLZXJuZWwgZGVuc2l0eSBwbG90DQpwbG90KGQsIGx3ZCA9IDIsIG1haW4gPSAiU0ogYmFuZHdpZHRoIikNCmBgYA0KDQpPc3RyemXFvGVuaWUhDQoNCjogICBTemVyb2tvxZvEhyBwYXNtYSBtdXNpIGJ5xIcgYmFyZHpvIHN0YXJhbm5pZSBkb2JyYW5hISBNYcWCYSBzemVyb2tvxZvEhyBwYXNtYSBzcG93b2R1amUgcG93c3RhbmllIG5hZG1pZXJuaWUgZG9wYXNvd2FuZWoga3J6eXdlaiwgbmF0b21pYXN0IHpieXQgZHXFvGEgc3plcm9rb8WbxIcgcGFzbWEgc3Bvd29kdWplIHBvd3N0YW5pZSBrcnp5d2VqIG5hZG1pZXJuaWUgd3lnxYJhZHpvbmVqLg0KDQojIMSGd2ljemVuaWUgMS4NCg0KVXJ1Y2hvbSBkZW1vIGVzdHltYXRvcmEgZnVua2NqaSBnxJlzdG/Fm2NpIGtlcm5lbC4gWm1pZW5pYWogemFyw7N3bm8gZGFuZSB3ZWrFm2Npb3dlLCBqYWsgaSBvcGNqZSBlc3R5bWF0b3JhIC0gc3plcm9rb8WbxIcgcGFzbWEgb3JheiByb2R6YWogZnVua2NqaSBqxIVkcm93ZWouIEN6eSB3aWR6aXN6IGlzdG90bmUgcsOzxbxuaWNlIHcgb3N6YWNvd2FuaXU/DQoNCg0KDQpgYGB7ciBjd2ljemVuaWUxfQ0KI2luc3RhbGwucGFja2FnZXMoInJlbW90ZXMiKSAjdHlsa28gcmF6ISBwb3RlbSAjDQojaW5zdGFsbF9naXRodWIoImhlcmlja3MvS0RFIikgI3R5bGtvIHJheiEgcG90ZW0gIw0KI2luc3RhbGwucGFja2FnZXMoImh0bWx0b29scyIpDQpsaWJyYXJ5KGdncGxvdDIpDQpgYGANClJ6ZWN6eXdpxZtjaWUgbW/FvG5hIHphdXdhxbx5xIcgd3lyYcW6bmUgcsOzxbxuaWNlLiBad2nEmWtzemVuaWUgd2FydG/Fm2NpIG9kY2h5bGVuaWEgc3RhbmRhcmRvd2VnbyBwb3dvZHVqZSwgxbxlIHJvemvFgmFkIHN0YWplIHNpxJkgYmFyZHppZWogcMWCYXNraSBpIHJvemNpxIVnbmnEmXR5LCBjbyBvem5hY3phIHdpxJlrc3plIHJvenByb3N6ZW5pZSBkYW55Y2ggd29rw7PFgiDFm3JlZG5pZWouIFoga29sZWkgem1uaWVqc3plbmllIG9kY2h5bGVuaWEgc3RhbmRhcmRvd2VnbyBzcHJhd2lhLCDFvGUgcm96a8WCYWQgc3RhamUgc2nEmSB3xJnFvHN6eSBpIGJhcmR6aWVqIHNrdXBpb255Lg0KWm1pYW5hIHdhcnRvxZtjaSDFm3JlZG5pZWogcG93b2R1amUgcHJ6ZXN1bmnEmWNpZSDFm3JvZGthIHJvemvFgmFkdSBuYSBvc2kgcG96aW9tZWogKG/FmyB4KS4gSW5ueW1pIHPFgm93eSwgY2HFgnkgcm96a8WCYWQgcHJ6ZXN1d2Egc2nEmSB3IHByYXdvIGx1YiB3IGxld28gdyB6YWxlxbxub8WbY2kgb2QgdGVnbywgY3p5IMWbcmVkbmlhIHpvc3RhxYJhIHp3acSZa3N6b25hLCBjenkgem1uaWVqc3pvbmEsIGFsZSBzYW0ga3N6dGHFgnQgcm96a8WCYWR1IHBvem9zdGFqZSB0YWtpIHNhbS4NCk5hdG9taWFzdCBtb2R5ZmlrYWNqYSBzemVyb2tvxZtjaSBwYXNtYSAoYmFuZHdpZHRoKSB3cMWCeXdhIG5hIHN0b3BpZcWELCB3IGpha2ltIGtyenl3YSBkb3Bhc293dWplIHNpxJkgZG8gZGFueWNoLiBJbSBtbmllanN6YSB3YXJ0b8WbxIcgcGFzbWEsIHR5bSBiYXJkemllaiBrcnp5d2Egc3RhamUgc2nEmSBzemN6ZWfDs8WCb3dhIGkgbGVwaWVqIG9kd3pvcm93dWplIGxva2FsbmUgem1pYW55IHcgZGFueWNoLiBKZWRuYWsgd2nEmWtzemEgc3plcm9rb8WbxIcgcGFzbWEgcG93b2R1amUgd3lnxYJhZHplbmllIGtyenl3ZWosIGNvIHByb3dhZHppIGRvIGJhcmR6aWVqIG9nw7NsbmVnbyBkb3Bhc293YW5pYSBpIHBvbWluacSZY2lhIGRyb2JueWNoIHN6Y3plZ8OzxYLDs3cuDQoNCiMgxIZ3aWN6ZW5pZSAyLg0KDQpXeWtvcnp5c3R1asSFYyBkb3dvbG7EhSBmdW5rY2rEmSBSIGRvIGVzdHltYWNqaSBmdW5rY2ppIGfEmXN0b8WbY2kgb3N6YWN1aiBqZWogcHJ6ZWJpZWcgZGxhIHd5bmFncm9kemXFhCAoemJpw7NyIGRhbnljaCBzYWxhcmllcykgc3RyYcW8YWvDs3cgdyBTYW4gRnJhbmNpc2NvLiBXeWtvcnp5c3RhaiBtZXRvZHkgZ3JhZmljem5lIGRvc3TEmXBuZSB3IHBha2llY2llIGdncGxvdDIuIE1pbGUgd2lkemlhbmUgcHJ6ZWtyb2plIG9yYXogb2Rwb3dpZWR6aSBuYSBweXRhbmlhIGJhZGF3Y3plIHphZGFuZSBuYSB3c3TEmXBpZS4NCg0KDQpgYGB7ciB9DQpzYWxhcmllc0ZGU2FuRiA8LSBzdWJzZXQoc2FsYXJpZXMsIEpvYlRpdGxlPT0iRmlyZWZpZ2h0ZXIiICYgQWdlbmN5PT0iU2FuIEZyYW5jaXNjbyIpDQpkaW0oc2FsYXJpZXNGRlNhbkYpDQpgYGANCkhpc3RvZ3JhbSB3eW5hZ3JvZHplxYQgZGxhIHN0cmHFvGFrw7N3IHogU2FuIEZyYW5jaXNjby4NCg0KYGBge3J9DQpzcl9tZWRpYW5hX2xpbmllIDwtIGRhdGEuZnJhbWUobmF6d2EgPSBjKCLFmnJlZG5pYSIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIk1lZGlhbmEiKSwNCiAgICAgICAgICAgICAgICAgICAgICAgd2FydG/Fm8SHICAgICA9IGMobWVhbihzYWxhcmllc0ZGU2FuRiRUb3RhbFBheSksDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbWVkaWFuKHNhbGFyaWVzRkZTYW5GJFRvdGFsUGF5KSkpDQoNCmdncGxvdChzYWxhcmllc0ZGU2FuRixhZXMoeCA9IFRvdGFsUGF5KSkgKw0KICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aD0xMDAwMCwgY29sb3I9InB1cnBsZSIsZmlsbD0ieWVsbG93IikgKw0KICBsYWJzKHggPSAiUGF5IChpbiBkb2xsYXJzKSIsIHkgPSAiRnJlcXVlbmN5IiwgdGl0bGUgPSAiSGlzdG9ncmFtIHd5bmFncm9kemXFhCBzdHJhxbxha8OzdyB3IFNhbiBGcmFuY2lzY28iKSArDQogIGdlb21fdmxpbmUoZGF0YT1zcl9tZWRpYW5hX2xpbmllLGFlcyh4aW50ZXJjZXB0ID0gd2FydG/Fm8SHLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxpbmV0eXBlID0gbmF6d2EsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sID0gbmF6d2EpLGxpbmV3aWR0aD0xLGNvbG9yPSJibHVlIikrDQogIHRoZW1lX21pbmltYWwoKQ0KICANCmBgYA0KDQojI1BvbmnFvGVqIGhpc3RvZ3JhbSAgem1pYW4gd3luYWdyb2R6ZcWEIHN0cmHFvGFrw7N3IHogU0YgbmEgcHJ6ZXN0cnplbmkgbGF0DQoNCmBgYHtyfQ0Kc3JfbWVkaWFuYV9saW5pZV8yMDEyIDwtIGRhdGEuZnJhbWUobmF6d2EgPSBjKCLFmnJlZG5pYSB6IDIwMTIiLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJNZWRpYW5hIHogMjAxMiIpLCB3YXJ0b8WbxIcgPSBjKG1lYW4oc2FsYXJpZXNGRlNhbkYkVG90YWxQYXlbc2FsYXJpZXNGRlNhbkYkWWVhcj09MjAxMl0pLA0KIG1lZGlhbihzYWxhcmllc0ZGU2FuRiRUb3RhbFBheVtzYWxhcmllc0ZGU2FuRiRZZWFyPT0yMDEyXSkpKQ0KDQpnZ3Bsb3Qoc2FsYXJpZXNGRlNhbkYsYWVzKHggPSBUb3RhbFBheSkpICsNCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGg9MjUwMDAsIGNvbG9yPSJwdXJwbGUiLGZpbGw9InllbGxvdyIpICsNCiAgbGFicyh4ID0gIlBheSAoaW4gZG9sbGFycykiLCB5ID0gIkZyZXF1ZW5jeSIsIHRpdGxlID0gIkhpc3RvZ3JhbSB3eW5hZ3JvZHplxYQgc3RyYcW8YWvDs3cgdyBTYW4gRnJhbmNpc2NvIikgKw0KICBnZW9tX3ZsaW5lKGRhdGE9c3JfbWVkaWFuYV9saW5pZSxhZXMoeGludGVyY2VwdCA9IHdhcnRvxZvEhywNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsaW5ldHlwZSA9IG5hendhLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbCA9IG5hendhKSxsaW5ld2lkdGg9MSxjb2xvcj0iYmx1ZSIpKw0KICB0aGVtZV9taW5pbWFsKCkrDQogIGZhY2V0X2dyaWQoLiB+IFllYXIpDQoNCmBgYA0KTW/FvGVteSB6YXV3YcW8ecSHLCDFvGUgdyBwb3LDs3duYW5pdSBkbyAyMDEyIHJva3UgdyAyMDEzIHJva3UgbGljemJhIHN0cmHFvGFrw7N3LCBrdMOzcnp5IHphcmFiaWFsaSBrd290xJkgb2Rwb3dpYWRhasSFY8SFIMWbcmVkbmllaiBpIG1lZGlhbmllIHogMjAxMiByb2t1LCB3enJvc8WCYS4gSmVkbmFrIHcgMjAxNCByb2t1IHdhcnRvxZtjaSB0ZSBwb3dyw7NjacWCeSBkbyBwb3ppb23Ds3cgemJsacW8b255Y2ggZG8gdHljaCB6IDIwMTIgcm9rdS4gxZp3aWFkY3p5IG8gdHltIHptaWFuYSDFm3JlZG5pZWdvIHd5bmFncm9kemVuaWEgdyBhbmFsaXpvd2FueWNoIGxhdGFjaCDigJMgcG8gd3pyb8WbY2llIHcgMjAxMyByb2t1IG9zacSFZ27EmcWCbyBvbm8gbmFqd3nFvHN6eSBwb3ppb20sIGJ5IG5hc3TEmXBuaWUgdyAyMDE0IHJva3Ugc3BhxZvEhyBwb25pxbxlaiB3YXJ0b8WbY2kgb2Rub3Rvd2FuZWogdyAyMDEyIHJva3UuDQoNCmBgYHtyfQ0KDQpzcl9tZWRpYW55X2xhdGEgPC0gZGF0YS5mcmFtZShSb2sgPSBjKDIwMTIsMjAxMywyMDE0KSwgDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICDFm3JlZG5pZSA9IGMobWVhbihzYWxhcmllc0ZGU2FuRiRUb3RhbFBheVtzYWxhcmllc0ZGU2FuRiRZZWFyPT0yMDEyXSksDQptZWFuKHNhbGFyaWVzRkZTYW5GJFRvdGFsUGF5W3NhbGFyaWVzRkZTYW5GJFllYXI9PTIwMTNdKSwNCm1lYW4oc2FsYXJpZXNGRlNhbkYkVG90YWxQYXlbc2FsYXJpZXNGRlNhbkYkWWVhcj09MjAxNF0pKSwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1lZGlhbnkgPSBjKG1lZGlhbihzYWxhcmllc0ZGU2FuRiRUb3RhbFBheVtzYWxhcmllc0ZGU2FuRiRZZWFyPT0yMDEyXSksDQptZWRpYW4oc2FsYXJpZXNGRlNhbkYkVG90YWxQYXlbc2FsYXJpZXNGRlNhbkYkWWVhcj09MjAxM10pLA0KbWVkaWFuKHNhbGFyaWVzRkZTYW5GJFRvdGFsUGF5W3NhbGFyaWVzRkZTYW5GJFllYXI9PTIwMTRdKSkpDQpgYGANCg0KIyNXeWtyZXMgcG9rYXp1asSFY3kgem1pYW55IMWbcmVkbmllaiBpIG1lZGlhbnkgcMWCYWMgZGxhIHN0cmHFvGFrw7N3IHcgU2FuIEZyYW5jaXNjbyB3IGxhdGFjaCAyMDEyLTIwMTQNCg0KYGBge3J9DQpnZ3Bsb3Qoc3JfbWVkaWFueV9sYXRhLCBhZXMoUm9rLCDFm3JlZG5pZSkpICsgDQogICAgZ2VvbV9yaWJib24oYWVzKHltaW4gPSDFm3JlZG5pZSwgeW1heCA9IG1lZGlhbnkpLGZpbGwgPSAicHVycGxlIikrDQogIGdlb21fbGluZShhZXMoeD1Sb2sseT1tZWRpYW55LGNvbG9yPSJNZWRpYW55IiksbHdkPTEpKw0KICBnZW9tX2xpbmUoYWVzKHg9Um9rLHk9xZtyZWRuaWUsY29sb3I9IsWacmVkbmllIiksbHdkPTEpKw0KICBsYWJzKHggPSAiWWVhciIsIHkgPSAiVG90YWwgUGF5IChpbiBEb2xsYXJzKSIsIHRpdGxlID0gIlptaWFueSByw7PFvG5pY3kgbWnEmWR6eSDFm3JlZG5pxIUgaSBtZWRpYW7EhSBuYSBwcnplc3RyemVuaSB0cnplY2ggbGF0IHfFm3LDs2Qgc3RyYcW8YWvDs3cgU2FuIEZyYW5jaXNjbyIpDQoNCmBgYA0KTW/FvGVteSB6YW9ic2Vyd293YcSHIHptaWFueSB3IMWbcmVkbmllaiBpIG1lZGlhbmllIHd5bmFncm9kemXFhCBzdHJhxbxha8OzdyB3IGFuYWxpem93YW55Y2ggbGF0YWNoLCBqYWsgd3Nwb21uaWFubyB3Y3plxZtuaWVqLiBDbyB3acSZY2VqLCByw7PFvG5pY2EgbWnEmWR6eSBtZWRpYW7EhSBhIMWbcmVkbmnEhSBzdGFqZSBzacSZIGNvcmF6IGJhcmR6aWVqIHdpZG9jem5hLCBwcnp5IGN6eW0gxZtyZWRuaWEgd3lwYWRhIG5hIG5pxbxzenltIHBvemlvbWllIHcgc3Rvc3Vua3UgZG8gbWVkaWFueS4gVG8gd3NrYXp1amUsIMW8ZSB3acSZa3N6b8WbxIcgc3RyYcW8YWvDs3cgb3RyenltdWplIHd5bmFncm9kemVuaWUgd3nFvHN6ZSBuacW8IHByemVjacSZdG5hIHDFgmFjYSB3IHRlaiBncnVwaWUgemF3b2Rvd2VqLg0KDQotIEphayBwxYJhY2EgcG9kc3Rhd293YSwgd3luYWdyb2R6ZW5pZSB6YSBuYWRnb2R6aW55IGkgxZt3aWFkY3plbmlhIHPEhSByb3pkemllbGFuZSBwb21pxJlkenkgcsOzxbxuZSBncnVweT8NCmBgYHtyfQ0Kc2FsYXJpZXNGRlNhbkYkQmVuZWZpdHM8LWFzLm51bWVyaWMoc2FsYXJpZXNGRlNhbkYkQmVuZWZpdHMpDQpzYWxhcmllc0ZGU2FuRiRPdmVydGltZVBheTwtYXMubnVtZXJpYyhzYWxhcmllc0ZGU2FuRiRPdmVydGltZVBheSkNCnNhbGFyaWVzRkZTYW5GJEJhc2VQYXk8LWFzLm51bWVyaWMoc2FsYXJpZXNGRlNhbkYkQmFzZVBheSkNCmBgYA0KDQoNCiMjUMWCYWNhIHBvZHN0YXdvd2ENCg0KYGBge3J9DQpzcl9tZWRpYW5hX2xpbmllX2Jhc2VwYXkgPC0gZGF0YS5mcmFtZShuYXp3YSA9IGMoIsWacmVkbmlhIGRsYSBGVCIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIk1lZGlhbmEgZGxhIEZUIiksIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHdhcnRvxZvEhz1jKG1lYW4oc2FsYXJpZXNGRlNhbkYkQmFzZVBheVtzYWxhcmllc0ZGU2FuRiRTdGF0dXM9PSdGVCddKSwNCiBtZWRpYW4oc2FsYXJpZXNGRlNhbkYkQmFzZVBheVtzYWxhcmllc0ZGU2FuRiRTdGF0dXM9PSdGVCddKSkpDQoNCmdncGxvdChzYWxhcmllc0ZGU2FuRixhZXMoeCA9IEJhc2VQYXkpKSArDQogIGdlb21faGlzdG9ncmFtKGJpbndpZHRoPTI1MDAwLCBjb2xvcj0iZ3JlZW4iLGZpbGw9ImJsdWUiKSArDQogIGxhYnMoeCA9ICJCYXNlIFBheSIsIHkgPSAiRnJlcXVlbmN5IiwgdGl0bGUgPSAiSGlzdG9ncmFtIHDFgmFjeSBwb2RzdGF3b3dlaiBkbGEgc3RyYcW8YWvDs3cgdyBTYW4gRnJhbmNpc2NvIikgKw0KICBnZW9tX3ZsaW5lKGRhdGE9c3JfbWVkaWFuYV9saW5pZV9iYXNlcGF5LGFlcyh4aW50ZXJjZXB0ID0gd2FydG/Fm8SHLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxpbmV0eXBlID0gbmF6d2EsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sID0gbmF6d2EpLGxpbmV3aWR0aD0xLGNvbG9yPSJibGFjayIpKw0KICB0aGVtZV9taW5pbWFsKCkrDQogIGZhY2V0X2dyaWQoLiB+IFN0YXR1cykNCg0KYGBgDQoNCiMjV3luYWdyb2R6ZW5pZSB6YSBuYWRnb2R6aW55DQpgYGB7cn0NCnNyX21lZGlhbmFfbGluaWVfb3ZlcnRpbWUgPC0gZGF0YS5mcmFtZShuYXp3YSA9IGMoIsWacmVkbmlhIiwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiTWVkaWFuYSIpLCB3YXJ0b8WbxIcgPSBjKG1lYW4oc2FsYXJpZXNGRlNhbkYkT3ZlcnRpbWVQYXkpLA0KIG1lZGlhbihzYWxhcmllc0ZGU2FuRiRPdmVydGltZVBheSkpKQ0KDQpnZ3Bsb3Qoc2FsYXJpZXNGRlNhbkYsYWVzKHggPSBPdmVydGltZVBheSkpICsNCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGg9MjUwMDAsIGNvbG9yPSJncmVlbiIsZmlsbD0iYmx1ZSIpICsNCiAgbGFicyh4ID0gIk92ZXJ0aW1lIFBheSIsIHkgPSAiRnJlcXVlbmN5IiwgdGl0bGUgPSAiSGlzdG9ncmFtIHDFgmFjeSB6YSBuYWRnb2R6aW55IGRsYSBzdHJhxbxha8OzdyB3IFNhbiBGcmFuY2lzY28iKSArDQogIGdlb21fdmxpbmUoZGF0YT1zcl9tZWRpYW5hX2xpbmllX292ZXJ0aW1lLGFlcyh4aW50ZXJjZXB0ID0gd2FydG/Fm8SHLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxpbmV0eXBlID0gbmF6d2EsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sID0gbmF6d2EpLGxpbmV3aWR0aD0xLGNvbG9yPSJibGFjayIpKw0KICB0aGVtZV9taW5pbWFsKCkrDQogIGZhY2V0X2dyaWQoLiB+IFN0YXR1cykNCg0KYGBgDQoNCiMjQmVuZWZpdHkNCmBgYHtyfQ0Kc3JfbWVkaWFuYV9saW5pZV9iZW5lZml0cyA8LSBkYXRhLmZyYW1lKG5hendhID0gYygixZpyZWRuaWEiLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJNZWRpYW5hIiksIHdhcnRvxZvEhyA9IGMobWVhbihzYWxhcmllc0ZGU2FuRiRCZW5lZml0cyksDQogbWVkaWFuKHNhbGFyaWVzRkZTYW5GJEJlbmVmaXRzKSkpDQoNCmdncGxvdChzYWxhcmllc0ZGU2FuRixhZXMoeCA9IEJlbmVmaXRzKSkgKw0KICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aD0xMDAwMCwgY29sb3I9InJlZCIsZmlsbD0iYmx1ZSIpICsNCiAgbGFicyh4ID0gIkJlbmVmaXRzIiwgeSA9ICJGcmVxdWVuY3kiLCB0aXRsZSA9ICJIaXN0b2dyYW0gxZt3aWFkY3plxYQvQmVuZWZpdMOzdyBkbGEgc3RyYcW8YWvDs3cgdyBTYW4gRnJhbmNpc2NvIikgKw0KICBnZW9tX3ZsaW5lKGRhdGE9c3JfbWVkaWFuYV9saW5pZV9iZW5lZml0cyxhZXMoeGludGVyY2VwdCA9IHdhcnRvxZvEhywNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsaW5ldHlwZSA9IG5hendhLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbCA9IG5hendhKSxsaW5ld2lkdGg9MSxjb2xvcj0iYmxhY2siKSsNCiAgdGhlbWVfbWluaW1hbCgpKw0KICBmYWNldF9ncmlkKC4gfiBTdGF0dXMpDQoNCmBgYA0KTW/FvGVteSB6YXV3YcW8ecSHLCDFvGUgYnVkxbxldCBwcnplem5hY3pvbnkgbmEgd3luYWdyb2R6ZW5pZSBwb2RzdGF3b3dlLCBwxYJhdG5vxZtjaSB6YSBuYWRnb2R6aW55IG9yYXogxZt3aWFkY3plbmlhIHLDs8W8bmkgc2nEmSB3IHphbGXFvG5vxZtjaSBvZCB0ZWdvLCBjenkgc3RyYcW8YWN5IHPEhSB6YXRydWRuaWVuaSB3IHBlxYJueW0gd3ltaWFyemUgZ29kemluIChGVCksIGN6eSBuYSBww7PFgiBldGF0dSAoU1QpLiBaZ29kbmllIHogcHJ6ZXdpZHl3YW5pYW1pLCB3c3p5c3RraWUgdGUgd3lkYXRraSBzxIUgbmnFvHN6ZSB3IHByenlwYWRrdSBzdHJhxbxha8OzdyBwcmFjdWrEhWN5Y2ggdyBuaWVwZcWCbnltIHd5bWlhcnplIGdvZHppbi4NCg0KDQpPc3phY293YW5pZSBnxJlzdG/Fm2NpIHJvemvFgmFkdSB3eW5hZ3JvZHplxYQgc3RyYcW8YWvDs3cgdyBTYW4gRnJhbmNpc2NvOg0KYGBge3J9DQojIEtlcm5lbCBkZW5zaXR5IGVzdGltYXRpb24NCmQgPC0gZGVuc2l0eShzYWxhcmllc0ZGU2FuRiRUb3RhbFBheSwNCiAgICAgICAgICAgICBrZXJuZWwgPSAiZ2F1c3NpYW4iLA0KICAgICAgICAgICAgIGJ3ID0gNTAwMDApDQojIEtlcm5lbCBkZW5zaXR5IHBsb3QNCnBsb3QoZCwgbHdkID0gMiwgbWFpbiA9ICJHYXVzc2lhbiBrZXJuZWwiKQ0KYGBgDQpNYW15IHR1dGFqIHByenlrxYJhZCByb3prxYJhZHUgYmltb2RhbG5lZ28sIGt0w7NyeSBwb21hZ2EgenJvenVtaWXEhyB3Y3plxZtuaWVqIHphdXdhxbxvbmUgemphd2lza28sIMW8ZSBtZWRpYW5hIHByemV3ecW8c3phIMWbcmVkbmnEhS4gV3luaWthIHRvIHogb2JlY25vxZtjaSB6bmFjem5laiBncnVweSBzdHJhxbxha8OzdyAoc3Rhbm93acSFY2VqIGRydWdpZSBsb2thbG5lIG1ha3NpbXVtKSwga3TDs3J6eSB6YXJhYmlhasSFIG5pZXdpZWxraWUga3dvdHkgdyBwcnplZHppYWxlIDwwOyA1MCAwMDAgVVNEPi4gSWNoIHphcm9ia2kgem5hY3rEhWNvIG9ibmnFvGFqxIUgxZtyZWRuacSFIHDFgmFjLCBhbGUgaWNoIGxpY3piYSBuaWUgamVzdCBuYSB0eWxlIGR1xbxhLCBhYnkgdyBpc3RvdG55IHNwb3PDs2Igd3DFgnluxIXEhyBuYSBvYm5pxbxlbmllIG1lZGlhbnkuDQoNCg==