Dlaczego kwantylowa?
Dlaczego potrzebujemy regresji kwantylowej (QR)?
W szczególności, QR:
jest odporna na punkty odstające i wpływowe
nie zakłada stałej wariancji (znanej jako homoskedastyczność) dla
zmiennej odpowiedzi lub reszt
nie zakłada normalności ale główną zaletą QR w porównaniu z
regresją liniową (LR) jest to, że QR bada różne wartości zmiennej
odpowiedzi, a nie tylko średnią, i dostarcza w związku z tym
pełniejszego obrazu związków między zmiennymi!
Wprowadzenie
Regresja kwantylowa (ang. quantile regression) została zaproponowana
przez Koenkera i Bassetta (1978). Szczególny przypadek regresji
kwantylowej dla kwantyla rzędu 0,5 (czyli mediany) jest równoważny
estymatorowi LAD (ang. Least Absolute Deviation) – minimalizuje sumę
bezwzględnych błędów.
Wprowadzenie różnych kwantyli regresji daje pełniejszy opis rozkładów
warunkowych zwłaszcza w przypadku rozkładów asymetrycznych lub
uciętych.
Regresja kwantylowa jest kolejną wariacją na temat najmniejszych
kwadratów . Stratą jest współczynnik \(l_1\) funkcji:
\[
\phi(u) = \tau\max(u,0) - (1-\tau)\max(-u,0) = \frac{1}{2}|u| +
\left(\tau - \frac{1}{2}\right)u,
\]
gdzie \(\tau \in (0,1)\) oznacza
konkretny kwantyl. Problemem jak poprzednio jest minimalizacja
całkowitej straty resztowej. Model ten jest powszechnie stosowany w
ekologii, ochronie zdrowia i innych dziedzinach, gdzie sama średnia nie
wystarcza do uchwycenia złożonych zależności między zmiennymi.
Wymagania
Wymagana jest jedna liczbowa zmienna zależna. Zmienna przewidywana
musi być zmienną ilościową. Predyktory mogą być zmiennymi ilościowymi
lub sztucznymi zmiennymi w przypadku predyktorów jakościowych. Aby można
było uruchomić analizę, wymagany jest wyraz wolny lub co najmniej jeden
predyktor.
Regresja kwantylowa nie czyni założeń dotyczących rozkładu zmiennej
przewidywanej i jest odporna na wpływ obserwacji odstających.
Analiza kwantylowa jest pokrewna regresji metodą najmniejszych
kwadratów.
Przykład 1.
Wykorzystamy przykład z pakietu quantreg.
Jaki jest związek między całkowitym dochodem gospodarstwa domowego a
odsetkiem dochodów wydatkowanych na żywność? Prawo Engela w ekonomii
głosi, że w miarę wzrostu dochodów, część dochodów wydatkowanych na
żywność spada, nawet jeśli wydatki na żywność bezwzględnie rosną.
Stosując regresję kwantylową do tych danych, można określić, jakie
wydatki na żywność ponosi 90% rodzin (dla 100 rodzin z danym dochodem),
gdy nie interesują nas średnie wydatki na żywność.
Dane, które wykorzystamy - to zbiór “engel” - dane dotyczące wydatków
na żywność. Jest to zbiór danych regresyjnych składający się z 235
obserwacji dotyczących dochodów i wydatków na żywność dla belgijskich
gospodarstw domowych klasy robotniczej.

Powyższy wykres przedstawia dopasowanie regresji kwantylowej dla
\(\tau = (0.1, 0.25, 0.5, 0.75, 0.90,
0.95)\). Dopasowanie KMNK to gruba czarna linia.
Poniżej znajduje się tabela z oszacowanymi współczynnikami.
knitr::kable(fits, format = "html", caption = "Oszacowania z KMNK oraz `quantreg`") %>%
kable_styling("striped") %>%
column_spec(1:8, background = "#ececec")
Oszacowania z KMNK oraz quantreg
|
|
OLS
|
\(\tau_{0.10}\)
|
\(\tau_{0.25}\)
|
\(\tau_{0.50}\)
|
\(\tau_{0.75}\)
|
\(\tau_{0.90}\)
|
\(\tau_{0.95}\)
|
|
(Intercept)
|
147.4753885
|
110.1415742
|
95.4835396
|
81.4822474
|
62.3965855
|
67.3508721
|
64.1039632
|
|
income
|
0.4851784
|
0.4017658
|
0.4741032
|
0.5601806
|
0.6440141
|
0.6862995
|
0.7090685
|
Ok, możemy to zrobić bardziej przejrzyście i sformatować w ładnej
tabeli wyników:
##
## Wyniki regresji kwantylowych
## ==========================================
## Dependent variable:
## -----------------------------
## foodexp
## (1) (2) (3)
## ------------------------------------------
## income 0.474*** 0.560*** 0.644***
## (0.029) (0.028) (0.023)
##
## Constant 95.484*** 81.482*** 62.397***
## (21.392) (19.251) (16.305)
##
## ------------------------------------------
## Observations 235 235 235
## ==========================================
## Note: *p<0.1; **p<0.05; ***p<0.01
Finalnie, zaprezentujmy wyłącznie te 3 modele na wykresie:

Przykład 2.
Tutaj przeprowadzimy testy użycia pakietu quantreg, wykorzystując
wbudowany zbiór danych “mtcars”. Zmienna
“mpg” oznacza spalanie samochodów
(mile/galon).
Zamodulejmy zależność regresyjną dla tej zmiennej od kilku
predyktorów.
Najpierw oszacujmy regresję KMNK:
kmnk <- lm(mpg ~ disp + hp + factor(am) + factor(vs), data = mtcars)
summary(kmnk)
##
## Call:
## lm(formula = mpg ~ disp + hp + factor(am) + factor(vs), data = mtcars)
##
## Residuals:
## Min 1Q Median 3Q Max
## -4.7981 -1.9532 0.0111 1.5665 5.6321
##
## Coefficients:
## Estimate Std. Error t value Pr(>|t|)
## (Intercept) 24.832119 2.890418 8.591 3.32e-09 ***
## disp -0.008304 0.010087 -0.823 0.41757
## hp -0.037623 0.013846 -2.717 0.01135 *
## factor(am)1 4.419257 1.493243 2.960 0.00634 **
## factor(vs)1 2.052472 1.627096 1.261 0.21794
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
##
## Residual standard error: 2.812 on 27 degrees of freedom
## Multiple R-squared: 0.8104, Adjusted R-squared: 0.7823
## F-statistic: 28.85 on 4 and 27 DF, p-value: 2.13e-09
Teraz oszacujmy warunkowe regresje kwantylowe na różnych kwantylach,
błąd standardowy uzyskany przez bootstrap.
Zauważ, że istnieje gradient we współczynnikach kwantylowych
hp, jak również disp. Znak
disp odwraca się, również współczynnik na czynniku
am jest różny w zależności od kwantyli:
kwantyle <- c(0.25, 0.50, 0.75)
reg_kwantylowa <- rq(mpg ~ disp + hp + factor(am),tau = kwantyle,data = mtcars)
summary(reg_kwantylowa, se = "boot")
##
## Call: rq(formula = mpg ~ disp + hp + factor(am), tau = kwantyle, data = mtcars)
##
## tau: [1] 0.25
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 25.34665 1.52409 16.63070 0.00000
## disp -0.02441 0.00783 -3.11780 0.00419
## hp -0.01672 0.01487 -1.12393 0.27059
## factor(am)1 1.39719 1.33824 1.04406 0.30539
##
## Call: rq(formula = mpg ~ disp + hp + factor(am), tau = kwantyle, data = mtcars)
##
## tau: [1] 0.5
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 27.49722 1.81353 15.16229 0.00000
## disp -0.02253 0.01659 -1.35799 0.18531
## hp -0.02713 0.02432 -1.11549 0.27412
## factor(am)1 3.37328 2.00874 1.67930 0.10422
##
## Call: rq(formula = mpg ~ disp + hp + factor(am), tau = kwantyle, data = mtcars)
##
## tau: [1] 0.75
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 28.06384 1.60415 17.49448 0.00000
## disp 0.00445 0.01494 0.29795 0.76794
## hp -0.06662 0.01859 -3.58294 0.00127
## factor(am)1 7.91402 2.47587 3.19646 0.00344
Testy współczynników
Użyjemy funkcji rq.anova z pakietu regresji kwantylowej, aby
przeprowadzić test WALDA. Pamiętaj, że test WALDA mówi, że biorąc pod
uwagę nieograniczone oszacowania modelu, przetestujemy hipotezę zerową
mówiącą, że współczynniki spełniają pewne liniowe ograniczenia.
Aby ją przetestować, użyjemy obiektu zwróconego z uruchomienia
rq z różnymi liczbami kwantyli i ustawimy
opcję joint na true lub false. Gdy
joint jest true: “równość współczynników
kierunkowych powinna być wykonana jako wspólne testy na wszystkich
parametrach nachylenia”, gdy joint jest false:
“należy zgłaszać oddzielne testy na każdym z parametrów nachylenia”.
Zauważ, że testy kwantylowe są testami “linii równoległej”. Oznacza
to, że powinniśmy wyjąć różne x-wyrazy_wolne dla każdego kwantyla,
ponieważ reprezentują one poziomy rozkładów warunkowych. Jeśli jednak
współczynniki kwantyli dla współczynnikow są takie same, to nie ma
efektów specyficznych dla kwantyli, wystarczą efekty średnie.
Badanie statystycznej różnicy między 25. i 50. kwantylem
warunkowym:
Biorąc pod uwagę powyższe oszacowania kwantyli, różnica między
kwantylami 0,25 i 0,50 istnieje, ale czy są one wystarczająco duże, aby
być statystycznie różne? Jaka jest wartość p? Przeglądając poniższe
wyniki, nie są one statystycznie różne!
Po pierwsze, joint = TRUE. To nie jest testowanie, czy współczynnik
na disp jest taki sam jak współczynnik na hp. To jest wspólne
testowanie, czy współczynniki dla różnych kwantyli disp i różnych
kwantyli hp są takie same dla każdej zmiennej.
kwantyle <- c(0.25, 0.50)
reg_kwantylowa <- rq(mpg ~ disp + hp + factor(am),tau = kwantyle, data = mtcars)
anova(reg_kwantylowa, test = "Wald", joint=TRUE)
## Quantile Regression Analysis of Deviance Table
##
## Model: mpg ~ disp + hp + factor(am)
## Joint Test of Equality of Slopes: tau in { 0.25 0.5 }
##
## Df Resid Df F value Pr(>F)
## 1 3 61 0.8421 0.4761
Po drugie, joint = False:
anova(reg_kwantylowa, test = "Wald", joint=FALSE)
## Quantile Regression Analysis of Deviance Table
##
## Model: mpg ~ disp + hp + factor(am)
## Tests of Equality of Distinct Slopes: tau in { 0.25 0.5 }
##
## Df Resid Df F value Pr(>F)
## disp 1 63 0.0305 0.8619
## hp 1 63 0.5461 0.4627
## factor(am)1 1 63 1.3500 0.2497
Badanie statystycznej różnicy między 25, 50 i 75 kwantylem
warunkowym:
Pierwszy kwartyl i mediana nie wydają się być statystycznie różne,
teraz dołączymy trzeci kwartyl. Jak widać wcześniej, kwartyle wspólnie
wykazują gradient. Teraz możemy zobaczyć, że disp,
hp i am są oddzielnie statystycznie
różne.
Po pierwsze, joint = TRUE:
kwantyle <- c(0.25, 0.50, 0.75)
reg_kwantylowa <- rq(mpg ~ disp + hp + factor(am),tau = kwantyle, data = mtcars)
anova(reg_kwantylowa, test = "Wald", joint=TRUE)
## Quantile Regression Analysis of Deviance Table
##
## Model: mpg ~ disp + hp + factor(am)
## Joint Test of Equality of Slopes: tau in { 0.25 0.5 0.75 }
##
## Df Resid Df F value Pr(>F)
## 1 6 90 3.3173 0.005367 **
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Po drugie, joint = False:
anova(reg_kwantylowa, test = "Wald", joint=FALSE)
## Quantile Regression Analysis of Deviance Table
##
## Model: mpg ~ disp + hp + factor(am)
## Tests of Equality of Distinct Slopes: tau in { 0.25 0.5 0.75 }
##
## Df Resid Df F value Pr(>F)
## disp 2 94 5.4903 0.005558 **
## hp 2 94 6.7221 0.001868 **
## factor(am)1 2 94 7.2758 0.001154 **
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Dobroć dopasowania
Możemy obliczyć współczynniki dobroci dopasowania regresji
kwantylowej z wykorzystaniem reszt i reszt bezwarunkowych:
goodfit(resid, resid_nl, tau)
Miara dobroci dopasowania dla regresji kwantylowej jest szacowana
jako 1 minus stosunek sumy odchyleń bezwzględnych w modelach w pełni
sparametryzowanych do sumy odchyleń bezwzględnych w zerowym
(bezwarunkowym) modelu kwantylowym.
Wartości te są przydatne do porównań między modelami kwantylowymi,
ale nie są porównywalne ze standardowymi współczynnikami determinacji.
Te ostatnie oparte są na wariancji odchyleń kwadratowych, natomiast
wartości dobroci dopasowania dla regresji kwantylowej oparte są na
odchyleniach bezwzględnych. Wartości dobroci dopasowania zawsze będą
mniejsze niż wartości R2.
## model kwantylowy
model1 <- rq(mpg ~ disp + hp + factor(am),tau = 0.5, data = mtcars)
reszty1 <- resid(model1)
## bezwarunkowy (pusty) model kwantylowy
model2 <- rq(mpg ~ 1, tau = 0.5,data=mtcars)
reszty2 <- resid(model2)
goodfit(reszty1, reszty2, 0.5)
## [1] 0.5403311
## r2 modelu KMNK dla porównania
model_lm <- lm(mpg ~ disp + hp + factor(am), data = mtcars)
summary(model_lm)$r.squared
## [1] 0.7992061
Zadanie
Teraz Wasza kolej ;-)
Waszym zadaniem dzisiaj jest zamodelowanie - porównanie KMNK oraz
regresji kwantylowej (różno-poziomowej) dla zmiennej “earnings” -
wynagrodzenia.
Dobierz i przetestuj predyktory, kwantyle dla modeli. Wykonaj testy
różnic współczynnikow dla finalnych modeli.
W przypadku problemów - obejrzyj video tutorial (włącz polskie
napisy) oraz wejdź na jego stronę ze źródłami. Możesz również
wykorzystać w/w przykłady.
data("CPSSW9298")
dane <- CPSSW9298
dane92 <- dane %>% filter(year==1992)
dane98 <- dane %>% filter(year==1998)
Analiza zmiennej “earnings” w 1992 roku
plot <- ggplot(data = dane92) +
geom_point(mapping = aes(x = age, y = earnings), color = "purple")+ labs(title= "Wykres rozrzutu earnings względem age")
plot

Budowa i porównanie modeli za pomocą KMNK oraz regresji
kwantylowej.
model_liniowy <- lm(earnings~degree+gender+age, data=dane92)
model_qr1 <- rq(earnings~degree+gender+age, data=dane92, tau=0.25)
model_qr2 <- rq(earnings~degree+gender+age, data=dane92, tau=0.5)
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
model_qr3 <- rq(earnings~degree+gender+age, data=dane92, tau=0.75)
summary(model_qr1)
##
## Call: rq(formula = earnings ~ degree + gender + age, tau = 0.25, data = dane92)
##
## tau: [1] 0.25
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 1.92427 0.58803 3.27240 0.00107
## degreebachelor 3.41880 0.13130 26.03821 0.00000
## genderfemale -1.14850 0.11186 -10.26773 0.00000
## age 0.18222 0.01990 9.15846 0.00000
summary(model_qr3)
##
## Call: rq(formula = earnings ~ degree + gender + age, tau = 0.75, data = dane92)
##
## tau: [1] 0.75
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 1.24944 0.84896 1.47173 0.14114
## degreebachelor 5.17262 0.17629 29.34186 0.00000
## genderfemale -2.52525 0.16201 -15.58689 0.00000
## age 0.40965 0.02869 14.27611 0.00000
AIC(model_liniowy, model_qr1, model_qr2, model_qr3)
## df AIC
## model_liniowy 5 45918.10
## model_qr1 4 45184.45
## model_qr2 4 45737.55
## model_qr3 4 48091.87
AIC jest najniższe dla modelu “model_qr1”, co oznacza że model
regresji kwantylowej z tau=0.25 w najlepszy sposób wyjaśnia zmienność
tych danych spośród tych modeli. Model liniowy stworzony metodą KMNK
gorzej objaśnia zmienną niż model kwantylowy dla pierwszego kwantyla i
dla mediany.
W grupie osób, których zarobki znajdują się w pierwszym kwantylu (25%
najmniej zarabiających) posiadanie wyższego wykształcenia przykłada się
na zarobki wyższe srednio o 3,42 jednostki pieniężnej więcej względem
osób z wykształceniem średnim. Wsród 25% najlepiej zarabiających
(tau=0,75) różnica w zarobkach jest większa i wynosi 5,17 jednostek
pieniężnych na korzyść osób wyżej wykształconych. Kobiety w pierwszej
grupie zarabiają średnio o 1,14 jednostek mniej od mężczyzn, zaś w
grupie o najwyższych zarobkach ta różnica również jest większa, 2,52
jednotki na niekorzyść kobiet. Wraz z wiekiem w obu grupach zarobki
rosną przy czym również ta różnica jest wyższa w grupie
najbogatszych.
model_rq5 <- rq(earnings~degree+gender+age, data=dane92, tau = seq(0.25, 0.75, 0.25))
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
intercept_slope <- model_rq5 %>%
coef() %>%
t() %>%
data.frame() %>%
rename(intercept = X.Intercept., slope = age) %>%
mutate(quantile = row.names(.))
ggplot() +
geom_point(data = dane92, aes(age, earnings), alpha = 0.5) +
geom_abline(data = intercept_slope, aes(intercept = intercept, slope = slope, color = quantile)) +
theme_minimal() +
labs(x = "Wiek", y = "Zarobki", title = "Regresje kwantylowe z tau = 0.25, 0.50 oraz 0.75")

hist(dane92$earnings)

hist(log(dane92$earnings))

Na podstawie histogramu przedstawiającego rozkład zmiennej można
zauważyć, że zlogarytmowanie zmiennej może pomóc w uzyskaniu lepszego
modelu.
dane92$log_earnings <- log(dane92$earnings)
plot2 <- ggplot(data = dane92) +
geom_point(mapping = aes(x = age, y = log_earnings), color = "purple")+ labs(title= "Wykres rozrzutu log(earnings) względem age")
plot2

Budowa i porównanie modeli za pomocą KMNK oraz regresji kwantylowej
dla zmiennej earnings zlogarytmowanej.
model_liniowy2 <- lm(log_earnings~degree+gender+age, data=dane92)
model_qrlog1 <- rq(log_earnings~degree+gender+age, data=dane92, tau=0.25)
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
model_qrlog2 <- rq(log_earnings~degree+gender+age, data=dane92, tau=0.5)
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
model_qrlog3 <- rq(log_earnings~degree+gender+age, data=dane92, tau=0.75)
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
summary(model_qrlog1)
##
## Call: rq(formula = log_earnings ~ degree + gender + age, tau = 0.25,
## data = dane92)
##
## tau: [1] 0.25
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 1.32454 0.08001 16.55439 0.00000
## degreebachelor 0.41185 0.01520 27.10356 0.00000
## genderfemale -0.14206 0.01490 -9.53162 0.00000
## age 0.02206 0.00266 8.29927 0.00000
AIC(model_liniowy2, model_qrlog1, model_qrlog2, model_qrlog3)
## df AIC
## model_liniowy2 5 8750.943
## model_qrlog1 4 10873.415
## model_qrlog2 4 9249.473
## model_qrlog3 4 9527.585
W przypadku zlogarytmowanej zmiennej objaśnianej lepszą objaśnialność
ma model liniowy (najniższe AIC) niż modele regresji kwantylowej.
Spośród modeli regresji kwartylowej najlepszy zdaje się ten dla
mediany.
Według wybranego modelu, czyli modelu liniowego osoby posiadające
wykształcenie wyższe zarabiają średnio o 0,41 jednostki pieniężnej na
godzinę więcej niż osoby posiadające wykształcenie średnie. Kobiety
zarabiają przeciętnie 0,14 jednostki pieniężnej mniej niż mężczyźni.
Wraz ze starzeniem się, z każdym kolejnym rokiem zarabia się średno 0,02
jednostki pieniężne na godzinę więcej.
Analiza zmiennej “earnings” w 1998 roku
plot3 <- ggplot(data = dane98) +
geom_point(mapping = aes(x = age, y = earnings), color = "pink")+ labs(title= "Wykres rozrzutu earnings względem age")
plot3

Budowa i porównanie modeli za pomocą KMNK oraz regresji
kwantylowej.
model_liniowy3 <- lm(earnings~degree+gender+age, data=dane98)
model_qr6 <- rq(earnings~degree+gender+age, data=dane98, tau=0.25)
model_qr7 <- rq(earnings~degree+gender+age, data=dane98, tau=0.5)
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
model_qr8 <- rq(earnings~degree+gender+age, data=dane98, tau=0.75)
summary(model_qr6)
##
## Call: rq(formula = earnings ~ degree + gender + age, tau = 0.25, data = dane98)
##
## tau: [1] 0.25
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 3.12432 0.79243 3.94273 0.00008
## degreebachelor 3.80303 0.16637 22.85934 0.00000
## genderfemale -1.64320 0.15013 -10.94525 0.00000
## age 0.18444 0.02659 6.93726 0.00000
summary(model_qr8)
##
## Call: rq(formula = earnings ~ degree + gender + age, tau = 0.75, data = dane98)
##
## tau: [1] 0.75
##
## Coefficients:
## Value Std. Error t value Pr(>|t|)
## (Intercept) 1.44231 1.29329 1.11522 0.26480
## degreebachelor 6.25000 0.27978 22.33862 0.00000
## genderfemale -3.36538 0.24212 -13.89986 0.00000
## age 0.48077 0.04298 11.18579 0.00000
AIC(model_liniowy3, model_qr6, model_qr7, model_qr8)
## df AIC
## model_liniowy3 5 38436.17
## model_qr6 4 37529.01
## model_qr7 4 38116.57
## model_qr8 4 40304.82
W 1998 roku podobnie jak w 1992 roku model regresji kwantylowej z
tau=0.25 (najniższe AIC) w najlepszy sposób wyjaśnia zmienność zarobków.
Model liniowy gorzej objaśnia zmienną niż model kwantylowy dla
pierwszego kwantyla i dla mediany.
W 1998 roku w grupie osób, których zarobki znajdują się w pierwszym
kwantylu (25% najmniej zarabiających) posiadanie wyższego wykształcenia
przykłada się na zarobki wyższe srednio o 3,8 jednostki pieniężnej
więcej względem osób z wykształceniem średnim. Wsród 25% najlepiej
zarabiających (tau=0,75) różnica w zarobkach jest większa i wynosi 6,25
jednostek pieniężnych na korzyść osób wyżej wykształconych. Kobiety w
pierwszej grupie zarabiają średnio o 1,64 jednostek mniej od mężczyzn,
zaś w grupie o najwyższych zarobkach ta różnica również jest większa,
3,36 jednotki na niekorzyść kobiet. Wraz z wiekiem w obu grupach zarobki
rosną przy czym również ta różnica jest wyższa w grupie
najbogatszych.
Można zauważyć, że na przestrzeni lat od 1992 do 1998 roku w obu
grupach, tej zarabiąjących najmniej, jak i tej zarabiających najwięcej
różnice w osiąganych zarobkach ze względu na posiadane wykształcenie,
płeć oraz wiek pogłębiły się. Posiadanie wyższego wykształcenia bardziej
podwyższa zarobki, kobiety zarabiają jeszcze mniej względem mężczyzn, a
zarobki jeszcze bardziej rosną wraz z wiekiem (w przypadku tej zmiennej
głównie w grupie najlepiej zarabiających).
model_rq4 <- rq(earnings~degree+gender+age, data=dane98, tau = seq(0.25, 0.75, 0.25))
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
intercept_slope <- model_rq4 %>%
coef() %>%
t() %>%
data.frame() %>%
rename(intercept = X.Intercept., slope = age) %>%
mutate(quantile = row.names(.))
ggplot() +
geom_point(data = dane98, aes(age, earnings), alpha = 0.5) +
geom_abline(data = intercept_slope, aes(intercept = intercept, slope = slope, color = quantile)) +
theme_minimal() +
labs(x = "Wiek", y = "Zarobki", title = "Regresje kwantylowe z tau = 0.25, 0.50 oraz 0.75")

hist(dane98$earnings)

hist(log(dane98$earnings))

Również na podstawie histogramu danych z 1998 roku można zauważyć, że
zlogarytmowanie zmiennej może pomóc w uzyskaniu lepszego modelu.
dane98$log_earnings <- log(dane98$earnings)
plot4 <- ggplot(data = dane98) +
geom_point(mapping = aes(x = age, y = log_earnings), color = "pink")+ labs(title= "Wykres rozrzutu log(earnings) względem age")
plot4

Budowa i porównanie modeli za pomocą KMNK oraz regresji kwantylowej
dla zmiennej earnings zlogarytmowanej.
model_liniowy4 <- lm(log_earnings~degree+gender+age, data=dane98)
model_qrlog6 <- rq(log_earnings~degree+gender+age, data=dane98, tau=0.25)
## Warning in rq.fit.br(x, y, tau = tau, ...): Solution may be nonunique
model_qrlog7 <- rq(log_earnings~degree+gender+age, data=dane98, tau=0.5)
model_qrlog8 <- rq(log_earnings~degree+gender+age, data=dane98, tau=0.75)
summary(model_liniowy4)
##
## Call:
## lm(formula = log_earnings ~ degree + gender + age, data = dane98)
##
## Residuals:
## Min 1Q Median 3Q Max
## -1.96583 -0.27644 0.02536 0.30209 1.50215
##
## Coefficients:
## Estimate Std. Error t value Pr(>|t|)
## (Intercept) 1.78845 0.06230 28.71 <2e-16 ***
## degreebachelor 0.38277 0.01173 32.64 <2e-16 ***
## genderfemale -0.18003 0.01182 -15.23 <2e-16 ***
## age 0.02142 0.00207 10.35 <2e-16 ***
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
##
## Residual standard error: 0.4457 on 5907 degrees of freedom
## Multiple R-squared: 0.1828, Adjusted R-squared: 0.1824
## F-statistic: 440.5 on 3 and 5907 DF, p-value: < 2.2e-16
AIC(model_liniowy4, model_qrlog6, model_qrlog7, model_qrlog8)
## df AIC
## model_liniowy4 5 7228.160
## model_qrlog6 4 8725.172
## model_qrlog7 4 7555.391
## model_qrlog8 4 7968.480
W 1998 roku tak samo jak w 1992 w przypadku zlogarytmowanej zmiennej
objaśnianej dotyczącej zarobków lepszą objaśnialność ma model liniowy
(najniższe AIC) niż modele regresji kwantylowej. Spośród modeli regresji
kwartylowej najlepszy zdaje się ten dla mediany.
Według wybranego modelu, czyli modelu liniowego osoby posiadające
wykształcenie wyższe zarabiają średnio o 0,38 jednostki pieniężnej na
godzinę więcej niż osoby posiadające wykształcenie średnie. Kobiety
zarabiają przeciętnie 0,18 jednostki pieniężnej mniej niż mężczyźni.
Wraz ze starzeniem się, z każdym kolejnym rokiem zarabia się średno 0,02
jednostki pieniężne na godzinę więcej.
LS0tDQp0aXRsZTogJ05pZWtsYXN5Y3puZSBtZXRvZHkgc3RhdHlzdHlraScNCnN1YnRpdGxlOiAnUmVncmVzamEga3dhbnR5bG93YScNCmRhdGU6ICJgciBTeXMuRGF0ZSgpYCINCmF1dGhvcjogIkp1bGlhICBDaHnFgmEiDQpvdXRwdXQ6DQogIGh0bWxfZG9jdW1lbnQ6IA0KICAgIHRoZW1lOiBjZXJ1bGVhbg0KICAgIGhpZ2hsaWdodDogdGV4dG1hdGUNCiAgICBmb250c2l6ZTogMTBwdA0KICAgIHRvYzogeWVzDQogICAgY29kZV9kb3dubG9hZDogeWVzDQogICAgdG9jX2Zsb2F0Og0KICAgICAgY29sbGFwc2VkOiBubw0KICAgIGRmX3ByaW50OiBkZWZhdWx0DQogICAgdG9jX2RlcHRoOiA1DQplZGl0b3Jfb3B0aW9uczogDQogIG1hcmtkb3duOiANCiAgICB3cmFwOiA3Mg0KLS0tDQoNCmBgYHtyIHByZXJlcXMsIG1lc3NhZ2UgPSBGQUxTRSwgZWNobyA9IEZBTFNFfQ0KbGlicmFyeShDVlhSKQ0KbGlicmFyeShBRVIpDQpsaWJyYXJ5KHN0YXJnYXplcikNCmxpYnJhcnkodGlkeXZlcnNlKQ0KbGlicmFyeShrYWJsZUV4dHJhKQ0KbGlicmFyeShQb2dyb21jeURhbnljaCkNCmxpYnJhcnkoTWF0cml4KQ0KbGlicmFyeShxdWFudHJlZykNCmxpYnJhcnkoV1JURFN0aWRhbCkNCmBgYA0KDQojIyBEbGFjemVnbyBrd2FudHlsb3dhPw0KDQpEbGFjemVnbyBwb3RyemVidWplbXkgcmVncmVzamkga3dhbnR5bG93ZWogKFFSKT8NCg0KVyBzemN6ZWfDs2xub8WbY2ksIFFSOg0KDQotICAgamVzdCBvZHBvcm5hIG5hIHB1bmt0eSBvZHN0YWrEhWNlIGkgd3DFgnl3b3dlDQoNCi0gICBuaWUgemFrxYJhZGEgc3RhxYJlaiB3YXJpYW5jamkgKHpuYW5laiBqYWtvIGhvbW9za2VkYXN0eWN6bm/Fm8SHKSBkbGENCiAgICB6bWllbm5laiBvZHBvd2llZHppIGx1YiByZXN6dA0KDQotICAgbmllIHpha8WCYWRhIG5vcm1hbG5vxZtjaSBhbGUgZ8WCw7N3bsSFIHphbGV0xIUgUVIgdyBwb3LDs3duYW5pdSB6IHJlZ3Jlc2rEhQ0KICAgIGxpbmlvd8SFIChMUikgamVzdCB0bywgxbxlIFFSIGJhZGEgcsOzxbxuZSB3YXJ0b8WbY2kgem1pZW5uZWogb2Rwb3dpZWR6aSwNCiAgICBhIG5pZSB0eWxrbyDFm3JlZG5pxIUsIGkgZG9zdGFyY3phIHcgendpxIV6a3UgeiB0eW0gcGXFgm5pZWpzemVnbyBvYnJhenUNCiAgICB6d2nEhXprw7N3IG1pxJlkenkgem1pZW5ueW1pIQ0KDQojIyBXcHJvd2FkemVuaWUNCg0KUmVncmVzamEga3dhbnR5bG93YSAoYW5nLiBxdWFudGlsZSByZWdyZXNzaW9uKSB6b3N0YcWCYSB6YXByb3Bvbm93YW5hDQpwcnpleiBLb2Vua2VyYSBpIEJhc3NldHRhICgxOTc4KS4gU3pjemVnw7NsbnkgcHJ6eXBhZGVrIHJlZ3Jlc2ppDQprd2FudHlsb3dlaiBkbGEga3dhbnR5bGEgcnrEmWR1IDAsNSAoY3p5bGkgbWVkaWFueSkgamVzdCByw7N3bm93YcW8bnkNCmVzdHltYXRvcm93aSBMQUQgKGFuZy4gTGVhc3QgQWJzb2x1dGUgRGV2aWF0aW9uKSAtLSBtaW5pbWFsaXp1amUgc3VtxJkNCmJlend6Z2zEmWRueWNoIGLFgsSZZMOzdy5cDQpXcHJvd2FkemVuaWUgcsOzxbxueWNoIGt3YW50eWxpIHJlZ3Jlc2ppIGRhamUgcGXFgm5pZWpzenkgb3BpcyByb3prxYJhZMOzdw0Kd2FydW5rb3d5Y2ggenfFgmFzemN6YSB3IHByenlwYWRrdSByb3prxYJhZMOzdyBhc3ltZXRyeWN6bnljaCBsdWIgdWNpxJl0eWNoLg0KDQpSZWdyZXNqYSBrd2FudHlsb3dhIGplc3Qga29sZWpuxIUgd2FyaWFjasSFIG5hIHRlbWF0IG5ham1uaWVqc3p5Y2gNCmt3YWRyYXTDs3cgXGNpdGVwe3F1YW50aWxlfS4gU3RyYXTEhSBqZXN0IHdzcMOzxYJjenlubmlrICRsXzEkIGZ1bmtjamk6DQoNCiQkDQogICAgXHBoaSh1KSA9IFx0YXVcbWF4KHUsMCkgLSAoMS1cdGF1KVxtYXgoLXUsMCkgPSBcZnJhY3sxfXsyfXx1fCArIFxsZWZ0KFx0YXUgLSBcZnJhY3sxfXsyfVxyaWdodCl1LA0KJCQNCg0KZ2R6aWUgJFx0YXUgXGluICgwLDEpJCBvem5hY3phIGtvbmtyZXRueSBrd2FudHlsLiBQcm9ibGVtZW0gamFrDQpwb3ByemVkbmlvIGplc3QgbWluaW1hbGl6YWNqYSBjYcWCa293aXRlaiBzdHJhdHkgcmVzenRvd2VqLiBNb2RlbCB0ZW4NCmplc3QgcG93c3plY2huaWUgc3Rvc293YW55IHcgZWtvbG9naWksIG9jaHJvbmllIHpkcm93aWEgaSBpbm55Y2gNCmR6aWVkemluYWNoLCBnZHppZSBzYW1hIMWbcmVkbmlhIG5pZSB3eXN0YXJjemEgZG8gdWNod3ljZW5pYSB6xYJvxbxvbnljaA0KemFsZcW8bm/Fm2NpIG1pxJlkenkgem1pZW5ueW1pLg0KDQojIyBXeW1hZ2FuaWENCg0KV3ltYWdhbmEgamVzdCBqZWRuYSBsaWN6Ym93YSB6bWllbm5hIHphbGXFvG5hLiBabWllbm5hIHByemV3aWR5d2FuYSBtdXNpDQpiecSHIHptaWVubsSFIGlsb8WbY2lvd8SFLiBQcmVkeWt0b3J5IG1vZ8SFIGJ5xIcgem1pZW5ueW1pIGlsb8WbY2lvd3ltaSBsdWINCnN6dHVjem55bWkgem1pZW5ueW1pIHcgcHJ6eXBhZGt1IHByZWR5a3RvcsOzdyBqYWtvxZtjaW93eWNoLiBBYnkgbW/FvG5hDQpiecWCbyB1cnVjaG9tacSHIGFuYWxpesSZLCB3eW1hZ2FueSBqZXN0IHd5cmF6IHdvbG55IGx1YiBjbyBuYWptbmllaiBqZWRlbg0KcHJlZHlrdG9yLg0KDQpSZWdyZXNqYSBrd2FudHlsb3dhIG5pZSBjenluaSB6YcWCb8W8ZcWEIGRvdHljesSFY3ljaCByb3prxYJhZHUgem1pZW5uZWoNCnByemV3aWR5d2FuZWogaSBqZXN0IG9kcG9ybmEgbmEgd3DFgnl3IG9ic2Vyd2Fjamkgb2RzdGFqxIVjeWNoLg0KDQpBbmFsaXphIGt3YW50eWxvd2EgamVzdCBwb2tyZXduYSByZWdyZXNqaSBtZXRvZMSFIG5ham1uaWVqc3p5Y2gNCmt3YWRyYXTDs3cuDQoNCiMjIFByenlrxYJhZCAxLg0KDQpXeWtvcnp5c3RhbXkgcHJ6eWvFgmFkIHogcGFraWV0dSBxdWFudHJlZy4NCg0KSmFraSBqZXN0IHp3acSFemVrIG1pxJlkenkgY2HFgmtvd2l0eW0gZG9jaG9kZW0gZ29zcG9kYXJzdHdhIGRvbW93ZWdvIGENCm9kc2V0a2llbSBkb2Nob2TDs3cgd3lkYXRrb3dhbnljaCBuYSDFvHl3bm/Fm8SHPyBQcmF3byBFbmdlbGEgdyBla29ub21paQ0KZ8WCb3NpLCDFvGUgdyBtaWFyxJkgd3pyb3N0dSBkb2Nob2TDs3csIGN6xJnFm8SHIGRvY2hvZMOzdyB3eWRhdGtvd2FueWNoIG5hDQrFvHl3bm/Fm8SHIHNwYWRhLCBuYXdldCBqZcWbbGkgd3lkYXRraSBuYSDFvHl3bm/Fm8SHIGJlend6Z2zEmWRuaWUgcm9zbsSFLg0KU3Rvc3VqxIVjIHJlZ3Jlc2rEmSBrd2FudHlsb3fEhSBkbyB0eWNoIGRhbnljaCwgbW/FvG5hIG9rcmXFm2xpxIcsIGpha2llDQp3eWRhdGtpIG5hIMW8eXdub8WbxIcgcG9ub3NpIDkwJSByb2R6aW4gKGRsYSAxMDAgcm9kemluIHogZGFueW0gZG9jaG9kZW0pLA0KZ2R5IG5pZSBpbnRlcmVzdWrEhSBuYXMgxZtyZWRuaWUgd3lkYXRraSBuYSDFvHl3bm/Fm8SHLg0KDQpEYW5lLCBrdMOzcmUgd3lrb3J6eXN0YW15IC0gdG8gemJpw7NyICJlbmdlbCIgLSBkYW5lIGRvdHljesSFY2Ugd3lkYXRrw7N3IG5hDQrFvHl3bm/Fm8SHLiBKZXN0IHRvIHpiacOzciBkYW55Y2ggcmVncmVzeWpueWNoIHNrxYJhZGFqxIVjeSBzacSZIHogMjM1DQpvYnNlcndhY2ppIGRvdHljesSFY3ljaCBkb2Nob2TDs3cgaSB3eWRhdGvDs3cgbmEgxbx5d25vxZvEhyBkbGEgYmVsZ2lqc2tpY2gNCmdvc3BvZGFyc3R3IGRvbW93eWNoIGtsYXN5IHJvYm90bmljemVqLg0KDQpgYGB7ciBlY2hvPUZBTFNFfQ0KZGF0YShlbmdlbCkgI2RhbmUgDQpwIDwtIGdncGxvdChkYXRhID0gZW5nZWwpICsNCiAgICBnZW9tX3BvaW50KG1hcHBpbmcgPSBhZXMoeCA9IGluY29tZSwgeSA9IGZvb2RleHApLCBjb2xvciA9ICJibHVlIikNCnRhdXMgPC0gYygwLjEsIDAuMjUsIDAuNSwgMC43NSwgMC45MCwgMC45NSkNCmZpdHMgPC0gZGF0YS5mcmFtZSgNCiAgICBjb2VmKGxtKGZvb2RleHAgfiBpbmNvbWUsIGRhdGEgPSBlbmdlbCkpLA0KICAgIHNhcHBseSh0YXVzLCBmdW5jdGlvbih4KSBjb2VmKHJxKGZvcm11bGEgPSBmb29kZXhwIH4gaW5jb21lLCBkYXRhID0gZW5nZWwsIHRhdSA9IHgpKSkpDQpuYW1lcyhmaXRzKSA8LSBjKCJPTFMiLCBzcHJpbnRmKCIkXFx0YXVfeyUwLjJmfSQiLCB0YXVzKSkNCm5mIDwtIG5jb2woZml0cykNCmNvbG9ycyA8LSBjb2xvclJhbXBQYWxldHRlKGNvbG9ycyA9IGMoImJsYWNrIiwgInJlZCIpKShuZikNCnAgPC0gcCArIGdlb21fYWJsaW5lKGludGVyY2VwdCA9IGZpdHNbMSwgMV0sIHNsb3BlID0gZml0c1syLCAxXSwgY29sb3IgPSBjb2xvcnNbMV0sIGxpbmV3aWR0aCA9IDEuNSkNCmZvciAoaSBpbiBzZXFfbGVuKG5mKVstMV0pIHsNCiAgICBwIDwtIHAgKyBnZW9tX2FibGluZShpbnRlcmNlcHQgPSBmaXRzWzEsIGldLCBzbG9wZSA9IGZpdHNbMiwgaV0sIGNvbG9yID0gY29sb3JzW2ldKQ0KfQ0KcA0KYGBgDQoNClBvd3nFvHN6eSB3eWtyZXMgcHJ6ZWRzdGF3aWEgZG9wYXNvd2FuaWUgcmVncmVzamkga3dhbnR5bG93ZWogZGxhDQokXHRhdSA9ICgwLjEsIDAuMjUsIDAuNSwgMC43NSwgMC45MCwgMC45NSkkLiBEb3Bhc293YW5pZSBLTU5LIHRvIGdydWJhDQpjemFybmEgbGluaWEuDQoNClBvbmnFvGVqIHpuYWpkdWplIHNpxJkgdGFiZWxhIHogb3N6YWNvd2FueW1pIHdzcMOzxYJjenlubmlrYW1pLg0KDQpgYGB7cn0NCmtuaXRyOjprYWJsZShmaXRzLCBmb3JtYXQgPSAiaHRtbCIsIGNhcHRpb24gPSAiT3N6YWNvd2FuaWEgeiBLTU5LIG9yYXogYHF1YW50cmVnYCIpICU+JQ0KICAgIGthYmxlX3N0eWxpbmcoInN0cmlwZWQiKSAlPiUNCiAgICBjb2x1bW5fc3BlYygxOjgsIGJhY2tncm91bmQgPSAiI2VjZWNlYyIpDQpgYGANCg0KT2ssIG1vxbxlbXkgdG8genJvYmnEhyBiYXJkemllaiBwcnplanJ6ecWbY2llIGkgc2Zvcm1hdG93YcSHIHcgxYJhZG5laiB0YWJlbGkNCnd5bmlrw7N3Og0KDQpgYGB7ciBlY2hvPUZBTFNFLCBtZXNzYWdlPUZBTFNFLCB3YXJuaW5nPUZBTFNFLCBwYWdlZC5wcmludD1UUlVFfQ0KcTI1IDwtIHJxKGZvb2RleHAgfiBpbmNvbWUsIGRhdGEgPSBlbmdlbCwgdGF1ID0gMC4yNSkNCnE1MCA8LSBycShmb29kZXhwIH4gaW5jb21lLCBkYXRhID0gZW5nZWwsIHRhdSA9IDAuNTApDQpxNzUgPC0gcnEoZm9vZGV4cCB+IGluY29tZSwgZGF0YSA9IGVuZ2VsLCB0YXUgPSAwLjc1KQ0KDQojIFRhYmVsYSB6IHBvcsOzd25hbmllbSB3eW5pa8OzdyB0cnplY2ggbW9kZWxpOiANCg0Kc3RhcmdhemVyKHEyNSwgcTUwLCBxNzUsIHRpdGxlID0gIld5bmlraSByZWdyZXNqaSBrd2FudHlsb3d5Y2giLCB0eXBlID0gInRleHQiKQ0KYGBgDQoNCkZpbmFsbmllLCB6YXByZXplbnR1am15IHd5xYLEhWN6bmllIHRlIDMgbW9kZWxlIG5hIHd5a3Jlc2llOg0KDQpgYGB7ciBlY2hvPUZBTFNFfQ0KbXlfcXIgPC0gcnEoZm9vZGV4cCB+IGluY29tZSwgZGF0YSA9IGVuZ2VsLCB0YXUgPSBzZXEoMC4yNSwgMC43NSwgMC4yNSkpDQoNCmludGVyY2VwdF9zbG9wZSA8LSBteV9xciAlPiUgDQogIGNvZWYoKSAlPiUgDQogIHQoKSAlPiUgDQogIGRhdGEuZnJhbWUoKSAlPiUgDQogIHJlbmFtZShpbnRlcmNlcHQgPSBYLkludGVyY2VwdC4sIHNsb3BlID0gaW5jb21lKSAlPiUgDQogIG11dGF0ZShxdWFudGlsZSA9IHJvdy5uYW1lcyguKSkNCg0KZ2dwbG90KCkgKyANCiAgZ2VvbV9wb2ludChkYXRhID0gZW5nZWwsIGFlcyhpbmNvbWUsIGZvb2RleHApLCBhbHBoYSA9IDAuNSkgKyANCiAgZ2VvbV9hYmxpbmUoZGF0YSA9IGludGVyY2VwdF9zbG9wZSwgYWVzKGludGVyY2VwdCA9IGludGVyY2VwdCwgc2xvcGUgPSBzbG9wZSwgY29sb3IgPSBxdWFudGlsZSkpICsgDQogIHRoZW1lX21pbmltYWwoKSArIA0KICBsYWJzKHggPSAiRG9jaMOzZCIsIHkgPSAiV3lkYXRraSBuYSDFvHl3bm/Fm8SHIiwgdGl0bGUgPSAiUmVncmVzamUga3dhbnR5bG93ZSB6IHRhdSA9IDAuMjUsIDAuNTAgb3JheiAwLjc1IiwgDQogICAgICAgY2FwdGlvbiA9ICLFuXLDs2TFgm8gZGFueWNoOiBLb2Vua2VyIGFuZCBCYXNzZXR0ICgxOTgyKSIpDQpgYGANCg0KIyMgUHJ6eWvFgmFkIDIuDQoNClR1dGFqIHByemVwcm93YWR6aW15IHRlc3R5IHXFvHljaWEgcGFraWV0dSBxdWFudHJlZywgd3lrb3J6eXN0dWrEhWMNCndidWRvd2FueSB6YmnDs3IgZGFueWNoICIqKm10Y2FycyoqIi4gWm1pZW5uYSAiKiptcGcqKiIgb3puYWN6YSBzcGFsYW5pZQ0Kc2Ftb2Nob2TDs3cgKCptaWxlL2dhbG9uKikuDQoNClphbW9kdWxlam15IHphbGXFvG5vxZvEhyByZWdyZXN5am7EhSBkbGEgdGVqIHptaWVubmVqIG9kIGtpbGt1IHByZWR5a3RvcsOzdy4NCg0KTmFqcGllcncgb3N6YWN1am15IHJlZ3Jlc2rEmSBLTU5LOg0KDQpgYGB7cn0NCmttbmsgPC0gbG0obXBnIH4gZGlzcCArIGhwICsgZmFjdG9yKGFtKSArIGZhY3Rvcih2cyksIGRhdGEgPSBtdGNhcnMpDQpzdW1tYXJ5KGttbmspDQpgYGANCg0KVGVyYXogb3N6YWN1am15IHdhcnVua293ZSByZWdyZXNqZSBrd2FudHlsb3dlIG5hIHLDs8W8bnljaCBrd2FudHlsYWNoLA0KYsWCxIVkIHN0YW5kYXJkb3d5IHV6eXNrYW55IHByemV6ICoqKmJvb3RzdHJhcCoqKi4NCg0KWmF1d2HFvCwgxbxlIGlzdG5pZWplIGdyYWRpZW50IHdlIHdzcMOzxYJjenlubmlrYWNoIGt3YW50eWxvd3ljaCAqKmhwKiosIGphaw0KcsOzd25pZcW8ICoqZGlzcCoqLiBabmFrICoqZGlzcCoqIG9kd3JhY2Egc2nEmSwgcsOzd25pZcW8IHdzcMOzxYJjenlubmlrIG5hDQpjenlubmlrdSAqKmFtKiogamVzdCByw7PFvG55IHcgemFsZcW8bm/Fm2NpIG9kIGt3YW50eWxpOg0KDQpgYGB7cn0NCmt3YW50eWxlIDwtIGMoMC4yNSwgMC41MCwgMC43NSkNCnJlZ19rd2FudHlsb3dhIDwtIHJxKG1wZyB+IGRpc3AgKyBocCArIGZhY3RvcihhbSksdGF1ID0ga3dhbnR5bGUsZGF0YSA9IG10Y2FycykNCnN1bW1hcnkocmVnX2t3YW50eWxvd2EsIHNlID0gImJvb3QiKQ0KYGBgDQoNCiMjIyBUZXN0eSB3c3DDs8WCY3p5bm5pa8Ozdw0KDQpVxbx5amVteSBmdW5rY2ppIHJxLmFub3ZhIHogcGFraWV0dSByZWdyZXNqaSBrd2FudHlsb3dlaiwgYWJ5DQpwcnplcHJvd2FkemnEhyB0ZXN0IFdBTERBLiBQYW1pxJl0YWosIMW8ZSB0ZXN0IFdBTERBIG3Ds3dpLCDFvGUgYmlvcsSFYyBwb2QNCnV3YWfEmSBuaWVvZ3Jhbmljem9uZSBvc3phY293YW5pYSBtb2RlbHUsIHByemV0ZXN0dWplbXkgaGlwb3RlesSZIHplcm93xIUNCm3Ds3dpxIVjxIUsIMW8ZSB3c3DDs8WCY3p5bm5pa2kgc3BlxYJuaWFqxIUgcGV3bmUgbGluaW93ZSBvZ3JhbmljemVuaWEuDQoNCkFieSBqxIUgcHJ6ZXRlc3Rvd2HEhywgdcW8eWplbXkgb2JpZWt0dSB6d3LDs2NvbmVnbyB6IHVydWNob21pZW5pYSAqKipycSoqKg0KeiByw7PFvG55bWkgbGljemJhbWkga3dhbnR5bGkgaSB1c3Rhd2lteSBvcGNqxJkgKioqam9pbnQqKiogbmEgdHJ1ZSBsdWINCmZhbHNlLiBHZHkgKioqam9pbnQqKiogamVzdCB0cnVlOiAicsOzd25vxZvEhyB3c3DDs8WCY3p5bm5pa8OzdyBraWVydW5rb3d5Y2gNCnBvd2lubmEgYnnEhyB3eWtvbmFuYSBqYWtvIHdzcMOzbG5lIHRlc3R5IG5hIHdzenlzdGtpY2ggcGFyYW1ldHJhY2gNCm5hY2h5bGVuaWEiLCBnZHkgKioqam9pbnQqKiogamVzdCBmYWxzZTogIm5hbGXFvHkgemfFgmFzemHEhyBvZGR6aWVsbmUNCnRlc3R5IG5hIGthxbxkeW0geiBwYXJhbWV0csOzdyBuYWNoeWxlbmlhIi4NCg0KWmF1d2HFvCwgxbxlIHRlc3R5IGt3YW50eWxvd2Ugc8SFIHRlc3RhbWkgImxpbmlpIHLDs3dub2xlZ8WCZWoiLiBPem5hY3phIHRvLA0KxbxlIHBvd2lubmnFm215IHd5asSFxIcgcsOzxbxuZSB4LXd5cmF6eV93b2xuZSBkbGEga2HFvGRlZ28ga3dhbnR5bGEsIHBvbmlld2HFvA0KcmVwcmV6ZW50dWrEhSBvbmUgcG96aW9teSByb3prxYJhZMOzdyB3YXJ1bmtvd3ljaC4gSmXFm2xpIGplZG5haw0Kd3Nww7PFgmN6eW5uaWtpIGt3YW50eWxpIGRsYSB3c3DDs8WCY3p5bm5pa293IHPEhSB0YWtpZSBzYW1lLCB0byBuaWUgbWENCmVmZWt0w7N3IHNwZWN5Zmljem55Y2ggZGxhIGt3YW50eWxpLCB3eXN0YXJjesSFIGVmZWt0eSDFm3JlZG5pZS4NCg0KKipCYWRhbmllIHN0YXR5c3R5Y3puZWogcsOzxbxuaWN5IG1pxJlkenkgMjUuIGkgNTAuIGt3YW50eWxlbSB3YXJ1bmtvd3ltOioqDQoNCkJpb3LEhWMgcG9kIHV3YWfEmSBwb3d5xbxzemUgb3N6YWNvd2FuaWEga3dhbnR5bGksIHLDs8W8bmljYSBtacSZZHp5DQprd2FudHlsYW1pIDAsMjUgaSAwLDUwIGlzdG5pZWplLCBhbGUgY3p5IHPEhSBvbmUgd3lzdGFyY3phasSFY28gZHXFvGUsIGFieQ0KYnnEhyBzdGF0eXN0eWN6bmllIHLDs8W8bmU/IEpha2EgamVzdCB3YXJ0b8WbxIcgcD8gUHJ6ZWdsxIVkYWrEhWMgcG9uacW8c3plDQp3eW5pa2ksIG5pZSBzxIUgb25lIHN0YXR5c3R5Y3puaWUgcsOzxbxuZSENCg0KUG8gcGllcndzemUsIGpvaW50ID0gVFJVRS4gVG8gbmllIGplc3QgdGVzdG93YW5pZSwgY3p5IHdzcMOzxYJjenlubmlrIG5hDQpkaXNwIGplc3QgdGFraSBzYW0gamFrIHdzcMOzxYJjenlubmlrIG5hIGhwLiBUbyBqZXN0IHdzcMOzbG5lIHRlc3Rvd2FuaWUsDQpjenkgd3Nww7PFgmN6eW5uaWtpIGRsYSByw7PFvG55Y2gga3dhbnR5bGkgZGlzcCBpIHLDs8W8bnljaCBrd2FudHlsaSBocCBzxIUNCnRha2llIHNhbWUgZGxhIGthxbxkZWogem1pZW5uZWouDQoNCmBgYHtyfQ0Ka3dhbnR5bGUgPC0gYygwLjI1LCAwLjUwKQ0KcmVnX2t3YW50eWxvd2EgPC0gcnEobXBnIH4gZGlzcCArIGhwICsgZmFjdG9yKGFtKSx0YXUgPSBrd2FudHlsZSwgZGF0YSA9IG10Y2FycykNCmFub3ZhKHJlZ19rd2FudHlsb3dhLCB0ZXN0ID0gIldhbGQiLCBqb2ludD1UUlVFKQ0KYGBgDQoNClBvIGRydWdpZSwgam9pbnQgPSBGYWxzZToNCg0KYGBge3J9DQphbm92YShyZWdfa3dhbnR5bG93YSwgdGVzdCA9ICJXYWxkIiwgam9pbnQ9RkFMU0UpDQpgYGANCg0KKipCYWRhbmllIHN0YXR5c3R5Y3puZWogcsOzxbxuaWN5IG1pxJlkenkgMjUsIDUwIGkgNzUga3dhbnR5bGVtDQp3YXJ1bmtvd3ltOioqDQoNClBpZXJ3c3p5IGt3YXJ0eWwgaSBtZWRpYW5hIG5pZSB3eWRhasSFIHNpxJkgYnnEhyBzdGF0eXN0eWN6bmllIHLDs8W8bmUsIHRlcmF6DQpkb8WCxIVjenlteSB0cnplY2kga3dhcnR5bC4gSmFrIHdpZGHEhyB3Y3plxZtuaWVqLCBrd2FydHlsZSB3c3DDs2xuaWUNCnd5a2F6dWrEhSBncmFkaWVudC4gVGVyYXogbW/FvGVteSB6b2JhY3p5xIcsIMW8ZSAqKmRpc3AqKiwgKipocCoqIGkgKiphbSoqDQpzxIUgb2RkemllbG5pZSBzdGF0eXN0eWN6bmllIHLDs8W8bmUuDQoNClBvIHBpZXJ3c3plLCBqb2ludCA9IFRSVUU6DQoNCmBgYHtyIG1lc3NhZ2U9RkFMU0UsIHdhcm5pbmc9RkFMU0V9DQprd2FudHlsZSA8LSBjKDAuMjUsIDAuNTAsIDAuNzUpDQoNCnJlZ19rd2FudHlsb3dhIDwtIHJxKG1wZyB+IGRpc3AgKyBocCArIGZhY3RvcihhbSksdGF1ID0ga3dhbnR5bGUsIGRhdGEgPSBtdGNhcnMpDQoNCmFub3ZhKHJlZ19rd2FudHlsb3dhLCB0ZXN0ID0gIldhbGQiLCBqb2ludD1UUlVFKQ0KYGBgDQoNClBvIGRydWdpZSwgam9pbnQgPSBGYWxzZToNCg0KYGBge3IgbWVzc2FnZT1GQUxTRSwgd2FybmluZz1GQUxTRX0NCmFub3ZhKHJlZ19rd2FudHlsb3dhLCB0ZXN0ID0gIldhbGQiLCBqb2ludD1GQUxTRSkNCmBgYA0KDQojIyMgRG9icm/EhyBkb3Bhc293YW5pYQ0KDQpNb8W8ZW15IG9ibGljennEhyB3c3DDs8WCY3p5bm5pa2kgZG9icm9jaSBkb3Bhc293YW5pYSByZWdyZXNqaSBrd2FudHlsb3dlaiB6DQp3eWtvcnp5c3RhbmllbSByZXN6dCBpIHJlc3p0IGJlendhcnVua293eWNoOg0KDQpgYGAgcg0KZ29vZGZpdChyZXNpZCwgcmVzaWRfbmwsIHRhdSkNCmBgYA0KDQpNaWFyYSBkb2Jyb2NpIGRvcGFzb3dhbmlhIGRsYSByZWdyZXNqaSBrd2FudHlsb3dlaiBqZXN0IHN6YWNvd2FuYSBqYWtvIDENCm1pbnVzIHN0b3N1bmVrIHN1bXkgb2RjaHlsZcWEIGJlend6Z2zEmWRueWNoIHcgbW9kZWxhY2ggdyBwZcWCbmkNCnNwYXJhbWV0cnl6b3dhbnljaCBkbyBzdW15IG9kY2h5bGXFhCBiZXp3emdsxJlkbnljaCB3IHplcm93eW0NCihiZXp3YXJ1bmtvd3ltKSBtb2RlbHUga3dhbnR5bG93eW0uDQoNCldhcnRvxZtjaSB0ZSBzxIUgcHJ6eWRhdG5lIGRvIHBvcsOzd25hxYQgbWnEmWR6eSBtb2RlbGFtaSBrd2FudHlsb3d5bWksIGFsZQ0KbmllIHPEhSBwb3LDs3dueXdhbG5lIHplIHN0YW5kYXJkb3d5bWkgd3Nww7PFgmN6eW5uaWthbWkgZGV0ZXJtaW5hY2ppLiBUZQ0Kb3N0YXRuaWUgb3BhcnRlIHPEhSBuYSB3YXJpYW5jamkgb2RjaHlsZcWEIGt3YWRyYXRvd3ljaCwgbmF0b21pYXN0DQp3YXJ0b8WbY2kgZG9icm9jaSBkb3Bhc293YW5pYSBkbGEgcmVncmVzamkga3dhbnR5bG93ZWogb3BhcnRlIHPEhSBuYQ0Kb2RjaHlsZW5pYWNoIGJlend6Z2zEmWRueWNoLiBXYXJ0b8WbY2kgZG9icm9jaSBkb3Bhc293YW5pYSB6YXdzemUgYsSZZMSFDQptbmllanN6ZSBuacW8IHdhcnRvxZtjaSBSXjJeLg0KDQpgYGB7ciBtZXNzYWdlPUZBTFNFLCB3YXJuaW5nPUZBTFNFfQ0KIyMgbW9kZWwga3dhbnR5bG93eQ0KbW9kZWwxIDwtIHJxKG1wZyB+IGRpc3AgKyBocCArIGZhY3RvcihhbSksdGF1ID0gMC41LCBkYXRhID0gbXRjYXJzKQ0KcmVzenR5MSA8LSByZXNpZChtb2RlbDEpDQoNCiMjIGJlendhcnVua293eSAocHVzdHkpIG1vZGVsIGt3YW50eWxvd3kNCm1vZGVsMiA8LSBycShtcGcgfiAxLCB0YXUgPSAwLjUsZGF0YT1tdGNhcnMpDQpyZXN6dHkyIDwtIHJlc2lkKG1vZGVsMikNCg0KZ29vZGZpdChyZXN6dHkxLCByZXN6dHkyLCAwLjUpDQoNCiMjIHIyIG1vZGVsdSBLTU5LIGRsYSBwb3LDs3duYW5pYQ0KbW9kZWxfbG0gPC0gbG0obXBnIH4gZGlzcCArIGhwICsgZmFjdG9yKGFtKSwgZGF0YSA9IG10Y2FycykNCg0Kc3VtbWFyeShtb2RlbF9sbSkkci5zcXVhcmVkDQpgYGANCg0KIyMgWmFkYW5pZQ0KDQpUZXJheiBXYXN6YSBrb2xlaiA7LSkNCg0KV2FzenltIHphZGFuaWVtIGR6aXNpYWogamVzdCB6YW1vZGVsb3dhbmllIC0gcG9yw7N3bmFuaWUgS01OSyBvcmF6DQpyZWdyZXNqaSBrd2FudHlsb3dlaiAocsOzxbxuby1wb3ppb21vd2VqKSBkbGEgem1pZW5uZWogImVhcm5pbmdzIiAtDQp3eW5hZ3JvZHplbmlhLg0KDQpEb2JpZXJ6IGkgcHJ6ZXRlc3R1aiBwcmVkeWt0b3J5LCBrd2FudHlsZSBkbGEgbW9kZWxpLiBXeWtvbmFqIHRlc3R5DQpyw7PFvG5pYyB3c3DDs8WCY3p5bm5pa293IGRsYSBmaW5hbG55Y2ggbW9kZWxpLg0KDQpXIHByenlwYWRrdSBwcm9ibGVtw7N3IC0gb2JlanJ6eWogdmlkZW8gdHV0b3JpYWwgKHfFgsSFY3ogcG9sc2tpZSBuYXBpc3kpDQpvcmF6IHdlamTFuiBuYSBqZWdvIHN0cm9uxJkgemUgxbpyw7NkxYJhbWkuIE1vxbxlc3ogcsOzd25pZcW8IHd5a29yenlzdGHEhyB3L3cNCnByenlrxYJhZHkuDQoNCg0KYGBge3J9DQpkYXRhKCJDUFNTVzkyOTgiKQ0KZGFuZSA8LSBDUFNTVzkyOTgNCmRhbmU5MiA8LSBkYW5lICU+JSBmaWx0ZXIoeWVhcj09MTk5MikNCmRhbmU5OCA8LSBkYW5lICU+JSBmaWx0ZXIoeWVhcj09MTk5OCkNCmBgYA0KDQojIyBBbmFsaXphIHptaWVubmVqICJlYXJuaW5ncyIgdyAxOTkyIHJva3UgDQoNCmBgYHtyfQ0KcGxvdCA8LSBnZ3Bsb3QoZGF0YSA9IGRhbmU5MikgKw0KICAgIGdlb21fcG9pbnQobWFwcGluZyA9IGFlcyh4ID0gYWdlLCB5ID0gZWFybmluZ3MpLCBjb2xvciA9ICJwdXJwbGUiKSsgbGFicyh0aXRsZT0gIld5a3JlcyByb3pyenV0dSBlYXJuaW5ncyB3emdsxJlkZW0gYWdlIikNCnBsb3QNCmBgYA0KDQpCdWRvd2EgaSBwb3LDs3duYW5pZSBtb2RlbGkgemEgcG9tb2PEhSBLTU5LIG9yYXogcmVncmVzamkga3dhbnR5bG93ZWouDQoNCmBgYHtyfQ0KbW9kZWxfbGluaW93eSA8LSBsbShlYXJuaW5nc35kZWdyZWUrZ2VuZGVyK2FnZSwgZGF0YT1kYW5lOTIpDQptb2RlbF9xcjEgPC0gcnEoZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTkyLCB0YXU9MC4yNSkNCm1vZGVsX3FyMiA8LSBycShlYXJuaW5nc35kZWdyZWUrZ2VuZGVyK2FnZSwgZGF0YT1kYW5lOTIsIHRhdT0wLjUpDQptb2RlbF9xcjMgPC0gcnEoZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTkyLCB0YXU9MC43NSkNCnN1bW1hcnkobW9kZWxfcXIxKQ0Kc3VtbWFyeShtb2RlbF9xcjMpDQpBSUMobW9kZWxfbGluaW93eSwgbW9kZWxfcXIxLCBtb2RlbF9xcjIsIG1vZGVsX3FyMykNCmBgYA0KDQpBSUMgamVzdCBuYWpuacW8c3plIGRsYSBtb2RlbHUgIm1vZGVsX3FyMSIsIGNvIG96bmFjemEgxbxlIG1vZGVsIHJlZ3Jlc2ppIGt3YW50eWxvd2VqIHogdGF1PTAuMjUgdyBuYWpsZXBzenkgc3Bvc8OzYiB3eWphxZtuaWEgem1pZW5ub8WbxIcgdHljaCBkYW55Y2ggc3BvxZtyw7NkIHR5Y2ggbW9kZWxpLiBNb2RlbCBsaW5pb3d5IHN0d29yem9ueSBtZXRvZMSFIEtNTksgZ29yemVqIG9iamHFm25pYSB6bWllbm7EhSBuacW8IG1vZGVsIGt3YW50eWxvd3kgZGxhIHBpZXJ3c3plZ28ga3dhbnR5bGEgaSBkbGEgbWVkaWFueS4NCg0KVyBncnVwaWUgb3PDs2IsIGt0w7NyeWNoIHphcm9ia2kgem5hamR1asSFIHNpxJkgdyBwaWVyd3N6eW0ga3dhbnR5bHUgKDI1JSBuYWptbmllaiB6YXJhYmlhasSFY3ljaCkgcG9zaWFkYW5pZSB3ecW8c3plZ28gd3lrc3p0YcWCY2VuaWEgcHJ6eWvFgmFkYSBzacSZIG5hIHphcm9ia2kgd3nFvHN6ZSBzcmVkbmlvIG8gMyw0MiBqZWRub3N0a2kgcGllbmnEmcW8bmVqIHdpxJljZWogd3pnbMSZZGVtIG9zw7NiIHogd3lrc3p0YcWCY2VuaWVtIMWbcmVkbmltLiBXc3LDs2QgMjUlIG5hamxlcGllaiB6YXJhYmlhasSFY3ljaCAodGF1PTAsNzUpIHLDs8W8bmljYSB3IHphcm9ia2FjaCBqZXN0IHdpxJlrc3phIGkgd3lub3NpIDUsMTcgamVkbm9zdGVrIHBpZW5pxJnFvG55Y2ggbmEga29yennFm8SHIG9zw7NiIHd5xbxlaiB3eWtzenRhxYJjb255Y2guIEtvYmlldHkgdyBwaWVyd3N6ZWogZ3J1cGllIHphcmFiaWFqxIUgxZtyZWRuaW8gbyAxLDE0IGplZG5vc3RlayBtbmllaiBvZCBtxJnFvGN6eXpuLCB6YcWbIHcgZ3J1cGllIG8gbmFqd3nFvHN6eWNoIHphcm9ia2FjaCB0YSByw7PFvG5pY2EgcsOzd25pZcW8IGplc3Qgd2nEmWtzemEsIDIsNTIgamVkbm90a2kgbmEgbmlla29yennFm8SHIGtvYmlldC4gV3JheiB6IHdpZWtpZW0gdyBvYnUgZ3J1cGFjaCB6YXJvYmtpIHJvc27EhSBwcnp5IGN6eW0gcsOzd25pZcW8IHRhIHLDs8W8bmljYSBqZXN0IHd5xbxzemEgdyBncnVwaWUgbmFqYm9nYXRzenljaC4gICANCg0KYGBge3J9DQptb2RlbF9ycTUgPC0gcnEoZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTkyLCB0YXUgPSBzZXEoMC4yNSwgMC43NSwgMC4yNSkpDQoNCmludGVyY2VwdF9zbG9wZSA8LSBtb2RlbF9ycTUgJT4lIA0KICBjb2VmKCkgJT4lIA0KICB0KCkgJT4lIA0KICBkYXRhLmZyYW1lKCkgJT4lIA0KICByZW5hbWUoaW50ZXJjZXB0ID0gWC5JbnRlcmNlcHQuLCBzbG9wZSA9IGFnZSkgJT4lIA0KICBtdXRhdGUocXVhbnRpbGUgPSByb3cubmFtZXMoLikpDQoNCmdncGxvdCgpICsgDQogIGdlb21fcG9pbnQoZGF0YSA9IGRhbmU5MiwgYWVzKGFnZSwgZWFybmluZ3MpLCBhbHBoYSA9IDAuNSkgKyANCiAgZ2VvbV9hYmxpbmUoZGF0YSA9IGludGVyY2VwdF9zbG9wZSwgYWVzKGludGVyY2VwdCA9IGludGVyY2VwdCwgc2xvcGUgPSBzbG9wZSwgY29sb3IgPSBxdWFudGlsZSkpICsgDQogIHRoZW1lX21pbmltYWwoKSArIA0KICBsYWJzKHggPSAiV2llayIsIHkgPSAiWmFyb2JraSIsIHRpdGxlID0gIlJlZ3Jlc2plIGt3YW50eWxvd2UgeiB0YXUgPSAwLjI1LCAwLjUwIG9yYXogMC43NSIpDQpgYGANCg0KDQpgYGB7cn0NCmhpc3QoZGFuZTkyJGVhcm5pbmdzKQ0KaGlzdChsb2coZGFuZTkyJGVhcm5pbmdzKSkNCmBgYA0KDQpOYSBwb2RzdGF3aWUgaGlzdG9ncmFtdSBwcnplZHN0YXdpYWrEhWNlZ28gcm96a8WCYWQgem1pZW5uZWogbW/FvG5hIHphdXdhxbx5xIcsIMW8ZSB6bG9nYXJ5dG1vd2FuaWUgem1pZW5uZWogbW/FvGUgcG9tw7NjIHcgdXp5c2thbml1IGxlcHN6ZWdvIG1vZGVsdS4NCg0KYGBge3J9DQpkYW5lOTIkbG9nX2Vhcm5pbmdzIDwtIGxvZyhkYW5lOTIkZWFybmluZ3MpDQpgYGANCg0KYGBge3J9DQpwbG90MiA8LSBnZ3Bsb3QoZGF0YSA9IGRhbmU5MikgKw0KICAgIGdlb21fcG9pbnQobWFwcGluZyA9IGFlcyh4ID0gYWdlLCB5ID0gbG9nX2Vhcm5pbmdzKSwgY29sb3IgPSAicHVycGxlIikrIGxhYnModGl0bGU9ICJXeWtyZXMgcm96cnp1dHUgbG9nKGVhcm5pbmdzKSB3emdsxJlkZW0gYWdlIikNCnBsb3QyDQpgYGANCg0KQnVkb3dhIGkgcG9yw7N3bmFuaWUgbW9kZWxpIHphIHBvbW9jxIUgS01OSyBvcmF6IHJlZ3Jlc2ppIGt3YW50eWxvd2VqIGRsYSB6bWllbm5laiBlYXJuaW5ncyB6bG9nYXJ5dG1vd2FuZWouDQoNCmBgYHtyfQ0KbW9kZWxfbGluaW93eTIgPC0gbG0obG9nX2Vhcm5pbmdzfmRlZ3JlZStnZW5kZXIrYWdlLCBkYXRhPWRhbmU5MikNCm1vZGVsX3FybG9nMSA8LSBycShsb2dfZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTkyLCB0YXU9MC4yNSkNCm1vZGVsX3FybG9nMiA8LSBycShsb2dfZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTkyLCB0YXU9MC41KQ0KbW9kZWxfcXJsb2czIDwtIHJxKGxvZ19lYXJuaW5nc35kZWdyZWUrZ2VuZGVyK2FnZSwgZGF0YT1kYW5lOTIsIHRhdT0wLjc1KQ0Kc3VtbWFyeShtb2RlbF9xcmxvZzEpDQpBSUMobW9kZWxfbGluaW93eTIsIG1vZGVsX3FybG9nMSwgbW9kZWxfcXJsb2cyLCBtb2RlbF9xcmxvZzMpDQpgYGANCg0KVyBwcnp5cGFka3UgemxvZ2FyeXRtb3dhbmVqIHptaWVubmVqIG9iamHFm25pYW5laiBsZXBzesSFIG9iamHFm25pYWxub8WbxIcgbWEgbW9kZWwgbGluaW93eSAobmFqbmnFvHN6ZSBBSUMpIG5pxbwgbW9kZWxlIHJlZ3Jlc2ppIGt3YW50eWxvd2VqLiBTcG/Fm3LDs2QgbW9kZWxpIHJlZ3Jlc2ppIGt3YXJ0eWxvd2VqIG5hamxlcHN6eSB6ZGFqZSBzacSZIHRlbiBkbGEgbWVkaWFueS4NCg0KV2VkxYJ1ZyB3eWJyYW5lZ28gbW9kZWx1LCBjenlsaSBtb2RlbHUgbGluaW93ZWdvIG9zb2J5IHBvc2lhZGFqxIVjZSB3eWtzenRhxYJjZW5pZSB3ecW8c3plIHphcmFiaWFqxIUgxZtyZWRuaW8gbyAwLDQxIGplZG5vc3RraSBwaWVuacSZxbxuZWogbmEgZ29kemluxJkgd2nEmWNlaiBuacW8IG9zb2J5IHBvc2lhZGFqxIVjZSB3eWtzenRhxYJjZW5pZSDFm3JlZG5pZS4gS29iaWV0eSB6YXJhYmlhasSFIHByemVjacSZdG5pZSAwLDE0IGplZG5vc3RraSBwaWVuacSZxbxuZWogbW5pZWogbmnFvCBtxJnFvGN6ecW6bmkuIFdyYXogemUgc3RhcnplbmllbSBzacSZLCB6IGthxbxkeW0ga29sZWpueW0gcm9raWVtIHphcmFiaWEgc2nEmSDFm3JlZG5vIDAsMDIgamVkbm9zdGtpIHBpZW5pxJnFvG5lIG5hIGdvZHppbsSZIHdpxJljZWouIA0KDQojIyBBbmFsaXphIHptaWVubmVqICJlYXJuaW5ncyIgdyAxOTk4IHJva3UgDQoNCmBgYHtyfQ0KcGxvdDMgPC0gZ2dwbG90KGRhdGEgPSBkYW5lOTgpICsNCiAgICBnZW9tX3BvaW50KG1hcHBpbmcgPSBhZXMoeCA9IGFnZSwgeSA9IGVhcm5pbmdzKSwgY29sb3IgPSAicGluayIpKyBsYWJzKHRpdGxlPSAiV3lrcmVzIHJvenJ6dXR1IGVhcm5pbmdzIHd6Z2zEmWRlbSBhZ2UiKQ0KcGxvdDMNCmBgYA0KDQpCdWRvd2EgaSBwb3LDs3duYW5pZSBtb2RlbGkgemEgcG9tb2PEhSBLTU5LIG9yYXogcmVncmVzamkga3dhbnR5bG93ZWouDQoNCmBgYHtyfQ0KbW9kZWxfbGluaW93eTMgPC0gbG0oZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTk4KQ0KbW9kZWxfcXI2IDwtIHJxKGVhcm5pbmdzfmRlZ3JlZStnZW5kZXIrYWdlLCBkYXRhPWRhbmU5OCwgdGF1PTAuMjUpDQptb2RlbF9xcjcgPC0gcnEoZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTk4LCB0YXU9MC41KQ0KbW9kZWxfcXI4IDwtIHJxKGVhcm5pbmdzfmRlZ3JlZStnZW5kZXIrYWdlLCBkYXRhPWRhbmU5OCwgdGF1PTAuNzUpDQpzdW1tYXJ5KG1vZGVsX3FyNikNCnN1bW1hcnkobW9kZWxfcXI4KQ0KQUlDKG1vZGVsX2xpbmlvd3kzLCBtb2RlbF9xcjYsIG1vZGVsX3FyNywgbW9kZWxfcXI4KQ0KYGBgDQoNClcgMTk5OCByb2t1IHBvZG9ibmllIGphayB3IDE5OTIgcm9rdSBtb2RlbCByZWdyZXNqaSBrd2FudHlsb3dlaiB6IHRhdT0wLjI1IChuYWpuacW8c3plIEFJQykgdyBuYWpsZXBzenkgc3Bvc8OzYiB3eWphxZtuaWEgem1pZW5ub8WbxIcgemFyb2Jrw7N3LiBNb2RlbCBsaW5pb3d5IGdvcnplaiBvYmphxZtuaWEgem1pZW5uxIUgbmnFvCBtb2RlbCBrd2FudHlsb3d5IGRsYSBwaWVyd3N6ZWdvIGt3YW50eWxhIGkgZGxhIG1lZGlhbnkuDQoNClcgMTk5OCByb2t1IHcgZ3J1cGllIG9zw7NiLCBrdMOzcnljaCB6YXJvYmtpIHpuYWpkdWrEhSBzacSZIHcgcGllcndzenltIGt3YW50eWx1ICgyNSUgbmFqbW5pZWogemFyYWJpYWrEhWN5Y2gpIHBvc2lhZGFuaWUgd3nFvHN6ZWdvIHd5a3N6dGHFgmNlbmlhIHByenlrxYJhZGEgc2nEmSBuYSB6YXJvYmtpIHd5xbxzemUgc3JlZG5pbyBvIDMsOCBqZWRub3N0a2kgcGllbmnEmcW8bmVqIHdpxJljZWogd3pnbMSZZGVtIG9zw7NiIHogd3lrc3p0YcWCY2VuaWVtIMWbcmVkbmltLiBXc3LDs2QgMjUlIG5hamxlcGllaiB6YXJhYmlhasSFY3ljaCAodGF1PTAsNzUpIHLDs8W8bmljYSB3IHphcm9ia2FjaCBqZXN0IHdpxJlrc3phIGkgd3lub3NpIDYsMjUgamVkbm9zdGVrIHBpZW5pxJnFvG55Y2ggbmEga29yennFm8SHIG9zw7NiIHd5xbxlaiB3eWtzenRhxYJjb255Y2guIEtvYmlldHkgdyBwaWVyd3N6ZWogZ3J1cGllIHphcmFiaWFqxIUgxZtyZWRuaW8gbyAxLDY0IGplZG5vc3RlayBtbmllaiBvZCBtxJnFvGN6eXpuLCB6YcWbIHcgZ3J1cGllIG8gbmFqd3nFvHN6eWNoIHphcm9ia2FjaCB0YSByw7PFvG5pY2EgcsOzd25pZcW8IGplc3Qgd2nEmWtzemEsIDMsMzYgamVkbm90a2kgbmEgbmlla29yennFm8SHIGtvYmlldC4gV3JheiB6IHdpZWtpZW0gdyBvYnUgZ3J1cGFjaCB6YXJvYmtpIHJvc27EhSBwcnp5IGN6eW0gcsOzd25pZcW8IHRhIHLDs8W8bmljYSBqZXN0IHd5xbxzemEgdyBncnVwaWUgbmFqYm9nYXRzenljaC4NCg0KTW/FvG5hIHphdXdhxbx5xIcsIMW8ZSBuYSBwcnplc3RyemVuaSBsYXQgb2QgMTk5MiBkbyAxOTk4IHJva3UgdyBvYnUgZ3J1cGFjaCwgdGVqIHphcmFiacSFasSFY3ljaCBuYWptbmllaiwgamFrIGkgdGVqIHphcmFiaWFqxIVjeWNoIG5handpxJljZWogcsOzxbxuaWNlIHcgb3NpxIVnYW55Y2ggemFyb2JrYWNoIHplIHd6Z2zEmWR1IG5hIHBvc2lhZGFuZSB3eWtzenRhxYJjZW5pZSwgcMWCZcSHIG9yYXogd2llayBwb2fFgsSZYmnFgnkgc2nEmS4gUG9zaWFkYW5pZSB3ecW8c3plZ28gd3lrc3p0YcWCY2VuaWEgYmFyZHppZWogcG9kd3nFvHN6YSB6YXJvYmtpLCBrb2JpZXR5IHphcmFiaWFqxIUgamVzemN6ZSBtbmllaiB3emdsxJlkZW0gbcSZxbxjenl6biwgYSB6YXJvYmtpIGplc3pjemUgYmFyZHppZWogcm9zbsSFIHdyYXogeiB3aWVraWVtICh3IHByenlwYWRrdSB0ZWogem1pZW5uZWogZ8WCw7N3bmllIHcgZ3J1cGllIG5hamxlcGllaiB6YXJhYmlhasSFY3ljaCkuDQoNCmBgYHtyfQ0KbW9kZWxfcnE0IDwtIHJxKGVhcm5pbmdzfmRlZ3JlZStnZW5kZXIrYWdlLCBkYXRhPWRhbmU5OCwgdGF1ID0gc2VxKDAuMjUsIDAuNzUsIDAuMjUpKQ0KDQppbnRlcmNlcHRfc2xvcGUgPC0gbW9kZWxfcnE0ICU+JSANCiAgY29lZigpICU+JSANCiAgdCgpICU+JSANCiAgZGF0YS5mcmFtZSgpICU+JSANCiAgcmVuYW1lKGludGVyY2VwdCA9IFguSW50ZXJjZXB0Liwgc2xvcGUgPSBhZ2UpICU+JSANCiAgbXV0YXRlKHF1YW50aWxlID0gcm93Lm5hbWVzKC4pKQ0KDQpnZ3Bsb3QoKSArIA0KICBnZW9tX3BvaW50KGRhdGEgPSBkYW5lOTgsIGFlcyhhZ2UsIGVhcm5pbmdzKSwgYWxwaGEgPSAwLjUpICsgDQogIGdlb21fYWJsaW5lKGRhdGEgPSBpbnRlcmNlcHRfc2xvcGUsIGFlcyhpbnRlcmNlcHQgPSBpbnRlcmNlcHQsIHNsb3BlID0gc2xvcGUsIGNvbG9yID0gcXVhbnRpbGUpKSArIA0KICB0aGVtZV9taW5pbWFsKCkgKyANCiAgbGFicyh4ID0gIldpZWsiLCB5ID0gIlphcm9ia2kiLCB0aXRsZSA9ICJSZWdyZXNqZSBrd2FudHlsb3dlIHogdGF1ID0gMC4yNSwgMC41MCBvcmF6IDAuNzUiKQ0KYGBgDQoNCg0KYGBge3J9DQpoaXN0KGRhbmU5OCRlYXJuaW5ncykNCmhpc3QobG9nKGRhbmU5OCRlYXJuaW5ncykpDQpgYGANCg0KUsOzd25pZcW8IG5hIHBvZHN0YXdpZSBoaXN0b2dyYW11IGRhbnljaCB6IDE5OTggcm9rdSBtb8W8bmEgemF1d2HFvHnEhywgxbxlIHpsb2dhcnl0bW93YW5pZSB6bWllbm5laiBtb8W8ZSBwb23Ds2MgdyB1enlza2FuaXUgbGVwc3plZ28gbW9kZWx1Lg0KDQpgYGB7cn0NCmRhbmU5OCRsb2dfZWFybmluZ3MgPC0gbG9nKGRhbmU5OCRlYXJuaW5ncykNCmBgYA0KDQpgYGB7cn0NCnBsb3Q0IDwtIGdncGxvdChkYXRhID0gZGFuZTk4KSArDQogICAgZ2VvbV9wb2ludChtYXBwaW5nID0gYWVzKHggPSBhZ2UsIHkgPSBsb2dfZWFybmluZ3MpLCBjb2xvciA9ICJwaW5rIikrIGxhYnModGl0bGU9ICJXeWtyZXMgcm96cnp1dHUgbG9nKGVhcm5pbmdzKSB3emdsxJlkZW0gYWdlIikNCnBsb3Q0DQpgYGANCg0KQnVkb3dhIGkgcG9yw7N3bmFuaWUgbW9kZWxpIHphIHBvbW9jxIUgS01OSyBvcmF6IHJlZ3Jlc2ppIGt3YW50eWxvd2VqIGRsYSB6bWllbm5laiBlYXJuaW5ncyB6bG9nYXJ5dG1vd2FuZWouDQoNCmBgYHtyfQ0KbW9kZWxfbGluaW93eTQgPC0gbG0obG9nX2Vhcm5pbmdzfmRlZ3JlZStnZW5kZXIrYWdlLCBkYXRhPWRhbmU5OCkNCm1vZGVsX3FybG9nNiA8LSBycShsb2dfZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTk4LCB0YXU9MC4yNSkNCm1vZGVsX3FybG9nNyA8LSBycShsb2dfZWFybmluZ3N+ZGVncmVlK2dlbmRlcithZ2UsIGRhdGE9ZGFuZTk4LCB0YXU9MC41KQ0KbW9kZWxfcXJsb2c4IDwtIHJxKGxvZ19lYXJuaW5nc35kZWdyZWUrZ2VuZGVyK2FnZSwgZGF0YT1kYW5lOTgsIHRhdT0wLjc1KQ0Kc3VtbWFyeShtb2RlbF9saW5pb3d5NCkNCkFJQyhtb2RlbF9saW5pb3d5NCwgbW9kZWxfcXJsb2c2LCBtb2RlbF9xcmxvZzcsIG1vZGVsX3FybG9nOCkNCmBgYA0KDQpXIDE5OTggcm9rdSB0YWsgc2FtbyBqYWsgdyAxOTkyIHcgcHJ6eXBhZGt1IHpsb2dhcnl0bW93YW5laiB6bWllbm5laiBvYmphxZtuaWFuZWogZG90eWN6xIVjZWogemFyb2Jrw7N3IGxlcHN6xIUgb2JqYcWbbmlhbG5vxZvEhyBtYSBtb2RlbCBsaW5pb3d5IChuYWpuacW8c3plIEFJQykgbmnFvCBtb2RlbGUgcmVncmVzamkga3dhbnR5bG93ZWouIFNwb8WbcsOzZCBtb2RlbGkgcmVncmVzamkga3dhcnR5bG93ZWogbmFqbGVwc3p5IHpkYWplIHNpxJkgdGVuIGRsYSBtZWRpYW55Lg0KDQpXZWTFgnVnIHd5YnJhbmVnbyBtb2RlbHUsIGN6eWxpIG1vZGVsdSBsaW5pb3dlZ28gb3NvYnkgcG9zaWFkYWrEhWNlIHd5a3N6dGHFgmNlbmllIHd5xbxzemUgemFyYWJpYWrEhSDFm3JlZG5pbyBvIDAsMzggamVkbm9zdGtpIHBpZW5pxJnFvG5laiBuYSBnb2R6aW7EmSB3acSZY2VqIG5pxbwgb3NvYnkgcG9zaWFkYWrEhWNlIHd5a3N6dGHFgmNlbmllIMWbcmVkbmllLiBLb2JpZXR5IHphcmFiaWFqxIUgcHJ6ZWNpxJl0bmllIDAsMTggamVkbm9zdGtpIHBpZW5pxJnFvG5laiBtbmllaiBuacW8IG3EmcW8Y3p5xbpuaS4gV3JheiB6ZSBzdGFyemVuaWVtIHNpxJksIHoga2HFvGR5bSBrb2xlam55bSByb2tpZW0gemFyYWJpYSBzacSZIMWbcmVkbm8gMCwwMiBqZWRub3N0a2kgcGllbmnEmcW8bmUgbmEgZ29kemluxJkgd2nEmWNlai4gDQoNCg0KDQo=