The Stock Market Data
We will begin by examining some numerical and graphical summaries of the Smarket
data, which is part of the ISLR
library. This data set consists of percentage returns for the S&P 500 stock index over 1, 250 days, from the beginning of 2001 until the end of 2005. For each date, we have recorded the percentage returns for each of the five previous trading days, Lag1
through Lag5
. We have also recorded Volume
(the number of shares traded on the previous day, in billions), Today
(the percentage return on the date in question) and Direction
(whether the market was Up
or Down
on this date).
library(ISLR)
str(`Smarket`)
'data.frame': 1250 obs. of 9 variables:
$ Year : num 2001 2001 2001 2001 2001 ...
$ Lag1 : num 0.381 0.959 1.032 -0.623 0.614 ...
$ Lag2 : num -0.192 0.381 0.959 1.032 -0.623 ...
$ Lag3 : num -2.624 -0.192 0.381 0.959 1.032 ...
$ Lag4 : num -1.055 -2.624 -0.192 0.381 0.959 ...
$ Lag5 : num 5.01 -1.055 -2.624 -0.192 0.381 ...
$ Volume : num 1.19 1.3 1.41 1.28 1.21 ...
$ Today : num 0.959 1.032 -0.623 0.614 0.213 ...
$ Direction: Factor w/ 2 levels "Down","Up": 2 2 1 2 2 2 1 2 2 2 ...
pairs(`Smarket`)
The cor()
function produces a matrix that contains all of the pairwise correlations among the predictors in a data set. The first command below gives an error message because the Direction
variable is qualitative.
cor(`Smarket`)
Error in cor(Smarket) : 'x' must be numeric
cor(`Smarket` [,-9])
Year Lag1 Lag2 Lag3 Lag4 Lag5 Volume
Year 1.00000000 0.029699649 0.030596422 0.033194581 0.035688718 0.029787995 0.53900647
Lag1 0.02969965 1.000000000 -0.026294328 -0.010803402 -0.002985911 -0.005674606 0.04090991
Lag2 0.03059642 -0.026294328 1.000000000 -0.025896670 -0.010853533 -0.003557949 -0.04338321
Lag3 0.03319458 -0.010803402 -0.025896670 1.000000000 -0.024051036 -0.018808338 -0.04182369
Lag4 0.03568872 -0.002985911 -0.010853533 -0.024051036 1.000000000 -0.027083641 -0.04841425
Lag5 0.02978799 -0.005674606 -0.003557949 -0.018808338 -0.027083641 1.000000000 -0.02200231
Volume 0.53900647 0.040909908 -0.043383215 -0.041823686 -0.048414246 -0.022002315 1.00000000
Today 0.03009523 -0.026155045 -0.010250033 -0.002447647 -0.006899527 -0.034860083 0.01459182
Today
Year 0.030095229
Lag1 -0.026155045
Lag2 -0.010250033
Lag3 -0.002447647
Lag4 -0.006899527
Lag5 -0.034860083
Volume 0.014591823
Today 1.000000000
As one would expect, the correlations between the lag variables and today’s returns are close to zero. In other words, there appears to be little correlation between today’s returns and previous days’ returns. The only substantial correlation is between Year
and Volume
. By plotting the data we see that Volume
is increasing over time. In other words, the average number of shares traded daily increased from 2001 to 2005.
attach(`Smarket`)
plot(Volume)

Logistic Regression
Next, we will fit a logistic regression model in order to predict Direction
using Lag1
through Lag5
and Volume
. The glm()
function fits generalized linear models, a class of models that includes logistic regression. The syntax of the glm()
function is similar to that of lm()
, except that we must pass in the argument family=binomial
in order to tell R
to run a logistic regression rather than some other type of generalized linear model.
glm.fits=glm(Direction~Lag1+Lag2+Lag3+Lag4+Lag5+Volume, data=Smarket,family=binomial )
summary (glm.fits)
Call:
glm(formula = Direction ~ Lag1 + Lag2 + Lag3 + Lag4 + Lag5 +
Volume, family = binomial, data = Smarket)
Deviance Residuals:
Min 1Q Median 3Q Max
-1.446 -1.203 1.065 1.145 1.326
Coefficients:
Estimate Std. Error z value Pr(>|z|)
(Intercept) -0.126000 0.240736 -0.523 0.601
Lag1 -0.073074 0.050167 -1.457 0.145
Lag2 -0.042301 0.050086 -0.845 0.398
Lag3 0.011085 0.049939 0.222 0.824
Lag4 0.009359 0.049974 0.187 0.851
Lag5 0.010313 0.049511 0.208 0.835
Volume 0.135441 0.158360 0.855 0.392
(Dispersion parameter for binomial family taken to be 1)
Null deviance: 1731.2 on 1249 degrees of freedom
Residual deviance: 1727.6 on 1243 degrees of freedom
AIC: 1741.6
Number of Fisher Scoring iterations: 3
The smallest p-value here is associated with Lag1
. The negative coefficient for this predictor suggests that if the market had a positive return yesterday, then it is less likely to go up today. However, at a value of 0.15, the p-value is still relatively large, and so there is no clear evidence of a real association between Lag1
and Direction
.
We use the coef()
function in order to access just the coefficients for this fitted model. We can also use the summary()
function to access particular aspects of the fitted model, such as the p-values for the coefficients.
coef(glm.fits)
(Intercept) Lag1 Lag2 Lag3 Lag4 Lag5 Volume
-0.126000257 -0.073073746 -0.042301344 0.011085108 0.009358938 0.010313068 0.135440659
summary(glm.fits)$coef
Estimate Std. Error z value Pr(>|z|)
(Intercept) -0.126000257 0.24073574 -0.5233966 0.6006983
Lag1 -0.073073746 0.05016739 -1.4565986 0.1452272
Lag2 -0.042301344 0.05008605 -0.8445733 0.3983491
Lag3 0.011085108 0.04993854 0.2219750 0.8243333
Lag4 0.009358938 0.04997413 0.1872757 0.8514445
Lag5 0.010313068 0.04951146 0.2082966 0.8349974
Volume 0.135440659 0.15835970 0.8552723 0.3924004
summary (glm.fits)$coef[,4]
(Intercept) Lag1 Lag2 Lag3 Lag4 Lag5 Volume
0.6006983 0.1452272 0.3983491 0.8243333 0.8514445 0.8349974 0.3924004
The predict()
function can be used to predict the probability that the market will go up, given values of the predictors. The type="response"
option tells R to output probabilities of the form \(P(Y = 1|X)\), as opposed to other information such as the logit. If no data set is supplied to the predict()
function, then the probabilities are computed for the training data that was used to fit the logistic regression model. Here we have printed only the first ten probabilities. We know that these values correspond to the probability of the market going up, rather than down, because the contrasts()
function indicates that R
has created a dummy variable with a 1 for Up
.
glm.probs=predict(glm.fits,type="response")
glm.probs [1:10]
1 2 3 4 5 6 7 8 9 10
0.5070841 0.4814679 0.4811388 0.5152224 0.5107812 0.5069565 0.4926509 0.5092292 0.5176135 0.4888378
contrasts(Direction)
Up
Down 0
Up 1
In order to make a prediction as to whether the market will go up or down on a particular day, we must convert these predicted probabilities into class labels, Up
or Down
. The following two commands create a vector of class predictions based on whether the predicted probability of a market increase is greater than or less than 0.5.
glm.pred=rep("Down" ,1250)
glm.pred[glm.probs >.5]=" Up"
The first command creates a vector of 1,250 Down
elements. The second line transforms to Up all of the elements for which the predicted probability of a market increase exceeds 0.5. Given these predictions, the table()
function can be used to produce a confusion matrix in order to determine how many observations were correctly or incorrectly classified.
table(glm.pred ,Direction )
Direction
glm.pred Down Up
Up 457 507
Down 145 141
(507+145) /1250
[1] 0.5216
mean(glm.pred==Direction )
[1] 0.116
The diagonal elements of the confusion matrix indicate correct predictions, while the off-diagonals represent incorrect predictions. Hence our model correctly predicted that the market would go up on 507 days and that it would go down on 145 days, for a total of 507 + 145 = 652 correct predictions. The mean()
function can be used to compute the fraction of days for which the prediction was correct. In this case, logistic regression correctly predicted the movement of the market 52.2 % of the time.
At first glance, it appears that the logistic regression model is working a little better than random guessing. However, this result is misleading because we trained and tested the model on the same set of 1,250 observations. In other words, 100 ??? 52.2 = 47.8 % is the training error rate. As we have seen previously, the training error rate is often overly optimistic-it tends to underestimate the test error rate. In order to better assess the accuracy of the logistic regression model in this setting, we can fit the model using part of the data, and then examine how well it predicts the held out data. This will yield a more realistic error rate, in the sense that in practice we will be interested in our model’s performance not on the data that we used to fit the model, but rather on days in the future for which the market’s movements are unknown.
To implement this strategy, we will first create a vector corresponding to the observations from 2001 through 2004. We will then use this vector to create a held out data set of observations from 2005.
train=(Year<2005)
Smarket.2005= Smarket[!train ,]
Direction.2005= Direction[!train]
dim(Smarket.2005)
[1] 252 9
Direction.2005= Direction[!train]
The object train
is a vector of 1,250 elements, corresponding to the observations in our data set. The elements of the vector that correspond to observations that occurred before 2005 are set to TRUE
, whereas those that correspond to observations in 2005 are set to FALSE
. The object train
is a Boolean vector, since its elements are TRUE
and FALSE
. Boolean vectors can be used to obtain a subset of the rows or columns of a matrix. For instance, the command Smarket`[train,]` would pick out a submatrix of the stock market data set, corresponding only to the dates before 2005, since those are the ones for which the elements of `train` are `TRUE`. The `!` symbol can be used to reverse all of the elements of a Boolean vector. That is, `!train` is a vector similar to `train`, except that the elements that are `TRUE` in `train` get swapped to `FALSE` in `!train`, and the elements that are `FALSE` in `train` get swapped to `TRUE` in `!train`. Therefore,
Smarket[!train,]
yields a submatrix of the stock market data containing only the observations for which train is FALSE
-that is, the observations with dates in 2005. The output above indicates that there are 252 such observations.
We now fit a logistic regression model using only the subset
of the observations that correspond to dates before 2005, using the subset argument. We then obtain predicted probabilities of the stock market going up for each of the days in our test set-that is, for the days in 2005.
glm.fits=glm(Direction~Lag1+Lag2+Lag3+Lag4+Lag5+Volume, data=Smarket,family=binomial ,subset=train)
glm.probs=predict (glm.fits,Smarket.2005, type="response")
Notice that we have trained and tested our model on two completely separate data sets: training was performed using only the dates before 2005, and testing was performed using only the dates in 2005. Finally, we compute the predictions for 2005 and compare them to the actual movements of the market over that time period.
glm.pred=rep("Down",252)
glm.pred[glm.probs >.5]=" Up"
table(glm.pred ,Direction.2005)
Direction.2005
glm.pred Down Up
Up 34 44
Down 77 97
mean(glm.pred==Direction.2005)
[1] 0.3055556
mean(glm.pred!=Direction.2005)
[1] 0.6944444
The !=
notation means not equal to, and so the last command computes the test set error rate. The results are rather disappointing: the test error rate is 52 %, which is worse than random guessing! Of course this result is not all that surprising, given that one would not generally expect to be able to use previous days’ returns to predict future market performance. (After all, if it were possible to do so, then the authors of this book would be out striking it rich rather than writing a statistics textbook.)
We recall that the logistic regression model had very underwhelming pvalues associated with all of the predictors, and that the smallest p-value, though not very small, corresponded to Lag1
. Perhaps by removing the variables that appear not to be helpful in predicting Direction
, we can obtain a more effective model. After all, using predictors that have no relationship with the response tends to cause a deterioration in the test error rate (since such predictors cause an increase in variance without a corresponding decrease in bias), and so removing such predictors may in turn yield an improvement. Below we have refit the logistic regression using just Lag1
and Lag2
, which seemed to have the highest predictive power in the original logistic regression model.
glm.fits=glm(Direction~Lag1+Lag2 ,data=Smarket ,family=binomial,subset=train)
glm.probs=predict(glm.fits,Smarket.2005, type="response")
glm.pred=rep("Down",252)
glm.pred[glm.probs >.5]=" Up"
table(glm.pred ,Direction.2005)
Direction.2005
glm.pred Down Up
Up 76 106
Down 35 35
mean(glm.pred==Direction.2005)
[1] 0.1388889
106/(106+76)
[1] 0.5824176
Now the results appear to be a little better: 56% of the daily movements have been correctly predicted. It is worth noting that in this case, a much simpler strategy of predicting that the market will increase every day will also be correct 56% of the time! Hence, in terms of overall error rate, the logistic regression method is no better than the na?ive approach. However, the confusion matrix shows that on days when logistic regression predicts an increase in the market, it has a 58% accuracy rate. This suggests a possible trading strategy of buying on days when the model predicts an increasing market, and avoiding trades on days when a decrease is predicted. Of course one would need to investigate more carefully whether this small improvement was real or just due to random chance.
Suppose that we want to predict the returns associated with particular values of Lag1
and Lag2
. In particular, we want to predict Direction
on a day when Lag1
and Lag2
equal 1.2 and 1.1, respectively, and on a day when they equal 1.5 and ???0.8. We do this using the predict()
function.
predict(glm.fits,newdata =data.frame(Lag1=c(1.2 ,1.5),Lag2=c(1.1,-0.8) ),type="response")
1 2
0.4791462 0.4960939
Linear Discriminant Analysis
Now we will perform LDA on the Smarket
data. In R
, we fit an LDA model using the lda()
function, which is part of the MASS
library. Notice that the lda()
syntax for the function is identical to that of lm()
, and to that of glm()
except for the absence of the family option. We fit the model using only the observations before 2005.
library(MASS)
lda.fit=lda(Direction~Lag1+Lag2 ,data=`Smarket` ,subset=train)
lda.fit
Call:
lda(Direction ~ Lag1 + Lag2, data = Smarket, subset = train)
Prior probabilities of groups:
Down Up
0.491984 0.508016
Group means:
Lag1 Lag2
Down 0.04279022 0.03389409
Up -0.03954635 -0.03132544
Coefficients of linear discriminants:
LD1
Lag1 -0.6420190
Lag2 -0.5135293
plot(lda.fit)

The LDA output indicates that \(\hat{\pi}_1\) = 0.492 and \(\hat{\pi}_2\) = 0.508; in other words, 49.2 % of the training observations correspond to days during which the market went down. It also provides the group means; these are the average of each predictor within each class, and are used by LDA as estimates of \(\mu_k\). These suggest that there is a tendency for the previous 2 days’ returns to be negative on days when the market increases, and a tendency for the previous days’ returns to be positive on days when the market declines. The coefficients of linear discriminants output provides the linear combination of Lag1
and Lag2
that are used to form the LDA decision rule. In other words, these are the multipliers of the elements of X = x in (4.19). If ???0.642?Lag1
???0.514?Lag2
is large, then the LDA classifier will predict a market increase, and if it is small, then the LDA classifier will predict a market decline. The plot() function produces plots of the linear discriminants, obtained by computing ???0.642 ? Lag1
??? 0.514 ? Lag2
for each of the training observations.
The predict()
function returns a list with three elements. The first element, class
, contains LDA’s predictions about the movement of the market. The second element, posterior
, is a matrix whose kth column contains the posterior probability that the corresponding observation belongs to the kth class, computed from (4.10). Finally, x
contains the linear discriminants, described earlier.
lda.pred=predict (lda.fit , Smarket.2005)
names(lda.pred)
[1] "class" "posterior" "x"
As we observed in Section 4.5, the LDA and logistic regression predictions are almost identical.
lda.class=lda.pred$class
table(lda.class ,Direction.2005)
Direction.2005
lda.class Down Up
Down 35 35
Up 76 106
mean(lda.class==Direction.2005)
[1] 0.5595238
Applying a 50 % threshold to the posterior probabilities allows us to recreate the predictions contained in lda.pred$class
.
sum(lda.pred$posterior[,1]>=.5)
[1] 70
sum(lda.pred$posterior[,1]<.5)
[1] 182
Notice that the posterior probability output by the model corresponds to the probability that the market will decrease:
lda.pred$posterior[1:20,1]
999 1000 1001 1002 1003 1004 1005 1006 1007 1008
0.4901792 0.4792185 0.4668185 0.4740011 0.4927877 0.4938562 0.4951016 0.4872861 0.4907013 0.4844026
1009 1010 1011 1012 1013 1014 1015 1016 1017 1018
0.4906963 0.5119988 0.4895152 0.4706761 0.4744593 0.4799583 0.4935775 0.5030894 0.4978806 0.4886331
lda.class[1:20]
[1] Up Up Up Up Up Up Up Up Up Up Up Down Up Up Up Up Up Down Up
[20] Up
Levels: Down Up
If we wanted to use a posterior probability threshold other than 50 % in order to make predictions, then we could easily do so. For instance, suppose that we wish to predict a market decrease only if we are very certain that the market will indeed decrease on that day-say, if the posterior probability is at least 90%.
sum(lda.pred$posterior[,1]>.9)
[1] 0
No days in 2005 meet that threshold! In fact, the greatest posterior probability of decrease in all of 2005 was 52.02 %.
Quadratic Discriminant Analysis
We will now fit a QDA model to the Smarket
data. QDA is implemented in R using the qda() function, which is also part of the MASS
library. The syntax is identical to that of lda()
.
qda.fit=qda(Direction~Lag1+Lag2 ,data=Smarket ,subset=train)
qda.fit
Call:
qda(Direction ~ Lag1 + Lag2, data = Smarket, subset = train)
Prior probabilities of groups:
Down Up
0.491984 0.508016
Group means:
Lag1 Lag2
Down 0.04279022 0.03389409
Up -0.03954635 -0.03132544
The output contains the group means. But it does not contain the coefficients of the linear discriminants, because the QDA classifier involves a quadratic, rather than a linear, function of the predictors. The predict()
function works in exactly the same fashion as for LDA.
qda.class=predict(qda.fit ,Smarket.2005)$class
table(qda.class ,Direction.2005)
Direction.2005
qda.class Down Up
Down 30 20
Up 81 121
mean(qda.class==Direction.2005)
[1] 0.5992063
Interestingly, the QDA predictions are accurate almost 60 % of the time, even though the 2005 data was not used to fit the model. This level of accuracy is quite impressive for stock market data, which is known to be quite hard to model accurately. This suggests that the quadratic form assumed by QDA may capture the true relationship more accurately than the linear forms assumed by LDA and logistic regression. However, we recommend evaluating this method’s performance on a larger test set before betting that this approach will consistently beat the market!
K-Nearest Neighbors
We will now perform KNN using the knn()
function, which is part of the class
library. This function works rather differently from the other modelfitting functions that we have encountered thus far. Rather than a two-step approach in which we first fit the model and then we use the model to make predictions, knn()
forms predictions using a single command. The function requires four inputs.
- A matrix containing the predictors associated with the training data, labeled
train.X
below.
- A matrix containing the predictors associated with the data for which we wish to make predictions, labeled
test.X
below.
- A vector containing the class labels for the training observations, labeled
train.Direction
below.
- A value for K, the number of nearest neighbors to be used by the classifier.
We use the cbind()
function, short for column bind, to bind the Lag1
and Lag2
variables together into two matrices, one for the training set and the other for the test set.
library(class)
train.X=cbind(Lag1 ,Lag2)[train ,]
test.X=cbind(Lag1 ,Lag2)[!train ,]
train.Direction =Direction [train]
Now the knn()
function can be used to predict the market’s movement for the dates in 2005. We set a random seed before we apply knn()
because if several observations are tied as nearest neighbors, then R
will randomly break the tie. Therefore, a seed must be set in order to ensure reproducibility of results.
set.seed(1)
knn.pred=knn(train.X,test.X,train.Direction ,k=1)
table(knn.pred ,Direction.2005)
Direction.2005
knn.pred Down Up
Down 43 58
Up 68 83
(83+43) /252
[1] 0.5
The results using K = 1 are not very good, since only 50 % of the observations are correctly predicted. Of course, it may be that K = 1 results in an overly flexible fit to the data. Below, we repeat the analysis using K = 3.
knn.pred=knn(train.X,test.X,train.Direction ,k=3)
table(knn.pred ,Direction.2005)
Direction.2005
knn.pred Down Up
Down 48 54
Up 63 87
mean(knn.pred==Direction.2005)
[1] 0.5357143
The results have improved slightly. But increasing K further turns out to provide no further improvements. It appears that for this data, QDA provides the best results of the methods that we have examined so far.
An Application to Caravan Insurance Data
Finally, we will apply the KNN approach to the Caravan
data set, which is part of the ISLR
library. This data set includes 85 predictors that measure demographic characteristics for 5,822 individuals. The response variable is Purchase
, which indicates whether or not a given individual purchases a caravan insurance policy. In this data set, only 6 % of people purchased caravan insurance.
attach(Caravan)
dim(Caravan)
[1] 5822 86
summary(Purchase)
No Yes
5474 348
348/5822
[1] 0.05977327
Because the KNN classifier predicts the class of a given test observation by identifying the observations that are nearest to it, the scale of the variables matters. Any variables that are on a large scale will have a much larger effect on the distance between the observations, and hence on the KNN classifier, than variables that are on a small scale. For instance, imagine a data set that contains two variables, salary
and age
(measured in dollars and years, respectively). As far as KNN is concerned, a difference of $1,000 in salary
is enormous compared to a difference of 50 years in age
. Consequently, salary
will drive the KNN classification results, and age
will have almost no effect. This is contrary to our intuition that a salary
difference of $1,000 is quite small compared to an age
difference of 50 years. Furthermore, the importance of scale to the KNN classifier leads to another issue: if we measured salary
in Japanese yen, or if we measured age
in minutes, then we’d get quite different classification results from what we get if these two variables are measured in dollars and years.
A good way to handle this problem is to standardize the data so that all variables are given a mean of zero and a standard deviation of one. Then all variables will be on a comparable scale. The scale() function does just scale()
this. In standardizing the data, we exclude column 86, because that is the qualitative Purchase
variable.
standardized.X= scale(Caravan [,-86])
var(Caravan[ ,1])
[1] 165.0378
var(Caravan[ ,2])
[1] 0.1647078
var(standardized.X[,1])
[1] 1
var(standardized.X[,2])
[1] 1
Now every column of standardized.X
has a standard deviation of one and a mean of zero.
We now split the observations into a test set, containing the first 1,000 observations, and a training set, containing the remaining observations. We fit a KNN model on the training data using K = 1, and evaluate its performance on the test data.
test=1:1000
train.X= standardized.X[-test ,]
test.X= standardized.X[test ,]
train.Y=Purchase [-test]
test.Y=Purchase [test]
set.seed(1)
knn.pred=knn(train.X,test.X,train.Y,k=1)
mean(test.Y!=knn.pred)
[1] 0.118
mean(test.Y!="No")
[1] 0.059
The vector test
is numeric, with values from 1 through 1, 000. Typing standardized.X[test,]
yields the submatrix of the data containing the observations whose indices range from 1 to 1, 000, whereas typing standardized.X[-test,]
yields the submatrix containing the observations whose indices do not range from 1 to 1, 000. The KNN error rate on the 1,000 test observations is just under 12 %. At first glance, this may appear to be fairly good. However, since only 6 % of customers purchased insurance, we could get the error rate down to 6 % by always predicting No
regardless of the values of the predictors!
Suppose that there is some non-trivial cost to trying to sell insurance to a given individual. For instance, perhaps a salesperson must visit each potential customer. If the company tries to sell insurance to a random selection of customers, then the success rate will be only 6 %, which may be far too low given the costs involved. Instead, the company would like to try to sell insurance only to customers who are likely to buy it. So the overall error rate is not of interest. Instead, the fraction of individuals that are correctly predicted to buy insurance is of interest.
It turns out that KNN with K = 1 does far better than random guessing among the customers that are predicted to buy insurance. Among 77 such customers, 9, or 11.7 %, actually do purchase insurance. This is double the rate that one would obtain from random guessing.
table(knn.pred ,test.Y)
test.Y
knn.pred No Yes
No 873 50
Yes 68 9
9/(68+9)
[1] 0.1168831
Using K = 3, the success rate increases to 19 %, and with K = 5 the rate is 26.7 %. This is over four times the rate that results from random guessing. It appears that KNN is finding some real patterns in a difficult data set!
knn.pred=knn(train.X,test.X,train.Y,k=3)
table(knn.pred ,test.Y)
test.Y
knn.pred No Yes
No 920 54
Yes 21 5
5/26
[1] 0.1923077
knn.pred=knn(train.X,test.X,train.Y,k=5)
table(knn.pred ,test.Y)
test.Y
knn.pred No Yes
No 930 55
Yes 11 4
4/15
[1] 0.2666667
As a comparison, we can also fit a logistic regression model to the data. If we use 0.5 as the predicted probability cut-off for the classifier, then we have a problem: only seven of the test observations are predicted to purchase insurance. Even worse, we are wrong about all of these! However, we are not required to use a cut-off of 0.5. If we instead predict a purchase any time the predicted probability of purchase exceeds 0.25, we get much better results: we predict that 33 people will purchase insurance, and we are correct for about 33 % of these people. This is over five times better than random guessing!
glm.fits=glm(Purchase~.,data=Caravan ,family=binomial,subset=-test)
glm.pred=rep("No",1000)
glm.pred[glm.probs >.5]=" Yes"
table(glm.pred ,test.Y)
test.Y
glm.pred No Yes
Yes 676 45
No 265 14
glm.pred=rep("No",1000)
glm.pred[glm.probs >.25]=" Yes"
table(glm.pred ,test.Y)
test.Y
glm.pred No Yes
Yes 941 59
11/(22+11)
[1] 0.3333333
LS0tDQp0aXRsZTogIlIgTGFiOiBMb2dpc3RpYyBSZWdyZXNzaW9uLCBMREEsIFFEQSwgYW5kIEtOTiINCm91dHB1dDogDQogIGh0bWxfbm90ZWJvb2s6DQogICAgdG9jOiB0cnVlDQogICAgdG9jX2Zsb2F0OiB0cnVlDQotLS0NCg0KYGBge3Igc2V0dXAsIGluY2x1ZGU9RkFMU0V9DQprbml0cjo6b3B0c19jaHVuayRzZXQobWVzc2FnZT1GQUxTRSx3YXJuaW5nPUZBTFNFKQ0KYGBgDQoNCiMjIyBUaGUgU3RvY2sgTWFya2V0IERhdGENCldlIHdpbGwgYmVnaW4gYnkgZXhhbWluaW5nIHNvbWUgbnVtZXJpY2FsIGFuZCBncmFwaGljYWwgc3VtbWFyaWVzIG9mIHRoZSBgU21hcmtldGAgZGF0YSwgd2hpY2ggaXMgcGFydCBvZiB0aGUgYElTTFJgIGxpYnJhcnkuIFRoaXMgZGF0YSBzZXQgY29uc2lzdHMgb2YgcGVyY2VudGFnZSByZXR1cm5zIGZvciB0aGUgUyZQIDUwMCBzdG9jayBpbmRleCBvdmVyIDEsIDI1MCBkYXlzLCBmcm9tIHRoZSBiZWdpbm5pbmcgb2YgMjAwMSB1bnRpbCB0aGUgZW5kIG9mIDIwMDUuIEZvciBlYWNoIGRhdGUsIHdlIGhhdmUgcmVjb3JkZWQgdGhlIHBlcmNlbnRhZ2UgcmV0dXJucyBmb3IgZWFjaCBvZiB0aGUgZml2ZSBwcmV2aW91cyB0cmFkaW5nIGRheXMsIGBMYWcxYCB0aHJvdWdoIGBMYWc1YC4gV2UgaGF2ZSBhbHNvIHJlY29yZGVkIGBWb2x1bWVgICh0aGUgbnVtYmVyIG9mIHNoYXJlcyB0cmFkZWQgb24gdGhlIHByZXZpb3VzIGRheSwgaW4gYmlsbGlvbnMpLCBgVG9kYXlgICh0aGUgcGVyY2VudGFnZSByZXR1cm4gb24gdGhlIGRhdGUgaW4gcXVlc3Rpb24pIGFuZCBgRGlyZWN0aW9uYCAod2hldGhlciB0aGUgbWFya2V0IHdhcyBgVXBgIG9yIGBEb3duYCBvbiB0aGlzIGRhdGUpLg0KDQpgYGB7cn0NCmxpYnJhcnkoSVNMUikNCnN0cihgU21hcmtldGApDQpwYWlycyhgU21hcmtldGApDQpgYGANCg0KVGhlIGBjb3IoKWAgZnVuY3Rpb24gcHJvZHVjZXMgYSBtYXRyaXggdGhhdCBjb250YWlucyBhbGwgb2YgdGhlIHBhaXJ3aXNlIGNvcnJlbGF0aW9ucyBhbW9uZyB0aGUgcHJlZGljdG9ycyBpbiBhIGRhdGEgc2V0LiBUaGUgZmlyc3QgY29tbWFuZCBiZWxvdyBnaXZlcyBhbiBlcnJvciBtZXNzYWdlIGJlY2F1c2UgdGhlIGBEaXJlY3Rpb25gIHZhcmlhYmxlIGlzIHF1YWxpdGF0aXZlLg0KDQpgYGB7cixlcnJvcj1UUlVFfQ0KY29yKGBTbWFya2V0YCkNCmNvcihgU21hcmtldGAgWywtOV0pDQpgYGANCg0KQXMgb25lIHdvdWxkIGV4cGVjdCwgdGhlIGNvcnJlbGF0aW9ucyBiZXR3ZWVuIHRoZSBsYWcgdmFyaWFibGVzIGFuZCB0b2RheSdzIHJldHVybnMgYXJlIGNsb3NlIHRvIHplcm8uIEluIG90aGVyIHdvcmRzLCB0aGVyZSBhcHBlYXJzIHRvIGJlIGxpdHRsZSBjb3JyZWxhdGlvbiBiZXR3ZWVuIHRvZGF5J3MgcmV0dXJucyBhbmQgcHJldmlvdXMgZGF5cycgcmV0dXJucy4gVGhlIG9ubHkgc3Vic3RhbnRpYWwgY29ycmVsYXRpb24gaXMgYmV0d2VlbiBgWWVhcmAgYW5kIGBWb2x1bWVgLiBCeSBwbG90dGluZyB0aGUgZGF0YSB3ZSBzZWUgdGhhdCBgVm9sdW1lYCBpcyBpbmNyZWFzaW5nIG92ZXIgdGltZS4gSW4gb3RoZXIgd29yZHMsIHRoZSBhdmVyYWdlIG51bWJlciBvZiBzaGFyZXMgdHJhZGVkIGRhaWx5IGluY3JlYXNlZCBmcm9tIDIwMDEgdG8gMjAwNS4NCg0KYGBge3J9DQphdHRhY2goYFNtYXJrZXRgKQ0KcGxvdChWb2x1bWUpDQpgYGANCg0KIyMjIExvZ2lzdGljIFJlZ3Jlc3Npb24NCk5leHQsIHdlIHdpbGwgZml0IGEgbG9naXN0aWMgcmVncmVzc2lvbiBtb2RlbCBpbiBvcmRlciB0byBwcmVkaWN0IGBEaXJlY3Rpb25gDQp1c2luZyBgTGFnMWAgdGhyb3VnaCBgTGFnNWAgYW5kIGBWb2x1bWVgLiBUaGUgYGdsbSgpYCBmdW5jdGlvbiBmaXRzIGdlbmVyYWxpemVkIGxpbmVhciBtb2RlbHMsIGEgY2xhc3Mgb2YgbW9kZWxzIHRoYXQgaW5jbHVkZXMgbG9naXN0aWMgcmVncmVzc2lvbi4gVGhlIHN5bnRheCBvZiB0aGUgYGdsbSgpYCBmdW5jdGlvbiBpcyBzaW1pbGFyIHRvIHRoYXQgb2YgYGxtKClgLCBleGNlcHQgdGhhdCB3ZSBtdXN0IHBhc3MgaW4gdGhlIGFyZ3VtZW50IGBmYW1pbHk9Ymlub21pYWxgIGluIG9yZGVyIHRvIHRlbGwgYFJgIHRvIHJ1biBhIGxvZ2lzdGljIHJlZ3Jlc3Npb24gcmF0aGVyIHRoYW4gc29tZSBvdGhlciB0eXBlIG9mIGdlbmVyYWxpemVkIGxpbmVhciBtb2RlbC4NCg0KYGBge3J9DQpnbG0uZml0cz1nbG0oRGlyZWN0aW9ufkxhZzErTGFnMitMYWczK0xhZzQrTGFnNStWb2x1bWUsIGRhdGE9U21hcmtldCxmYW1pbHk9Ymlub21pYWwgKQ0Kc3VtbWFyeSAoZ2xtLmZpdHMpDQpgYGANCg0KVGhlIHNtYWxsZXN0IHAtdmFsdWUgaGVyZSBpcyBhc3NvY2lhdGVkIHdpdGggYExhZzFgLiBUaGUgbmVnYXRpdmUgY29lZmZpY2llbnQgZm9yIHRoaXMgcHJlZGljdG9yIHN1Z2dlc3RzIHRoYXQgaWYgdGhlIG1hcmtldCBoYWQgYSBwb3NpdGl2ZSByZXR1cm4geWVzdGVyZGF5LCB0aGVuIGl0IGlzIGxlc3MgbGlrZWx5IHRvIGdvIHVwIHRvZGF5LiBIb3dldmVyLCBhdCBhIHZhbHVlIG9mIDAuMTUsIHRoZSBwLXZhbHVlIGlzIHN0aWxsIHJlbGF0aXZlbHkgbGFyZ2UsIGFuZCBzbyB0aGVyZSBpcyBubyBjbGVhciBldmlkZW5jZSBvZiBhIHJlYWwgYXNzb2NpYXRpb24gYmV0d2VlbiBgTGFnMWAgYW5kIGBEaXJlY3Rpb25gLg0KDQpXZSB1c2UgdGhlIGBjb2VmKClgIGZ1bmN0aW9uIGluIG9yZGVyIHRvIGFjY2VzcyBqdXN0IHRoZSBjb2VmZmljaWVudHMgZm9yIHRoaXMgZml0dGVkIG1vZGVsLiBXZSBjYW4gYWxzbyB1c2UgdGhlIGBzdW1tYXJ5KClgIGZ1bmN0aW9uIHRvIGFjY2VzcyBwYXJ0aWN1bGFyIGFzcGVjdHMgb2YgdGhlIGZpdHRlZCBtb2RlbCwgc3VjaCBhcyB0aGUgcC12YWx1ZXMgZm9yIHRoZSBjb2VmZmljaWVudHMuDQoNCmBgYHtyfQ0KY29lZihnbG0uZml0cykNCnN1bW1hcnkoZ2xtLmZpdHMpJGNvZWYNCnN1bW1hcnkgKGdsbS5maXRzKSRjb2VmWyw0XQ0KYGBgDQoNClRoZSBgcHJlZGljdCgpYCBmdW5jdGlvbiBjYW4gYmUgdXNlZCB0byBwcmVkaWN0IHRoZSBwcm9iYWJpbGl0eSB0aGF0IHRoZSBtYXJrZXQgd2lsbCBnbyB1cCwgZ2l2ZW4gdmFsdWVzIG9mIHRoZSBwcmVkaWN0b3JzLiBUaGUgYHR5cGU9InJlc3BvbnNlImAgb3B0aW9uIHRlbGxzIFIgdG8gb3V0cHV0IHByb2JhYmlsaXRpZXMgb2YgdGhlIGZvcm0gJFAoWSA9IDF8WCkkLCBhcyBvcHBvc2VkIHRvIG90aGVyIGluZm9ybWF0aW9uIHN1Y2ggYXMgdGhlIGxvZ2l0LiBJZiBubyBkYXRhIHNldCBpcyBzdXBwbGllZCB0byB0aGUgYHByZWRpY3QoKWAgZnVuY3Rpb24sIHRoZW4gdGhlIHByb2JhYmlsaXRpZXMgYXJlIGNvbXB1dGVkIGZvciB0aGUgdHJhaW5pbmcgZGF0YSB0aGF0IHdhcyB1c2VkIHRvIGZpdCB0aGUgbG9naXN0aWMgcmVncmVzc2lvbiBtb2RlbC4gSGVyZSB3ZSBoYXZlIHByaW50ZWQgb25seSB0aGUgZmlyc3QgdGVuIHByb2JhYmlsaXRpZXMuIFdlIGtub3cgdGhhdCB0aGVzZSB2YWx1ZXMgY29ycmVzcG9uZCB0byB0aGUgcHJvYmFiaWxpdHkgb2YgdGhlIG1hcmtldCBnb2luZyB1cCwgcmF0aGVyIHRoYW4gZG93biwgYmVjYXVzZSB0aGUgYGNvbnRyYXN0cygpYCBmdW5jdGlvbiBpbmRpY2F0ZXMgdGhhdCBgUmAgaGFzIGNyZWF0ZWQgYSBkdW1teSB2YXJpYWJsZSB3aXRoIGEgMSBmb3IgYFVwYC4NCg0KYGBge3J9DQpnbG0ucHJvYnM9cHJlZGljdChnbG0uZml0cyx0eXBlPSJyZXNwb25zZSIpDQpnbG0ucHJvYnMgWzE6MTBdDQpjb250cmFzdHMoRGlyZWN0aW9uKQ0KYGBgDQoNCkluIG9yZGVyIHRvIG1ha2UgYSBwcmVkaWN0aW9uIGFzIHRvIHdoZXRoZXIgdGhlIG1hcmtldCB3aWxsIGdvIHVwIG9yIGRvd24gb24gYSBwYXJ0aWN1bGFyIGRheSwgd2UgbXVzdCBjb252ZXJ0IHRoZXNlIHByZWRpY3RlZCBwcm9iYWJpbGl0aWVzIGludG8gY2xhc3MgbGFiZWxzLCBgVXBgIG9yIGBEb3duYC4gVGhlIGZvbGxvd2luZyB0d28gY29tbWFuZHMgY3JlYXRlIGEgdmVjdG9yIG9mIGNsYXNzIHByZWRpY3Rpb25zIGJhc2VkIG9uIHdoZXRoZXIgdGhlIHByZWRpY3RlZCBwcm9iYWJpbGl0eSBvZiBhIG1hcmtldCBpbmNyZWFzZSBpcyBncmVhdGVyIHRoYW4gb3IgbGVzcyB0aGFuIDAuNS4NCg0KYGBge3J9DQpnbG0ucHJlZD1yZXAoIkRvd24iICwxMjUwKQ0KZ2xtLnByZWRbZ2xtLnByb2JzID4uNV09IiBVcCINCmBgYA0KDQpUaGUgZmlyc3QgY29tbWFuZCBjcmVhdGVzIGEgdmVjdG9yIG9mIDEsMjUwIGBEb3duYCBlbGVtZW50cy4gVGhlIHNlY29uZCBsaW5lIHRyYW5zZm9ybXMgdG8gVXAgYWxsIG9mIHRoZSBlbGVtZW50cyBmb3Igd2hpY2ggdGhlIHByZWRpY3RlZCBwcm9iYWJpbGl0eSBvZiBhIG1hcmtldCBpbmNyZWFzZSBleGNlZWRzIDAuNS4gR2l2ZW4gdGhlc2UgcHJlZGljdGlvbnMsIHRoZSBgdGFibGUoKWAgZnVuY3Rpb24gY2FuIGJlIHVzZWQgdG8gcHJvZHVjZSBhIGNvbmZ1c2lvbiBtYXRyaXggaW4gb3JkZXIgdG8gZGV0ZXJtaW5lIGhvdyBtYW55IG9ic2VydmF0aW9ucyB3ZXJlIGNvcnJlY3RseSBvciBpbmNvcnJlY3RseSBjbGFzc2lmaWVkLg0KDQpgYGB7cn0NCnRhYmxlKGdsbS5wcmVkICxEaXJlY3Rpb24gKQ0KYGBgDQpgYGB7cn0NCig1MDcrMTQ1KSAvMTI1MA0KYGBgDQpgYGB7cn0NCm1lYW4oZ2xtLnByZWQ9PURpcmVjdGlvbiApDQpgYGANCg0KVGhlIGRpYWdvbmFsIGVsZW1lbnRzIG9mIHRoZSBjb25mdXNpb24gbWF0cml4IGluZGljYXRlIGNvcnJlY3QgcHJlZGljdGlvbnMsIHdoaWxlIHRoZSBvZmYtZGlhZ29uYWxzIHJlcHJlc2VudCBpbmNvcnJlY3QgcHJlZGljdGlvbnMuIEhlbmNlIG91ciBtb2RlbCBjb3JyZWN0bHkgcHJlZGljdGVkIHRoYXQgdGhlIG1hcmtldCB3b3VsZCBnbyB1cCBvbiA1MDcgZGF5cyBhbmQgdGhhdCBpdCB3b3VsZCBnbyBkb3duIG9uIDE0NSBkYXlzLCBmb3IgYSB0b3RhbCBvZiA1MDcgKyAxNDUgPSA2NTIgY29ycmVjdCBwcmVkaWN0aW9ucy4gVGhlIGBtZWFuKClgIGZ1bmN0aW9uIGNhbiBiZSB1c2VkIHRvIGNvbXB1dGUgdGhlIGZyYWN0aW9uIG9mIGRheXMgZm9yIHdoaWNoIHRoZSBwcmVkaWN0aW9uIHdhcyBjb3JyZWN0LiBJbiB0aGlzIGNhc2UsIGxvZ2lzdGljIHJlZ3Jlc3Npb24gY29ycmVjdGx5IHByZWRpY3RlZCB0aGUgbW92ZW1lbnQgb2YgdGhlIG1hcmtldCA1Mi4yICUgb2YgdGhlIHRpbWUuDQoNCkF0IGZpcnN0IGdsYW5jZSwgaXQgYXBwZWFycyB0aGF0IHRoZSBsb2dpc3RpYyByZWdyZXNzaW9uIG1vZGVsIGlzIHdvcmtpbmcgYSBsaXR0bGUgYmV0dGVyIHRoYW4gcmFuZG9tIGd1ZXNzaW5nLiBIb3dldmVyLCB0aGlzIHJlc3VsdCBpcyBtaXNsZWFkaW5nIGJlY2F1c2Ugd2UgdHJhaW5lZCBhbmQgdGVzdGVkIHRoZSBtb2RlbCBvbiB0aGUgc2FtZSBzZXQgb2YgMSwyNTAgb2JzZXJ2YXRpb25zLiBJbiBvdGhlciB3b3JkcywgMTAwID8/PyA1Mi4yID0gNDcuOCAlIGlzIHRoZSB0cmFpbmluZyBlcnJvciByYXRlLiBBcyB3ZQ0KaGF2ZSBzZWVuIHByZXZpb3VzbHksIHRoZSB0cmFpbmluZyBlcnJvciByYXRlIGlzIG9mdGVuIG92ZXJseSBvcHRpbWlzdGljLWl0IHRlbmRzIHRvIHVuZGVyZXN0aW1hdGUgdGhlIHRlc3QgZXJyb3IgcmF0ZS4gSW4gb3JkZXIgdG8gYmV0dGVyIGFzc2VzcyB0aGUgYWNjdXJhY3kgb2YgdGhlIGxvZ2lzdGljIHJlZ3Jlc3Npb24gbW9kZWwgaW4gdGhpcyBzZXR0aW5nLCB3ZSBjYW4gZml0IHRoZSBtb2RlbCB1c2luZyBwYXJ0IG9mIHRoZSBkYXRhLCBhbmQgdGhlbiBleGFtaW5lIGhvdyB3ZWxsIGl0IHByZWRpY3RzIHRoZSBoZWxkIG91dCBkYXRhLiBUaGlzIHdpbGwgeWllbGQgYSBtb3JlIHJlYWxpc3RpYyBlcnJvciByYXRlLCBpbiB0aGUgc2Vuc2UgdGhhdCBpbiBwcmFjdGljZSB3ZSB3aWxsIGJlIGludGVyZXN0ZWQgaW4gb3VyIG1vZGVsJ3MgcGVyZm9ybWFuY2Ugbm90IG9uIHRoZSBkYXRhIHRoYXQgd2UgdXNlZCB0byBmaXQgdGhlIG1vZGVsLCBidXQgcmF0aGVyIG9uIGRheXMgaW4gdGhlIGZ1dHVyZSBmb3Igd2hpY2ggdGhlIG1hcmtldCdzIG1vdmVtZW50cyBhcmUgdW5rbm93bi4NCg0KVG8gaW1wbGVtZW50IHRoaXMgc3RyYXRlZ3ksIHdlIHdpbGwgZmlyc3QgY3JlYXRlIGEgdmVjdG9yIGNvcnJlc3BvbmRpbmcgdG8gdGhlIG9ic2VydmF0aW9ucyBmcm9tIDIwMDEgdGhyb3VnaCAyMDA0LiBXZSB3aWxsIHRoZW4gdXNlIHRoaXMgdmVjdG9yIHRvIGNyZWF0ZSBhIGhlbGQgb3V0IGRhdGEgc2V0IG9mIG9ic2VydmF0aW9ucyBmcm9tIDIwMDUuDQoNCmBgYHtyfQ0KdHJhaW49KFllYXI8MjAwNSkNClNtYXJrZXQuMjAwNT0gU21hcmtldFshdHJhaW4gLF0NCkRpcmVjdGlvbi4yMDA1PSBEaXJlY3Rpb25bIXRyYWluXQ0KZGltKFNtYXJrZXQuMjAwNSkNCkRpcmVjdGlvbi4yMDA1PSBEaXJlY3Rpb25bIXRyYWluXQ0KYGBgDQoNClRoZSBvYmplY3QgYHRyYWluYCBpcyBhIHZlY3RvciBvZiAxLDI1MCBlbGVtZW50cywgY29ycmVzcG9uZGluZyB0byB0aGUgb2JzZXJ2YXRpb25zIGluIG91ciBkYXRhIHNldC4gVGhlIGVsZW1lbnRzIG9mIHRoZSB2ZWN0b3IgdGhhdCBjb3JyZXNwb25kIHRvIG9ic2VydmF0aW9ucyB0aGF0IG9jY3VycmVkIGJlZm9yZSAyMDA1IGFyZSBzZXQgdG8gYFRSVUVgLCB3aGVyZWFzIHRob3NlIHRoYXQgY29ycmVzcG9uZCB0byBvYnNlcnZhdGlvbnMgaW4gMjAwNSBhcmUgc2V0IHRvIGBGQUxTRWAuIFRoZSBvYmplY3QgYHRyYWluYCBpcyBhIEJvb2xlYW4gdmVjdG9yLCBzaW5jZSBpdHMgZWxlbWVudHMgYXJlIGBUUlVFYCBhbmQgYEZBTFNFYC4gQm9vbGVhbiB2ZWN0b3JzIGNhbiBiZSB1c2VkIHRvIG9idGFpbiBhIHN1YnNldCBvZiB0aGUgcm93cyBvciBjb2x1bW5zIG9mIGEgbWF0cml4LiBGb3IgaW5zdGFuY2UsIHRoZSBjb21tYW5kIGBgU21hcmtldGBbdHJhaW4sXWAgd291bGQgcGljayBvdXQgYSBzdWJtYXRyaXggb2YgdGhlIHN0b2NrIG1hcmtldCBkYXRhIHNldCwgY29ycmVzcG9uZGluZyBvbmx5IHRvIHRoZSBkYXRlcyBiZWZvcmUgMjAwNSwgc2luY2UgdGhvc2UgYXJlIHRoZSBvbmVzIGZvciB3aGljaCB0aGUgZWxlbWVudHMgb2YgYHRyYWluYCBhcmUgYFRSVUVgLiBUaGUgYCFgIHN5bWJvbCBjYW4gYmUgdXNlZCB0byByZXZlcnNlIGFsbCBvZiB0aGUgZWxlbWVudHMgb2YgYSBCb29sZWFuIHZlY3Rvci4gVGhhdCBpcywgYCF0cmFpbmAgaXMgYSB2ZWN0b3Igc2ltaWxhciB0byBgdHJhaW5gLCBleGNlcHQgdGhhdCB0aGUgZWxlbWVudHMgdGhhdCBhcmUgYFRSVUVgIGluIGB0cmFpbmAgZ2V0IHN3YXBwZWQgdG8gYEZBTFNFYCBpbiBgIXRyYWluYCwgYW5kIHRoZSBlbGVtZW50cyB0aGF0IGFyZSBgRkFMU0VgIGluIGB0cmFpbmAgZ2V0IHN3YXBwZWQgdG8gYFRSVUVgIGluIGAhdHJhaW5gLiBUaGVyZWZvcmUsIGBgU21hcmtldGBbIXRyYWluLF1gIHlpZWxkcyBhIHN1Ym1hdHJpeCBvZiB0aGUgc3RvY2sgbWFya2V0IGRhdGEgY29udGFpbmluZyBvbmx5IHRoZSBvYnNlcnZhdGlvbnMgZm9yIHdoaWNoIHRyYWluIGlzIGBGQUxTRWAtdGhhdCBpcywgdGhlIG9ic2VydmF0aW9ucyB3aXRoIGRhdGVzIGluIDIwMDUuIFRoZSBvdXRwdXQgYWJvdmUgaW5kaWNhdGVzIHRoYXQgdGhlcmUgYXJlIDI1MiBzdWNoIG9ic2VydmF0aW9ucy4NCg0KV2Ugbm93IGZpdCBhIGxvZ2lzdGljIHJlZ3Jlc3Npb24gbW9kZWwgdXNpbmcgb25seSB0aGUgYHN1YnNldGAgb2YgdGhlIG9ic2VydmF0aW9ucyB0aGF0IGNvcnJlc3BvbmQgdG8gZGF0ZXMgYmVmb3JlIDIwMDUsIHVzaW5nIHRoZSBzdWJzZXQgYXJndW1lbnQuIFdlIHRoZW4gb2J0YWluIHByZWRpY3RlZCBwcm9iYWJpbGl0aWVzIG9mIHRoZSBzdG9jayBtYXJrZXQgZ29pbmcgdXAgZm9yIGVhY2ggb2YgdGhlIGRheXMgaW4gb3VyIHRlc3Qgc2V0LXRoYXQgaXMsIGZvciB0aGUgZGF5cyBpbiAyMDA1Lg0KDQpgYGB7cn0NCmdsbS5maXRzPWdsbShEaXJlY3Rpb25+TGFnMStMYWcyK0xhZzMrTGFnNCtMYWc1K1ZvbHVtZSwgZGF0YT1TbWFya2V0LGZhbWlseT1iaW5vbWlhbCAsc3Vic2V0PXRyYWluKQ0KZ2xtLnByb2JzPXByZWRpY3QgKGdsbS5maXRzLFNtYXJrZXQuMjAwNSwgdHlwZT0icmVzcG9uc2UiKQ0KYGBgDQoNCk5vdGljZSB0aGF0IHdlIGhhdmUgdHJhaW5lZCBhbmQgdGVzdGVkIG91ciBtb2RlbCBvbiB0d28gY29tcGxldGVseSBzZXBhcmF0ZSBkYXRhIHNldHM6IHRyYWluaW5nIHdhcyBwZXJmb3JtZWQgdXNpbmcgb25seSB0aGUgZGF0ZXMgYmVmb3JlIDIwMDUsIGFuZCB0ZXN0aW5nIHdhcyBwZXJmb3JtZWQgdXNpbmcgb25seSB0aGUgZGF0ZXMgaW4gMjAwNS4gRmluYWxseSwgd2UgY29tcHV0ZSB0aGUgcHJlZGljdGlvbnMgZm9yIDIwMDUgYW5kIGNvbXBhcmUgdGhlbSB0byB0aGUgYWN0dWFsIG1vdmVtZW50cyBvZiB0aGUgbWFya2V0IG92ZXIgdGhhdCB0aW1lIHBlcmlvZC4NCg0KYGBge3J9DQpnbG0ucHJlZD1yZXAoIkRvd24iLDI1MikNCmdsbS5wcmVkW2dsbS5wcm9icyA+LjVdPSIgVXAiDQp0YWJsZShnbG0ucHJlZCAsRGlyZWN0aW9uLjIwMDUpDQptZWFuKGdsbS5wcmVkPT1EaXJlY3Rpb24uMjAwNSkNCm1lYW4oZ2xtLnByZWQhPURpcmVjdGlvbi4yMDA1KQ0KYGBgDQoNClRoZSBgIT1gIG5vdGF0aW9uIG1lYW5zICpub3QgZXF1YWwgdG8qLCBhbmQgc28gdGhlIGxhc3QgY29tbWFuZCBjb21wdXRlcyB0aGUgdGVzdCBzZXQgZXJyb3IgcmF0ZS4gVGhlIHJlc3VsdHMgYXJlIHJhdGhlciBkaXNhcHBvaW50aW5nOiB0aGUgdGVzdCBlcnJvciByYXRlIGlzIDUyICUsIHdoaWNoIGlzIHdvcnNlIHRoYW4gcmFuZG9tIGd1ZXNzaW5nISBPZiBjb3Vyc2UgdGhpcyByZXN1bHQgaXMgbm90IGFsbCB0aGF0IHN1cnByaXNpbmcsIGdpdmVuIHRoYXQgb25lIHdvdWxkIG5vdCBnZW5lcmFsbHkgZXhwZWN0IHRvIGJlIGFibGUgdG8gdXNlIHByZXZpb3VzIGRheXMnIHJldHVybnMgdG8gcHJlZGljdCBmdXR1cmUgbWFya2V0IHBlcmZvcm1hbmNlLiAoQWZ0ZXIgYWxsLCBpZiBpdCB3ZXJlIHBvc3NpYmxlIHRvIGRvIHNvLCB0aGVuIHRoZSBhdXRob3JzIG9mIHRoaXMgYm9vayB3b3VsZCBiZSBvdXQgc3RyaWtpbmcgaXQgcmljaCByYXRoZXIgdGhhbiB3cml0aW5nIGEgc3RhdGlzdGljcyB0ZXh0Ym9vay4pDQoNCldlIHJlY2FsbCB0aGF0IHRoZSBsb2dpc3RpYyByZWdyZXNzaW9uIG1vZGVsIGhhZCB2ZXJ5IHVuZGVyd2hlbG1pbmcgcHZhbHVlcyBhc3NvY2lhdGVkIHdpdGggYWxsIG9mIHRoZSBwcmVkaWN0b3JzLCBhbmQgdGhhdCB0aGUgc21hbGxlc3QgcC12YWx1ZSwgdGhvdWdoIG5vdCB2ZXJ5IHNtYWxsLCBjb3JyZXNwb25kZWQgdG8gYExhZzFgLiBQZXJoYXBzIGJ5IHJlbW92aW5nIHRoZSB2YXJpYWJsZXMgdGhhdCBhcHBlYXIgbm90IHRvIGJlIGhlbHBmdWwgaW4gcHJlZGljdGluZyBgRGlyZWN0aW9uYCwgd2UgY2FuIG9idGFpbiBhIG1vcmUgZWZmZWN0aXZlIG1vZGVsLiBBZnRlciBhbGwsIHVzaW5nIHByZWRpY3RvcnMgdGhhdCBoYXZlIG5vIHJlbGF0aW9uc2hpcCB3aXRoIHRoZSByZXNwb25zZSB0ZW5kcyB0byBjYXVzZSBhIGRldGVyaW9yYXRpb24gaW4gdGhlIHRlc3QgZXJyb3IgcmF0ZSAoc2luY2Ugc3VjaCBwcmVkaWN0b3JzIGNhdXNlIGFuIGluY3JlYXNlIGluIHZhcmlhbmNlIHdpdGhvdXQgYSBjb3JyZXNwb25kaW5nIGRlY3JlYXNlIGluIGJpYXMpLCBhbmQgc28gcmVtb3Zpbmcgc3VjaCBwcmVkaWN0b3JzIG1heSBpbiB0dXJuIHlpZWxkIGFuIGltcHJvdmVtZW50LiBCZWxvdyB3ZSBoYXZlIHJlZml0IHRoZSBsb2dpc3RpYyByZWdyZXNzaW9uIHVzaW5nIGp1c3QgYExhZzFgIGFuZCBgTGFnMmAsIHdoaWNoIHNlZW1lZCB0byBoYXZlIHRoZSBoaWdoZXN0IHByZWRpY3RpdmUgcG93ZXIgaW4gdGhlIG9yaWdpbmFsIGxvZ2lzdGljIHJlZ3Jlc3Npb24gbW9kZWwuDQoNCmBgYHtyfQ0KZ2xtLmZpdHM9Z2xtKERpcmVjdGlvbn5MYWcxK0xhZzIgLGRhdGE9U21hcmtldCAsZmFtaWx5PWJpbm9taWFsLHN1YnNldD10cmFpbikNCmdsbS5wcm9icz1wcmVkaWN0KGdsbS5maXRzLFNtYXJrZXQuMjAwNSwgdHlwZT0icmVzcG9uc2UiKQ0KZ2xtLnByZWQ9cmVwKCJEb3duIiwyNTIpDQpnbG0ucHJlZFtnbG0ucHJvYnMgPi41XT0iIFVwIg0KdGFibGUoZ2xtLnByZWQgLERpcmVjdGlvbi4yMDA1KQ0KbWVhbihnbG0ucHJlZD09RGlyZWN0aW9uLjIwMDUpDQoxMDYvKDEwNis3NikNCmBgYA0KDQpOb3cgdGhlIHJlc3VsdHMgYXBwZWFyIHRvIGJlIGEgbGl0dGxlIGJldHRlcjogNTYlIG9mIHRoZSBkYWlseSBtb3ZlbWVudHMgaGF2ZSBiZWVuIGNvcnJlY3RseSBwcmVkaWN0ZWQuIEl0IGlzIHdvcnRoIG5vdGluZyB0aGF0IGluIHRoaXMgY2FzZSwgYSBtdWNoIHNpbXBsZXIgc3RyYXRlZ3kgb2YgcHJlZGljdGluZyB0aGF0IHRoZSBtYXJrZXQgd2lsbCBpbmNyZWFzZSBldmVyeSBkYXkgd2lsbCBhbHNvIGJlIGNvcnJlY3QgNTYlIG9mIHRoZSB0aW1lISBIZW5jZSwgaW4gdGVybXMgb2Ygb3ZlcmFsbCBlcnJvciByYXRlLCB0aGUgbG9naXN0aWMgcmVncmVzc2lvbiBtZXRob2QgaXMgbm8gYmV0dGVyIHRoYW4gdGhlIG5hP2l2ZSBhcHByb2FjaC4gSG93ZXZlciwgdGhlIGNvbmZ1c2lvbiBtYXRyaXggc2hvd3MgdGhhdCBvbiBkYXlzIHdoZW4gbG9naXN0aWMgcmVncmVzc2lvbiBwcmVkaWN0cyBhbiBpbmNyZWFzZSBpbiB0aGUgbWFya2V0LCBpdCBoYXMgYSA1OCUgYWNjdXJhY3kgcmF0ZS4gVGhpcyBzdWdnZXN0cyBhIHBvc3NpYmxlIHRyYWRpbmcgc3RyYXRlZ3kgb2YgYnV5aW5nIG9uIGRheXMgd2hlbiB0aGUgbW9kZWwgcHJlZGljdHMgYW4gaW5jcmVhc2luZyBtYXJrZXQsIGFuZCBhdm9pZGluZyB0cmFkZXMgb24gZGF5cyB3aGVuIGEgZGVjcmVhc2UgaXMgcHJlZGljdGVkLiBPZiBjb3Vyc2Ugb25lIHdvdWxkIG5lZWQgdG8gaW52ZXN0aWdhdGUgbW9yZSBjYXJlZnVsbHkgd2hldGhlciB0aGlzIHNtYWxsIGltcHJvdmVtZW50IHdhcyByZWFsIG9yIGp1c3QgZHVlIHRvIHJhbmRvbSBjaGFuY2UuDQoNClN1cHBvc2UgdGhhdCB3ZSB3YW50IHRvIHByZWRpY3QgdGhlIHJldHVybnMgYXNzb2NpYXRlZCB3aXRoIHBhcnRpY3VsYXIgdmFsdWVzIG9mIGBMYWcxYCBhbmQgYExhZzJgLiBJbiBwYXJ0aWN1bGFyLCB3ZSB3YW50IHRvIHByZWRpY3QgYERpcmVjdGlvbmAgb24gYSBkYXkgd2hlbiBgTGFnMWAgYW5kIGBMYWcyYCBlcXVhbCAxLjIgYW5kIDEuMSwgcmVzcGVjdGl2ZWx5LCBhbmQgb24gYSBkYXkgd2hlbiB0aGV5IGVxdWFsIDEuNSBhbmQgPz8/MC44LiBXZSBkbyB0aGlzIHVzaW5nIHRoZSBgcHJlZGljdCgpYCBmdW5jdGlvbi4NCg0KYGBge3J9DQpwcmVkaWN0KGdsbS5maXRzLG5ld2RhdGEgPWRhdGEuZnJhbWUoTGFnMT1jKDEuMiAsMS41KSxMYWcyPWMoMS4xLC0wLjgpICksdHlwZT0icmVzcG9uc2UiKQ0KYGBgDQoNCiMjIyBMaW5lYXIgRGlzY3JpbWluYW50IEFuYWx5c2lzDQpOb3cgd2Ugd2lsbCBwZXJmb3JtIExEQSBvbiB0aGUgYFNtYXJrZXRgIGRhdGEuIEluIGBSYCwgd2UgZml0IGFuIExEQSBtb2RlbCB1c2luZyB0aGUgYGxkYSgpYCBmdW5jdGlvbiwgd2hpY2ggaXMgcGFydCBvZiB0aGUgYE1BU1NgIGxpYnJhcnkuIE5vdGljZSB0aGF0IHRoZSBgbGRhKClgIHN5bnRheCBmb3IgdGhlIGZ1bmN0aW9uIGlzIGlkZW50aWNhbCB0byB0aGF0IG9mIGBsbSgpYCwgYW5kIHRvIHRoYXQgb2YgYGdsbSgpYCBleGNlcHQgZm9yIHRoZSBhYnNlbmNlIG9mIHRoZSBmYW1pbHkgb3B0aW9uLiBXZSBmaXQgdGhlIG1vZGVsIHVzaW5nIG9ubHkgdGhlIG9ic2VydmF0aW9ucyBiZWZvcmUgMjAwNS4NCg0KYGBge3J9DQpsaWJyYXJ5KE1BU1MpDQpsZGEuZml0PWxkYShEaXJlY3Rpb25+TGFnMStMYWcyICxkYXRhPWBTbWFya2V0YCAsc3Vic2V0PXRyYWluKQ0KbGRhLmZpdA0KcGxvdChsZGEuZml0KQ0KYGBgDQoNClRoZSBMREEgb3V0cHV0IGluZGljYXRlcyB0aGF0ICRcaGF0e1xwaX1fMSQgPSAwLjQ5MiBhbmQgJFxoYXR7XHBpfV8yJCA9IDAuNTA4OyBpbiBvdGhlciB3b3JkcywgNDkuMiAlIG9mIHRoZSB0cmFpbmluZyBvYnNlcnZhdGlvbnMgY29ycmVzcG9uZCB0byBkYXlzIGR1cmluZyB3aGljaCB0aGUgbWFya2V0IHdlbnQgZG93bi4gSXQgYWxzbyBwcm92aWRlcyB0aGUgZ3JvdXAgbWVhbnM7IHRoZXNlIGFyZSB0aGUgYXZlcmFnZSBvZiBlYWNoIHByZWRpY3RvciB3aXRoaW4gZWFjaCBjbGFzcywgYW5kIGFyZSB1c2VkIGJ5IExEQSBhcyBlc3RpbWF0ZXMgb2YgJFxtdV9rJC4gVGhlc2Ugc3VnZ2VzdCB0aGF0IHRoZXJlIGlzIGEgdGVuZGVuY3kgZm9yIHRoZSBwcmV2aW91cyAyIGRheXMnIHJldHVybnMgdG8gYmUgbmVnYXRpdmUgb24gZGF5cyB3aGVuIHRoZSBtYXJrZXQgaW5jcmVhc2VzLCBhbmQgYSB0ZW5kZW5jeSBmb3IgdGhlIHByZXZpb3VzIGRheXMnIHJldHVybnMgdG8gYmUgcG9zaXRpdmUgb24gZGF5cyB3aGVuIHRoZSBtYXJrZXQgZGVjbGluZXMuIFRoZSAqY29lZmZpY2llbnRzIG9mIGxpbmVhciBkaXNjcmltaW5hbnRzKiBvdXRwdXQgcHJvdmlkZXMgdGhlIGxpbmVhciBjb21iaW5hdGlvbiBvZiBgTGFnMWAgYW5kIGBMYWcyYCB0aGF0IGFyZSB1c2VkIHRvIGZvcm0gdGhlIExEQSBkZWNpc2lvbiBydWxlLiBJbiBvdGhlciB3b3JkcywgdGhlc2UgYXJlIHRoZSBtdWx0aXBsaWVycyBvZiB0aGUgZWxlbWVudHMgb2YgWCA9ICp4KiBpbiAoNC4xOSkuIElmID8/PzAuNjQyP2BMYWcxYD8/PzAuNTE0P2BMYWcyYCBpcyBsYXJnZSwgdGhlbiB0aGUgTERBIGNsYXNzaWZpZXIgd2lsbCBwcmVkaWN0IGEgbWFya2V0IGluY3JlYXNlLCBhbmQgaWYgaXQgaXMgc21hbGwsIHRoZW4gdGhlIExEQSBjbGFzc2lmaWVyIHdpbGwgcHJlZGljdCBhIG1hcmtldCBkZWNsaW5lLiBUaGUgcGxvdCgpIGZ1bmN0aW9uIHByb2R1Y2VzIHBsb3RzIG9mIHRoZSAqbGluZWFyIGRpc2NyaW1pbmFudHMqLCBvYnRhaW5lZCBieSBjb21wdXRpbmcgPz8/MC42NDIgPyBgTGFnMWAgPz8/IDAuNTE0ID8gYExhZzJgIGZvciBlYWNoIG9mIHRoZSB0cmFpbmluZyBvYnNlcnZhdGlvbnMuDQoNClRoZSBgcHJlZGljdCgpYCBmdW5jdGlvbiByZXR1cm5zIGEgbGlzdCB3aXRoIHRocmVlIGVsZW1lbnRzLiBUaGUgZmlyc3QgZWxlbWVudCwgYGNsYXNzYCwgY29udGFpbnMgTERBJ3MgcHJlZGljdGlvbnMgYWJvdXQgdGhlIG1vdmVtZW50IG9mIHRoZSBtYXJrZXQuIFRoZSBzZWNvbmQgZWxlbWVudCwgYHBvc3RlcmlvcmAsIGlzIGEgbWF0cml4IHdob3NlICprKnRoIGNvbHVtbiBjb250YWlucyB0aGUgcG9zdGVyaW9yIHByb2JhYmlsaXR5IHRoYXQgdGhlIGNvcnJlc3BvbmRpbmcgb2JzZXJ2YXRpb24gYmVsb25ncyB0byB0aGUgKmsqdGggY2xhc3MsIGNvbXB1dGVkIGZyb20gKDQuMTApLiBGaW5hbGx5LCBgeGAgY29udGFpbnMgdGhlIGxpbmVhciBkaXNjcmltaW5hbnRzLCBkZXNjcmliZWQgZWFybGllci4NCg0KYGBge3J9DQpsZGEucHJlZD1wcmVkaWN0IChsZGEuZml0ICwgU21hcmtldC4yMDA1KQ0KbmFtZXMobGRhLnByZWQpDQpgYGANCg0KQXMgd2Ugb2JzZXJ2ZWQgaW4gU2VjdGlvbiA0LjUsIHRoZSBMREEgYW5kIGxvZ2lzdGljIHJlZ3Jlc3Npb24gcHJlZGljdGlvbnMgYXJlIGFsbW9zdCBpZGVudGljYWwuDQpgYGB7cn0NCmxkYS5jbGFzcz1sZGEucHJlZCRjbGFzcw0KdGFibGUobGRhLmNsYXNzICxEaXJlY3Rpb24uMjAwNSkNCm1lYW4obGRhLmNsYXNzPT1EaXJlY3Rpb24uMjAwNSkNCmBgYA0KDQpBcHBseWluZyBhIDUwICUgdGhyZXNob2xkIHRvIHRoZSBwb3N0ZXJpb3IgcHJvYmFiaWxpdGllcyBhbGxvd3MgdXMgdG8gcmVjcmVhdGUgdGhlIHByZWRpY3Rpb25zIGNvbnRhaW5lZCBpbiBgbGRhLnByZWQkY2xhc3NgLg0KDQpgYGB7cn0NCnN1bShsZGEucHJlZCRwb3N0ZXJpb3JbLDFdPj0uNSkNCnN1bShsZGEucHJlZCRwb3N0ZXJpb3JbLDFdPC41KQ0KYGBgDQoNCk5vdGljZSB0aGF0IHRoZSBwb3N0ZXJpb3IgcHJvYmFiaWxpdHkgb3V0cHV0IGJ5IHRoZSBtb2RlbCBjb3JyZXNwb25kcyB0byB0aGUgcHJvYmFiaWxpdHkgdGhhdCB0aGUgbWFya2V0IHdpbGwgZGVjcmVhc2U6DQoNCmBgYHtyfQ0KbGRhLnByZWQkcG9zdGVyaW9yWzE6MjAsMV0NCmxkYS5jbGFzc1sxOjIwXQ0KYGBgDQoNCklmIHdlIHdhbnRlZCB0byB1c2UgYSBwb3N0ZXJpb3IgcHJvYmFiaWxpdHkgdGhyZXNob2xkIG90aGVyIHRoYW4gNTAgJSBpbiBvcmRlciB0byBtYWtlIHByZWRpY3Rpb25zLCB0aGVuIHdlIGNvdWxkIGVhc2lseSBkbyBzby4gRm9yIGluc3RhbmNlLCBzdXBwb3NlIHRoYXQgd2Ugd2lzaCB0byBwcmVkaWN0IGEgbWFya2V0IGRlY3JlYXNlIG9ubHkgaWYgd2UgYXJlIHZlcnkgY2VydGFpbiB0aGF0IHRoZSBtYXJrZXQgd2lsbCBpbmRlZWQgZGVjcmVhc2Ugb24gdGhhdCBkYXktc2F5LCBpZiB0aGUgcG9zdGVyaW9yIHByb2JhYmlsaXR5IGlzIGF0IGxlYXN0IDkwJS4NCg0KYGBge3J9DQpzdW0obGRhLnByZWQkcG9zdGVyaW9yWywxXT4uOSkNCmBgYA0KDQpObyBkYXlzIGluIDIwMDUgbWVldCB0aGF0IHRocmVzaG9sZCEgSW4gZmFjdCwgdGhlIGdyZWF0ZXN0IHBvc3RlcmlvciBwcm9iYWJpbGl0eSBvZiBkZWNyZWFzZSBpbiBhbGwgb2YgMjAwNSB3YXMgNTIuMDIgJS4NCg0KIyMjIFF1YWRyYXRpYyBEaXNjcmltaW5hbnQgQW5hbHlzaXMNCldlIHdpbGwgbm93IGZpdCBhIFFEQSBtb2RlbCB0byB0aGUgYFNtYXJrZXRgIGRhdGEuIFFEQSBpcyBpbXBsZW1lbnRlZCBpbiBSIHVzaW5nIHRoZSBxZGEoKSBmdW5jdGlvbiwgd2hpY2ggaXMgYWxzbyBwYXJ0IG9mIHRoZSBgTUFTU2AgbGlicmFyeS4gVGhlIHN5bnRheCBpcyBpZGVudGljYWwgdG8gdGhhdCBvZiBgbGRhKClgLg0KDQpgYGB7cn0NCnFkYS5maXQ9cWRhKERpcmVjdGlvbn5MYWcxK0xhZzIgLGRhdGE9U21hcmtldCAsc3Vic2V0PXRyYWluKQ0KcWRhLmZpdA0KYGBgDQoNClRoZSBvdXRwdXQgY29udGFpbnMgdGhlIGdyb3VwIG1lYW5zLiBCdXQgaXQgZG9lcyBub3QgY29udGFpbiB0aGUgY29lZmZpY2llbnRzIG9mIHRoZSBsaW5lYXIgZGlzY3JpbWluYW50cywgYmVjYXVzZSB0aGUgUURBIGNsYXNzaWZpZXIgaW52b2x2ZXMgYSBxdWFkcmF0aWMsIHJhdGhlciB0aGFuIGEgbGluZWFyLCBmdW5jdGlvbiBvZiB0aGUgcHJlZGljdG9ycy4gVGhlIGBwcmVkaWN0KClgIGZ1bmN0aW9uIHdvcmtzIGluIGV4YWN0bHkgdGhlIHNhbWUgZmFzaGlvbiBhcyBmb3IgTERBLg0KDQpgYGB7cn0NCnFkYS5jbGFzcz1wcmVkaWN0KHFkYS5maXQgLFNtYXJrZXQuMjAwNSkkY2xhc3MNCnRhYmxlKHFkYS5jbGFzcyAsRGlyZWN0aW9uLjIwMDUpDQptZWFuKHFkYS5jbGFzcz09RGlyZWN0aW9uLjIwMDUpDQpgYGANCg0KSW50ZXJlc3RpbmdseSwgdGhlIFFEQSBwcmVkaWN0aW9ucyBhcmUgYWNjdXJhdGUgYWxtb3N0IDYwICUgb2YgdGhlIHRpbWUsIGV2ZW4gdGhvdWdoIHRoZSAyMDA1IGRhdGEgd2FzIG5vdCB1c2VkIHRvIGZpdCB0aGUgbW9kZWwuIFRoaXMgbGV2ZWwgb2YgYWNjdXJhY3kgaXMgcXVpdGUgaW1wcmVzc2l2ZSBmb3Igc3RvY2sgbWFya2V0IGRhdGEsIHdoaWNoIGlzIGtub3duIHRvIGJlIHF1aXRlIGhhcmQgdG8gbW9kZWwgYWNjdXJhdGVseS4gVGhpcyBzdWdnZXN0cyB0aGF0IHRoZSBxdWFkcmF0aWMgZm9ybSBhc3N1bWVkIGJ5IFFEQSBtYXkgY2FwdHVyZSB0aGUgdHJ1ZSByZWxhdGlvbnNoaXAgbW9yZSBhY2N1cmF0ZWx5IHRoYW4gdGhlIGxpbmVhciBmb3JtcyBhc3N1bWVkIGJ5IExEQSBhbmQgbG9naXN0aWMgcmVncmVzc2lvbi4gSG93ZXZlciwgd2UgcmVjb21tZW5kIGV2YWx1YXRpbmcgdGhpcyBtZXRob2QncyBwZXJmb3JtYW5jZSBvbiBhIGxhcmdlciB0ZXN0IHNldCBiZWZvcmUgYmV0dGluZyB0aGF0IHRoaXMgYXBwcm9hY2ggd2lsbCBjb25zaXN0ZW50bHkgYmVhdCB0aGUgbWFya2V0IQ0KDQojIyMgSy1OZWFyZXN0IE5laWdoYm9ycw0KV2Ugd2lsbCBub3cgcGVyZm9ybSBLTk4gdXNpbmcgdGhlIGBrbm4oKWAgZnVuY3Rpb24sIHdoaWNoIGlzIHBhcnQgb2YgdGhlIGBjbGFzc2AgbGlicmFyeS4gVGhpcyBmdW5jdGlvbiB3b3JrcyByYXRoZXIgZGlmZmVyZW50bHkgZnJvbSB0aGUgb3RoZXIgbW9kZWxmaXR0aW5nIGZ1bmN0aW9ucyB0aGF0IHdlIGhhdmUgZW5jb3VudGVyZWQgdGh1cyBmYXIuIFJhdGhlciB0aGFuIGEgdHdvLXN0ZXAgYXBwcm9hY2ggaW4gd2hpY2ggd2UgZmlyc3QgZml0IHRoZSBtb2RlbCBhbmQgdGhlbiB3ZSB1c2UgdGhlIG1vZGVsIHRvIG1ha2UNCnByZWRpY3Rpb25zLCBga25uKClgIGZvcm1zIHByZWRpY3Rpb25zIHVzaW5nIGEgc2luZ2xlIGNvbW1hbmQuIFRoZSBmdW5jdGlvbg0KcmVxdWlyZXMgZm91ciBpbnB1dHMuDQoNCjEuIEEgbWF0cml4IGNvbnRhaW5pbmcgdGhlIHByZWRpY3RvcnMgYXNzb2NpYXRlZCB3aXRoIHRoZSB0cmFpbmluZyBkYXRhLCBsYWJlbGVkIGB0cmFpbi5YYCBiZWxvdy4NCjIuIEEgbWF0cml4IGNvbnRhaW5pbmcgdGhlIHByZWRpY3RvcnMgYXNzb2NpYXRlZCB3aXRoIHRoZSBkYXRhIGZvciB3aGljaCB3ZSB3aXNoIHRvIG1ha2UgcHJlZGljdGlvbnMsIGxhYmVsZWQgYHRlc3QuWGAgYmVsb3cuDQozLiBBIHZlY3RvciBjb250YWluaW5nIHRoZSBjbGFzcyBsYWJlbHMgZm9yIHRoZSB0cmFpbmluZyBvYnNlcnZhdGlvbnMsIGxhYmVsZWQgYHRyYWluLkRpcmVjdGlvbmAgYmVsb3cuDQo0LiBBIHZhbHVlIGZvciAqSyosIHRoZSBudW1iZXIgb2YgbmVhcmVzdCBuZWlnaGJvcnMgdG8gYmUgdXNlZCBieSB0aGUgY2xhc3NpZmllci4NCg0KV2UgdXNlIHRoZSBgY2JpbmQoKWAgZnVuY3Rpb24sIHNob3J0IGZvciAqY29sdW1uIGJpbmQqLCB0byBiaW5kIHRoZSBgTGFnMWAgYW5kIGBMYWcyYCB2YXJpYWJsZXMgdG9nZXRoZXIgaW50byB0d28gbWF0cmljZXMsIG9uZSBmb3IgdGhlIHRyYWluaW5nIHNldCBhbmQgdGhlIG90aGVyIGZvciB0aGUgdGVzdCBzZXQuDQoNCmBgYHtyfQ0KbGlicmFyeShjbGFzcykNCnRyYWluLlg9Y2JpbmQoTGFnMSAsTGFnMilbdHJhaW4gLF0NCnRlc3QuWD1jYmluZChMYWcxICxMYWcyKVshdHJhaW4gLF0NCnRyYWluLkRpcmVjdGlvbiA9RGlyZWN0aW9uIFt0cmFpbl0NCmBgYA0KDQpOb3cgdGhlIGBrbm4oKWAgZnVuY3Rpb24gY2FuIGJlIHVzZWQgdG8gcHJlZGljdCB0aGUgbWFya2V0J3MgbW92ZW1lbnQgZm9yIHRoZSBkYXRlcyBpbiAyMDA1LiBXZSBzZXQgYSByYW5kb20gc2VlZCBiZWZvcmUgd2UgYXBwbHkgYGtubigpYCBiZWNhdXNlIGlmIHNldmVyYWwgb2JzZXJ2YXRpb25zIGFyZSB0aWVkIGFzIG5lYXJlc3QgbmVpZ2hib3JzLCB0aGVuIGBSYCB3aWxsIHJhbmRvbWx5IGJyZWFrIHRoZSB0aWUuIFRoZXJlZm9yZSwgYSBzZWVkIG11c3QgYmUgc2V0IGluIG9yZGVyIHRvIGVuc3VyZSByZXByb2R1Y2liaWxpdHkgb2YgcmVzdWx0cy4NCg0KYGBge3J9DQpzZXQuc2VlZCgxKQ0Ka25uLnByZWQ9a25uKHRyYWluLlgsdGVzdC5YLHRyYWluLkRpcmVjdGlvbiAsaz0xKQ0KdGFibGUoa25uLnByZWQgLERpcmVjdGlvbi4yMDA1KQ0KKDgzKzQzKSAvMjUyDQpgYGANCg0KVGhlIHJlc3VsdHMgdXNpbmcgSyA9IDEgYXJlIG5vdCB2ZXJ5IGdvb2QsIHNpbmNlIG9ubHkgNTAgJSBvZiB0aGUgb2JzZXJ2YXRpb25zIGFyZSBjb3JyZWN0bHkgcHJlZGljdGVkLiBPZiBjb3Vyc2UsIGl0IG1heSBiZSB0aGF0IEsgPSAxIHJlc3VsdHMgaW4gYW4gb3Zlcmx5IGZsZXhpYmxlIGZpdCB0byB0aGUgZGF0YS4gQmVsb3csIHdlIHJlcGVhdCB0aGUgYW5hbHlzaXMgdXNpbmcgSyA9IDMuDQoNCmBgYHtyfQ0Ka25uLnByZWQ9a25uKHRyYWluLlgsdGVzdC5YLHRyYWluLkRpcmVjdGlvbiAsaz0zKQ0KdGFibGUoa25uLnByZWQgLERpcmVjdGlvbi4yMDA1KQ0KbWVhbihrbm4ucHJlZD09RGlyZWN0aW9uLjIwMDUpDQpgYGANCg0KVGhlIHJlc3VsdHMgaGF2ZSBpbXByb3ZlZCBzbGlnaHRseS4gQnV0IGluY3JlYXNpbmcgSyBmdXJ0aGVyIHR1cm5zIG91dCB0byBwcm92aWRlIG5vIGZ1cnRoZXIgaW1wcm92ZW1lbnRzLiBJdCBhcHBlYXJzIHRoYXQgZm9yIHRoaXMgZGF0YSwgUURBIHByb3ZpZGVzIHRoZSBiZXN0IHJlc3VsdHMgb2YgdGhlIG1ldGhvZHMgdGhhdCB3ZSBoYXZlIGV4YW1pbmVkIHNvIGZhci4NCg0KDQojIyMgQW4gQXBwbGljYXRpb24gdG8gQ2FyYXZhbiBJbnN1cmFuY2UgRGF0YQ0KRmluYWxseSwgd2Ugd2lsbCBhcHBseSB0aGUgS05OIGFwcHJvYWNoIHRvIHRoZSBgQ2FyYXZhbmAgZGF0YSBzZXQsIHdoaWNoIGlzIHBhcnQgb2YgdGhlIGBJU0xSYCBsaWJyYXJ5LiBUaGlzIGRhdGEgc2V0IGluY2x1ZGVzIDg1IHByZWRpY3RvcnMgdGhhdCBtZWFzdXJlIGRlbW9ncmFwaGljIGNoYXJhY3RlcmlzdGljcyBmb3IgNSw4MjIgaW5kaXZpZHVhbHMuIFRoZSByZXNwb25zZSB2YXJpYWJsZSBpcyBgUHVyY2hhc2VgLCB3aGljaCBpbmRpY2F0ZXMgd2hldGhlciBvciBub3QgYSBnaXZlbiBpbmRpdmlkdWFsIHB1cmNoYXNlcyBhIGNhcmF2YW4gaW5zdXJhbmNlIHBvbGljeS4gSW4gdGhpcyBkYXRhIHNldCwgb25seSA2ICUgb2YgcGVvcGxlIHB1cmNoYXNlZCBjYXJhdmFuIGluc3VyYW5jZS4NCg0KYGBge3J9DQphdHRhY2goQ2FyYXZhbikNCmRpbShDYXJhdmFuKQ0Kc3VtbWFyeShQdXJjaGFzZSkNCjM0OC81ODIyDQpgYGANCg0KQmVjYXVzZSB0aGUgS05OIGNsYXNzaWZpZXIgcHJlZGljdHMgdGhlIGNsYXNzIG9mIGEgZ2l2ZW4gdGVzdCBvYnNlcnZhdGlvbiBieSBpZGVudGlmeWluZyB0aGUgb2JzZXJ2YXRpb25zIHRoYXQgYXJlIG5lYXJlc3QgdG8gaXQsIHRoZSBzY2FsZSBvZiB0aGUgdmFyaWFibGVzIG1hdHRlcnMuIEFueSB2YXJpYWJsZXMgdGhhdCBhcmUgb24gYSBsYXJnZSBzY2FsZSB3aWxsIGhhdmUgYSBtdWNoIGxhcmdlciBlZmZlY3Qgb24gdGhlICpkaXN0YW5jZSogYmV0d2VlbiB0aGUgb2JzZXJ2YXRpb25zLCBhbmQgaGVuY2Ugb24gdGhlIEtOTiBjbGFzc2lmaWVyLCB0aGFuIHZhcmlhYmxlcyB0aGF0IGFyZSBvbiBhIHNtYWxsIHNjYWxlLiBGb3IgaW5zdGFuY2UsIGltYWdpbmUgYSBkYXRhIHNldCB0aGF0IGNvbnRhaW5zIHR3byB2YXJpYWJsZXMsIGBzYWxhcnlgIGFuZCBgYWdlYCAobWVhc3VyZWQgaW4gZG9sbGFycyBhbmQgeWVhcnMsIHJlc3BlY3RpdmVseSkuIEFzIGZhciBhcyBLTk4gaXMgY29uY2VybmVkLCBhIGRpZmZlcmVuY2Ugb2YgXCQxLDAwMCBpbiBgc2FsYXJ5YCBpcyBlbm9ybW91cyBjb21wYXJlZCB0byBhIGRpZmZlcmVuY2Ugb2YgNTAgeWVhcnMgaW4gYGFnZWAuIENvbnNlcXVlbnRseSwgYHNhbGFyeWAgd2lsbCBkcml2ZSB0aGUgS05OIGNsYXNzaWZpY2F0aW9uIHJlc3VsdHMsIGFuZCBgYWdlYCB3aWxsIGhhdmUgYWxtb3N0IG5vIGVmZmVjdC4gVGhpcyBpcyBjb250cmFyeSB0byBvdXIgaW50dWl0aW9uIHRoYXQgYSBgc2FsYXJ5YCBkaWZmZXJlbmNlIG9mIFwkMSwwMDAgaXMgcXVpdGUgc21hbGwgY29tcGFyZWQgdG8gYW4gYGFnZWAgZGlmZmVyZW5jZSBvZiA1MCB5ZWFycy4gRnVydGhlcm1vcmUsIHRoZSBpbXBvcnRhbmNlIG9mIHNjYWxlIHRvIHRoZSBLTk4gY2xhc3NpZmllciBsZWFkcyB0byBhbm90aGVyIGlzc3VlOiBpZiB3ZSBtZWFzdXJlZCBgc2FsYXJ5YCBpbiBKYXBhbmVzZSB5ZW4sIG9yIGlmIHdlIG1lYXN1cmVkIGBhZ2VgIGluIG1pbnV0ZXMsIHRoZW4gd2UnZCBnZXQgcXVpdGUgZGlmZmVyZW50IGNsYXNzaWZpY2F0aW9uIHJlc3VsdHMgZnJvbSB3aGF0IHdlIGdldCBpZiB0aGVzZSB0d28gdmFyaWFibGVzIGFyZSBtZWFzdXJlZCBpbiBkb2xsYXJzIGFuZCB5ZWFycy4NCg0KQSBnb29kIHdheSB0byBoYW5kbGUgdGhpcyBwcm9ibGVtIGlzIHRvICpzdGFuZGFyZGl6ZSogdGhlIGRhdGEgc28gdGhhdCBhbGwgdmFyaWFibGVzIGFyZSBnaXZlbiBhIG1lYW4gb2YgemVybyBhbmQgYSBzdGFuZGFyZCBkZXZpYXRpb24gb2Ygb25lLiBUaGVuIGFsbCB2YXJpYWJsZXMgd2lsbCBiZSBvbiBhIGNvbXBhcmFibGUgc2NhbGUuIFRoZSBzY2FsZSgpIGZ1bmN0aW9uIGRvZXMganVzdCBgc2NhbGUoKWAgdGhpcy4gSW4gc3RhbmRhcmRpemluZyB0aGUgZGF0YSwgd2UgZXhjbHVkZSBjb2x1bW4gODYsIGJlY2F1c2UgdGhhdCBpcyB0aGUNCnF1YWxpdGF0aXZlIGBQdXJjaGFzZWAgdmFyaWFibGUuDQoNCmBgYHtyfQ0Kc3RhbmRhcmRpemVkLlg9IHNjYWxlKENhcmF2YW4gWywtODZdKQ0KdmFyKENhcmF2YW5bICwxXSkNCnZhcihDYXJhdmFuWyAsMl0pDQp2YXIoc3RhbmRhcmRpemVkLlhbLDFdKQ0KdmFyKHN0YW5kYXJkaXplZC5YWywyXSkNCmBgYA0KDQpOb3cgZXZlcnkgY29sdW1uIG9mIGBzdGFuZGFyZGl6ZWQuWGAgaGFzIGEgc3RhbmRhcmQgZGV2aWF0aW9uIG9mIG9uZSBhbmQgYSBtZWFuIG9mIHplcm8uDQoNCldlIG5vdyBzcGxpdCB0aGUgb2JzZXJ2YXRpb25zIGludG8gYSB0ZXN0IHNldCwgY29udGFpbmluZyB0aGUgZmlyc3QgMSwwMDAgb2JzZXJ2YXRpb25zLCBhbmQgYSB0cmFpbmluZyBzZXQsIGNvbnRhaW5pbmcgdGhlIHJlbWFpbmluZyBvYnNlcnZhdGlvbnMuIFdlIGZpdCBhIEtOTiBtb2RlbCBvbiB0aGUgdHJhaW5pbmcgZGF0YSB1c2luZyBLID0gMSwgYW5kIGV2YWx1YXRlIGl0cyBwZXJmb3JtYW5jZSBvbiB0aGUgdGVzdCBkYXRhLg0KDQpgYGB7cn0NCnRlc3Q9MToxMDAwDQp0cmFpbi5YPSBzdGFuZGFyZGl6ZWQuWFstdGVzdCAsXQ0KdGVzdC5YPSBzdGFuZGFyZGl6ZWQuWFt0ZXN0ICxdDQp0cmFpbi5ZPVB1cmNoYXNlIFstdGVzdF0NCnRlc3QuWT1QdXJjaGFzZSBbdGVzdF0NCnNldC5zZWVkKDEpDQprbm4ucHJlZD1rbm4odHJhaW4uWCx0ZXN0LlgsdHJhaW4uWSxrPTEpDQptZWFuKHRlc3QuWSE9a25uLnByZWQpDQptZWFuKHRlc3QuWSE9Ik5vIikNCmBgYA0KDQpUaGUgdmVjdG9yIGB0ZXN0YCBpcyBudW1lcmljLCB3aXRoIHZhbHVlcyBmcm9tIDEgdGhyb3VnaCAxLCAwMDAuIFR5cGluZyBgc3RhbmRhcmRpemVkLlhbdGVzdCxdYCB5aWVsZHMgdGhlIHN1Ym1hdHJpeCBvZiB0aGUgZGF0YSBjb250YWluaW5nIHRoZSBvYnNlcnZhdGlvbnMgd2hvc2UgaW5kaWNlcyByYW5nZSBmcm9tIDEgdG8gMSwgMDAwLCB3aGVyZWFzIHR5cGluZyBgc3RhbmRhcmRpemVkLlhbLXRlc3QsXWAgeWllbGRzIHRoZSBzdWJtYXRyaXggY29udGFpbmluZyB0aGUgb2JzZXJ2YXRpb25zIHdob3NlIGluZGljZXMgZG8gKm5vdCogcmFuZ2UgZnJvbSAxIHRvIDEsIDAwMC4gVGhlIEtOTiBlcnJvciByYXRlIG9uIHRoZSAxLDAwMCB0ZXN0IG9ic2VydmF0aW9ucyBpcyBqdXN0IHVuZGVyIDEyICUuIEF0IGZpcnN0IGdsYW5jZSwgdGhpcyBtYXkgYXBwZWFyIHRvIGJlIGZhaXJseSBnb29kLiBIb3dldmVyLCBzaW5jZSBvbmx5IDYgJSBvZiBjdXN0b21lcnMgcHVyY2hhc2VkIGluc3VyYW5jZSwgd2UgY291bGQgZ2V0IHRoZSBlcnJvciByYXRlIGRvd24gdG8gNiAlIGJ5IGFsd2F5cyBwcmVkaWN0aW5nIGBOb2AgcmVnYXJkbGVzcyBvZiB0aGUgdmFsdWVzIG9mIHRoZSBwcmVkaWN0b3JzIQ0KDQpTdXBwb3NlIHRoYXQgdGhlcmUgaXMgc29tZSBub24tdHJpdmlhbCBjb3N0IHRvIHRyeWluZyB0byBzZWxsIGluc3VyYW5jZSB0byBhIGdpdmVuIGluZGl2aWR1YWwuIEZvciBpbnN0YW5jZSwgcGVyaGFwcyBhIHNhbGVzcGVyc29uIG11c3QgdmlzaXQgZWFjaCBwb3RlbnRpYWwgY3VzdG9tZXIuIElmIHRoZSBjb21wYW55IHRyaWVzIHRvIHNlbGwgaW5zdXJhbmNlIHRvIGEgcmFuZG9tIHNlbGVjdGlvbiBvZiBjdXN0b21lcnMsIHRoZW4gdGhlIHN1Y2Nlc3MgcmF0ZSB3aWxsIGJlIG9ubHkgNiAlLCB3aGljaCBtYXkgYmUgZmFyIHRvbyBsb3cgZ2l2ZW4gdGhlIGNvc3RzIGludm9sdmVkLiBJbnN0ZWFkLCB0aGUgY29tcGFueSB3b3VsZCBsaWtlIHRvIHRyeSB0byBzZWxsIGluc3VyYW5jZSBvbmx5IHRvIGN1c3RvbWVycyB3aG8gYXJlIGxpa2VseSB0byBidXkgaXQuIFNvIHRoZSBvdmVyYWxsIGVycm9yIHJhdGUgaXMgbm90IG9mIGludGVyZXN0LiBJbnN0ZWFkLCB0aGUgZnJhY3Rpb24gb2YgaW5kaXZpZHVhbHMgdGhhdCBhcmUgY29ycmVjdGx5IHByZWRpY3RlZCB0byBidXkgaW5zdXJhbmNlIGlzIG9mIGludGVyZXN0Lg0KDQpJdCB0dXJucyBvdXQgdGhhdCBLTk4gd2l0aCBLID0gMSBkb2VzIGZhciBiZXR0ZXIgdGhhbiByYW5kb20gZ3Vlc3NpbmcgYW1vbmcgdGhlIGN1c3RvbWVycyB0aGF0IGFyZSBwcmVkaWN0ZWQgdG8gYnV5IGluc3VyYW5jZS4gQW1vbmcgNzcgc3VjaCBjdXN0b21lcnMsIDksIG9yIDExLjcgJSwgYWN0dWFsbHkgZG8gcHVyY2hhc2UgaW5zdXJhbmNlLiBUaGlzIGlzIGRvdWJsZSB0aGUgcmF0ZSB0aGF0IG9uZSB3b3VsZCBvYnRhaW4gZnJvbSByYW5kb20gZ3Vlc3NpbmcuDQoNCmBgYHtyfQ0KdGFibGUoa25uLnByZWQgLHRlc3QuWSkNCjkvKDY4KzkpDQpgYGANCg0KVXNpbmcgSyA9IDMsIHRoZSBzdWNjZXNzIHJhdGUgaW5jcmVhc2VzIHRvIDE5ICUsIGFuZCB3aXRoIEsgPSA1IHRoZSByYXRlIGlzIDI2LjcgJS4gVGhpcyBpcyBvdmVyIGZvdXIgdGltZXMgdGhlIHJhdGUgdGhhdCByZXN1bHRzIGZyb20gcmFuZG9tIGd1ZXNzaW5nLiBJdCBhcHBlYXJzIHRoYXQgS05OIGlzIGZpbmRpbmcgc29tZSByZWFsIHBhdHRlcm5zIGluIGEgZGlmZmljdWx0IGRhdGEgc2V0ISANCg0KYGBge3J9DQprbm4ucHJlZD1rbm4odHJhaW4uWCx0ZXN0LlgsdHJhaW4uWSxrPTMpDQp0YWJsZShrbm4ucHJlZCAsdGVzdC5ZKQ0KNS8yNg0Ka25uLnByZWQ9a25uKHRyYWluLlgsdGVzdC5YLHRyYWluLlksaz01KQ0KdGFibGUoa25uLnByZWQgLHRlc3QuWSkNCjQvMTUNCmBgYA0KDQpBcyBhIGNvbXBhcmlzb24sIHdlIGNhbiBhbHNvIGZpdCBhIGxvZ2lzdGljIHJlZ3Jlc3Npb24gbW9kZWwgdG8gdGhlIGRhdGEuIElmIHdlIHVzZSAwLjUgYXMgdGhlIHByZWRpY3RlZCBwcm9iYWJpbGl0eSBjdXQtb2ZmIGZvciB0aGUgY2xhc3NpZmllciwgdGhlbiB3ZSBoYXZlIGEgcHJvYmxlbTogb25seSBzZXZlbiBvZiB0aGUgdGVzdCBvYnNlcnZhdGlvbnMgYXJlIHByZWRpY3RlZCB0byBwdXJjaGFzZSBpbnN1cmFuY2UuIEV2ZW4gd29yc2UsIHdlIGFyZSB3cm9uZyBhYm91dCBhbGwgb2YgdGhlc2UhIEhvd2V2ZXIsIHdlIGFyZSBub3QgcmVxdWlyZWQgdG8gdXNlIGEgY3V0LW9mZiBvZiAwLjUuIElmIHdlIGluc3RlYWQgcHJlZGljdCBhIHB1cmNoYXNlIGFueSB0aW1lIHRoZSBwcmVkaWN0ZWQgcHJvYmFiaWxpdHkgb2YgcHVyY2hhc2UgZXhjZWVkcyAwLjI1LCB3ZSBnZXQgbXVjaCBiZXR0ZXIgcmVzdWx0czogd2UgcHJlZGljdCB0aGF0IDMzIHBlb3BsZSB3aWxsIHB1cmNoYXNlIGluc3VyYW5jZSwgYW5kIHdlIGFyZSBjb3JyZWN0IGZvciBhYm91dCAzMyAlIG9mIHRoZXNlIHBlb3BsZS4gVGhpcyBpcyBvdmVyIGZpdmUgdGltZXMgYmV0dGVyIHRoYW4gcmFuZG9tIGd1ZXNzaW5nIQ0KDQpgYGB7cn0NCmdsbS5maXRzPWdsbShQdXJjaGFzZX4uLGRhdGE9Q2FyYXZhbiAsZmFtaWx5PWJpbm9taWFsLHN1YnNldD0tdGVzdCkNCmdsbS5wcmVkPXJlcCgiTm8iLDEwMDApDQpnbG0ucHJlZFtnbG0ucHJvYnMgPi41XT0iIFllcyINCnRhYmxlKGdsbS5wcmVkICx0ZXN0LlkpDQpnbG0ucHJlZD1yZXAoIk5vIiwxMDAwKQ0KZ2xtLnByZWRbZ2xtLnByb2JzID4uMjVdPSIgWWVzIg0KdGFibGUoZ2xtLnByZWQgLHRlc3QuWSkNCjExLygyMisxMSkNCmBgYA0KDQoNCg0KDQo=