# Regresión Lineal
# Importar la base de datos de csv
df <- read.csv("C:\\Users\\ramir\\Downloads\\heart.csv")

Entender la base de datos

summary(df)
##       age             sex               cp            trestbps    
##  Min.   :29.00   Min.   :0.0000   Min.   :0.0000   Min.   : 94.0  
##  1st Qu.:48.00   1st Qu.:0.0000   1st Qu.:0.0000   1st Qu.:120.0  
##  Median :56.00   Median :1.0000   Median :1.0000   Median :130.0  
##  Mean   :54.43   Mean   :0.6956   Mean   :0.9424   Mean   :131.6  
##  3rd Qu.:61.00   3rd Qu.:1.0000   3rd Qu.:2.0000   3rd Qu.:140.0  
##  Max.   :77.00   Max.   :1.0000   Max.   :3.0000   Max.   :200.0  
##       chol          fbs            restecg          thalach     
##  Min.   :126   Min.   :0.0000   Min.   :0.0000   Min.   : 71.0  
##  1st Qu.:211   1st Qu.:0.0000   1st Qu.:0.0000   1st Qu.:132.0  
##  Median :240   Median :0.0000   Median :1.0000   Median :152.0  
##  Mean   :246   Mean   :0.1493   Mean   :0.5298   Mean   :149.1  
##  3rd Qu.:275   3rd Qu.:0.0000   3rd Qu.:1.0000   3rd Qu.:166.0  
##  Max.   :564   Max.   :1.0000   Max.   :2.0000   Max.   :202.0  
##      exang           oldpeak          slope             ca        
##  Min.   :0.0000   Min.   :0.000   Min.   :0.000   Min.   :0.0000  
##  1st Qu.:0.0000   1st Qu.:0.000   1st Qu.:1.000   1st Qu.:0.0000  
##  Median :0.0000   Median :0.800   Median :1.000   Median :0.0000  
##  Mean   :0.3366   Mean   :1.072   Mean   :1.385   Mean   :0.7541  
##  3rd Qu.:1.0000   3rd Qu.:1.800   3rd Qu.:2.000   3rd Qu.:1.0000  
##  Max.   :1.0000   Max.   :6.200   Max.   :2.000   Max.   :4.0000  
##       thal           target      
##  Min.   :0.000   Min.   :0.0000  
##  1st Qu.:2.000   1st Qu.:0.0000  
##  Median :2.000   Median :1.0000  
##  Mean   :2.324   Mean   :0.5132  
##  3rd Qu.:3.000   3rd Qu.:1.0000  
##  Max.   :3.000   Max.   :1.0000
str(df)
## 'data.frame':    1025 obs. of  14 variables:
##  $ age     : int  52 53 70 61 62 58 58 55 46 54 ...
##  $ sex     : int  1 1 1 1 0 0 1 1 1 1 ...
##  $ cp      : int  0 0 0 0 0 0 0 0 0 0 ...
##  $ trestbps: int  125 140 145 148 138 100 114 160 120 122 ...
##  $ chol    : int  212 203 174 203 294 248 318 289 249 286 ...
##  $ fbs     : int  0 1 0 0 1 0 0 0 0 0 ...
##  $ restecg : int  1 0 1 1 1 0 2 0 0 0 ...
##  $ thalach : int  168 155 125 161 106 122 140 145 144 116 ...
##  $ exang   : int  0 1 1 0 0 0 0 1 0 1 ...
##  $ oldpeak : num  1 3.1 2.6 0 1.9 1 4.4 0.8 0.8 3.2 ...
##  $ slope   : int  2 0 0 2 1 1 0 1 2 1 ...
##  $ ca      : int  2 0 0 1 3 0 3 1 0 2 ...
##  $ thal    : int  3 3 3 3 2 2 1 3 3 2 ...
##  $ target  : int  0 0 0 0 0 1 0 0 0 0 ...
df$sex <- as.factor(df$sex)
df$cp <- as.factor(df$cp)
df$fbs <- as.factor(df$fbs)
df$restecg <- as.factor(df$restecg)
df$exang <- as.factor(df$exang)
df$slope <- as.factor(df$slope)
df$thal <- as.factor(df$thal)
df$target <- as.factor(df$target)

Crear el modelo

modelo <- glm(target ~ ., data=df, family=binomial)
summary(modelo)
## 
## Call:
## glm(formula = target ~ ., family = binomial, data = df)
## 
## Coefficients:
##              Estimate Std. Error z value Pr(>|z|)    
## (Intercept)  0.974938   1.843806   0.529 0.596969    
## age         -0.004314   0.012820  -0.337 0.736479    
## sex1        -1.610703   0.283425  -5.683 1.32e-08 ***
## cp1          1.061225   0.301235   3.523 0.000427 ***
## cp2          1.963836   0.257085   7.639 2.19e-14 ***
## cp3          1.989568   0.352181   5.649 1.61e-08 ***
## trestbps    -0.014901   0.005819  -2.561 0.010443 *  
## chol        -0.005541   0.002130  -2.602 0.009277 ** 
## fbs1         0.048261   0.304550   0.158 0.874090    
## restecg1     0.511138   0.202653   2.522 0.011661 *  
## restecg2    -0.402546   1.224640  -0.329 0.742378    
## thalach      0.018227   0.005859   3.111 0.001865 ** 
## exang1      -0.751473   0.233353  -3.220 0.001280 ** 
## oldpeak     -0.506650   0.122129  -4.148 3.35e-05 ***
## slope1      -0.540297   0.456438  -1.184 0.236522    
## slope2       0.269358   0.492490   0.547 0.584427    
## ca          -0.813103   0.109901  -7.399 1.38e-13 ***
## thal1        1.918293   1.306918   1.468 0.142159    
## thal2        1.855539   1.263123   1.469 0.141831    
## thal3        0.523928   1.268851   0.413 0.679668    
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
## 
## (Dispersion parameter for binomial family taken to be 1)
## 
##     Null deviance: 1420.24  on 1024  degrees of freedom
## Residual deviance:  688.48  on 1005  degrees of freedom
## AIC: 728.48
## 
## Number of Fisher Scoring iterations: 6

Probar el modelo

prueba <- data.frame(
  age      = c(57, 65),
  sex      = factor(c(1, 0), levels = levels(df$sex)),
  cp       = factor(c(2, 2), levels = levels(df$cp)),
  trestbps = c(128, 160),
  chol     = c(229, 360),
  fbs      = factor(c(0, 0), levels = levels(df$fbs)),
  restecg  = factor(c(0, 0), levels = levels(df$restecg)),
  thalach  = c(150, 151),
  exang    = factor(c(0, 0), levels = levels(df$exang)),
  oldpeak  = c(0.4, 0.8),
  slope    = factor(c(1, 1), levels = levels(df$slope)),
  ca       = c(0, 0),
  thal     = factor(c(2, 2), levels = levels(df$thal))
)

probabilidad <- predict(modelo, newdata = prueba, type = "response")
cbind(prueba, Probabilidad_Target1 = probabilidad)
##   age sex cp trestbps chol fbs restecg thalach exang oldpeak slope ca thal
## 1  57   1  2      128  229   0       0     150     0     0.4     1  0    2
## 2  65   0  2      160  360   0       0     151     0     0.8     1  0    2
##   Probabilidad_Target1
## 1            0.8522842
## 2            0.8745388

Conclusiones

Para ambos pacientes, el modelo predice una alta probabilidad de pertenecer a la clase objetivo (target=1), con 0.85 y 0.87 respectivamente. El segundo perfil presenta una probabilidad ligeramente mayor, por lo que, según el modelo, se asocia más fuertemente con la clase target=1.

LS0tDQp0aXRsZTogIkhlYXJ0Ig0KYXV0aG9yOiAiSGVsZW5hIFJhbcOscmV6IEdpbGVzIg0KZGF0ZTogIjIwMjYtMDItMTkiDQpvdXRwdXQ6DQogIGh0bWxfZG9jdW1lbnQ6DQogICAgdG9jOiBUUlVFDQogICAgdG9jX2Zsb2F0OiBUUlVFDQogICAgY29kZV9kb3dubG9hZDogVFJVRQ0KICAgIHRoZW1lOiBjb3Ntbw0KLS0tDQo8Y2VudGVyPg0KIVtdKGRhdGE6aW1hZ2UvanBlZztiYXNlNjQsLzlqLzRBQVFTa1pKUmdBQkFRQUFBUUFCQUFELzJ3Q0VBQWtHQndnSEJna0lCd2dLQ2drTERSWVBEUXdNRFJzVUZSQVdJQjBpSWlBZEh4OGtLRFFzSkNZeEp4OGZMVDB0TVRVM09qbzZJeXMvUkQ4NFF6UTVPamNCQ2dvS0RRd05HZzhQR2pjbEh5VTNOemMzTnpjM056YzNOemMzTnpjM056YzNOemMzTnpjM056YzNOemMzTnpjM056YzNOemMzTnpjM056YzNOemMzTi8vQUFCRUlBSlFBcFFNQklnQUNFUUVERVFIL3hBQWNBQUVBQWdJREFRQUFBQUFBQUFBQUFBQUFCQVVHQndFQ0F3ai94QUE4RUFBQkF3SUVCQVVDQkFRRENRQUFBQUFCQUFJREJCRUZFaUV4QmhOQlVRY2lZWEdSZ2FFVUl6S3hRbEtTd1ROaThSVWtKa055c3NMUjhQL0VBQm9CQVFBREFRRUJBQUFBQUFBQUFBQUFBQUFDQXdRQkJRYi94QUFsRVFBREFBSUNBZ0VFQXdFQUFBQUFBQUFBQVFJREVTRXhCQklpQlRKQllTTXpVUlAvMmdBTUF3RUFBaEVERVFBL0FONG9pSUFpSWdDcU9KTWFHQzBUWnVVWkh5T3lNN0EySjErRmJMSCtPYVlWUERsU2RNMEpiSzBuME92MkpYR1NoSjB0bUYwM0dtSmppR0I5YlVacVI3c3BoamFBMFgrNStWSjhSWHRpeENLb2lKay9FUU41WWFkWG01MitsbGlUb3dYTmNXM0ExRmdwZUk0M0xKVHdSTjVZTUFMWXZMY3RCM0FVR2JWajlYdEhuRlZTWVpVeDFWUk1UV01CNWJJalpzZDlOVDFLNlUrSzFJeENLdk1yK2UxK2R2OEE5MVZMTEsrTU9tbmtCY05TU3JTREI2dWFpYmlPSlBGTlNPQTVUUTd6eVgydi9LRDhybkVuZE9ucEc4Y09xMjExQlRWY2Y2WjRteUQySXVwUzF4d0JWVlQ4WE5MRE83OEZGQnJFVGNDMmpiZGxzWUt4UFppeVI2Vm81UkVYU0FSRVFCRVJBRVJFQVJFUUJFUkFFUkVCd3FiakNhT0RoeXVkSWJCek1nOXliQlhTMXo0bllzUk5GaHJEZHJCelh0N25vbzA5SXR3dzd0SkdCNGhVdVphN3JkQUFWVVZWV3ltaGRQSTYzMVhURWFneVQ5ckwyNGF3cUxHdUk2V090SU5ORThHUzV1UGhWZXh1YTIrQ3ZncDhSeFNWa3ZMZkZUNzNjTEQzMVd3dUhzTy8ydkJIaGsxVFlEU09SaHVXT0EwMDZqMFZ4eHhTMGtFN1JEWnJXeGdCamRBT3l3dkJzWWZoV1AwdFFHamxDUVp4NmJGVk51bWFJU2pGdGZrdUs2aHhMaG5HYVNtNTQvRXp5dDVjc1owY013QS9mVmJtWmZLTTI5dFZoK01VY0ZmeEZoVTBwRHVXV3l3a2pjRWk0K3dLekFLL0c5N1BPOGg3VXQ5blpGeGRMcTB5bktMaGNvQWk0WEtBSWlJQWlJZ0NJaUFJaUlEcTVhTzRycmZ4Mk00aFVCN2orWTRON0JyZktBUGhicHhPZjhMaDlWVVh0eW9YUDE5QmRmUDBtYVdOMnVydFNTcytkOUk5VDZiajI2b3Bwblpua2s5Vlo4Q1hPSTFUbUM5NTdIMkFDcmFtSjBUcjM2OVZrWGgxQmMxMDdnUXhrenlTUCtscW9wL0UyNDUxa1cvMlduRU5ZNmVjaDdqbEZ5Ym5YL1JZMFMwU0dUTGM3QlRNVXFKSnBuT3NmTnFjM1QxVlBWVjdLU25ManFkN0hxcFlrUEthWENNcDRnNGp4QjNDOUhWeGd0bnBKbVJoNFA2bW4xOTdMWVZOeHZITFJRdmlwSkRJV0ROejNDTzV0cnB2OGdMVlBEdjRzVVBOcldBUlN1RDQ0SGkreHVIRWRGNjFtS1BpbE9jbCt1aE80WGQwdDZNdFRqdlhzdUViVFBITEdmNDFDUU83SmIvMlV5bDQwd2lZZ1N5dmdKMk1qZFBrTFNVdUp2bmxMQzQyT3R3bzlUaVVnWnlXSFhmZGRXVElucG5iOGJ4YWgwbG8ra0k2K2ttaDUwZFZDNkwrZHJ3UjhyR2NkOFFjSHdzT1pUeS9qSmgwaVBsQjlYZityclJVYzlTR09QUGxEU1RkdWMyWHR3eEUvSDhhZFNHcWlwcVdGaGtxcXVWN1FJbURycnVlaXQ5NmZCay80WTU1Yk42Y0RjV0RpYW5xZWJHSXFpbmY1bXR2Yktkaisvd3NxVlR3M1FZWGgrRTA4V0M4cDFJNW9jMldNaDNOdi9FVDFKVnNyVjBaSzE3Y0k1UkVYU0lSRVFCRVJBRVJFQmovQUI3UCtINFN4SWpkOFhML0FLamIrNjAwSXJRQ3gxVzFmRktZUjhNaU03elZFYlI4M1AyQzFpUUJhMjFsaDhsL0xSNy9BTktuK0p2OWxQWDBza3NidVdDWG5ZTE51RzhJa3dQaGR6SzE0am1uSmM2K2hKT3R2N0tvcDVXMDgwY3ZMRHkwM3M3WXJwaWVMMW1Jem5ueWdzYUxlWTVXanVxNWJwYU5XV0ZGZXgwbzhOcU1hcUhNcFdDVndHYVFnM3Q3cnptNGFocHExazJKdWpsZEdmeW9MK1VIdWUvc3ArQjRySmhtR3VpbzN0YUpuT0pjMXZtSTlDb05VOTV5dmtmbmNPdCs2dWxOSTg3TmtWMXlUS3Q3R1Iyek1jNXgxT2I3TEdNVGZtSnM2K3R0MU1ubURpRzN0MEhxcVdwMDgzcVZOSXoxVGJPekcyZEc3T0JaM1ZWYzlXNmZHWjJSUEFqWm8xdzJVOXJnK25kY0RNTnJxc3BvZjk1a2QxY1Z4ZHRsdDBrbEpNcXBKekdLZVI1Wk03VUJqZktSM3pYKzFrbzJPcFluUnhESUgvcmZwY2hTNDZZVFJoa3BPVzkybnEwK2k3aDVabHp0RGlDTE9icGV4M1VXMlhxVTNzeVhoWGlYaUtsanBhUERHVlVzRFpXa1J4UW1RRVp2TnNObHY1aEpBSkZyaTlsb21QeFA0a1pBMkdGdEF3TmJZTzVWeWZXMXdGdWpBS21Xc3dURHFxb2MxODAxTEhKSTVvc0M0dEJKSDFWK0pybzg3eXByYWRMUllJaUsweUJFUkFFUkVBUkVRR0FlTDBsc093eUVidXFTL3dDR0VmOEFrdGZNekVITVBaWnY0dlNYbXdxTzlnQks0bitrTEEyeTVScUxkejJYbjUvN0Q2VDZkeDQ2UFNlVVF0em01c05yS3ZEWDEwcktVR3dkcTQyMjlTdkt1ckE1eDFJYmZRZHl1V3pHa2lEQTYxUktidlA4bzdLV05hSStUYnJoRjNQSEdYWlljaklZYk1hTDdnS0JVdExtMnNRZWx0dmxkYWVjTmF4cDFiMGJicXVYeW1UTnFEZnNkbGNlWmUwK1NEVWtBNUpQTWI2TzdLc3FaVG1jMGczYWV2VlhGZEJkam5ONmhVZFJ6R2FPMVI4SGNhMjlubEcrNGNHblE5Vk9wYVpvL01BRytwVWFDSnhabXloVHFLZHpidzIvVWlPVzkwU3kyTnNld0o5VlA0VXd5Z3hYRlk4UHhNeVJ4eU84a2tSc1d1UFFuc1ZCaGFKSHRITEpMZHowVXVhUjlEUEZVUmpJV2tHN2VpNW83N05NeWpGZkNXcmJYTVpoZFRESlJTR3pqTzR0ZkY5Qm83N0xhV0JZWkZnMkVVZUcwN25PanBZV3hndTNOaHVWeGdkYzNFc0twcXRwdnpXQW4zNnF3RzZ2bVV1VVpNdVc2K05mZzVSRVVpb0lpSUFpSWdDSWlBMWg0dnlOWlg0V0hFQzhjbHIrN1ZxK294TU1MMmx3TGRRTExQUEhxVGwxMkVGeHMwUXluVDNhdFkwbE9aSkk2aXNibWhJdTFsOVQ2ckZrbjVzOTN4YjFnU0oxR0RNWFZsVTNMVHdqTTBXL3hIZEFvVTlVWFZCa2NMdmZ0cllOSG9wTlhXQ3FkUEJHMHNZR1hhZGdDUFJWQk9WMmcxdlpSNkpiMnk3cHFtUnh5a2dPQXRmdXJWMlZySTJ0OHhzc2FwcGl4NEx6YzMwOUZjVWM4UUljNXlzbG1iTkpMcjJ1ZlNlWDRWRE5zR3ZQc1ZmVlU3V3d0ZTRXQVB5cVN0RFhlZU93OUZLbVF3bzlxWEs2TU4zOVZJYkFJNVFRN1k2S3BwYWd4eUFIWldrSWJMZk1TQWsxdEhNdVBUMml3dytwaXBwclBsYTV0OW5MdmlqbVBnemg0SU8xdGdxcVNtYmwvTGNTUnZkU2FaelRGeTNuYlVabFBYQlQ3RzdmRFp4ZHdyVDVyWGE1d05sbFN3L3d3aWJCdzZZbS9wRW1ZSDNBV1lLeWVqTmsrOW5LSWlrUUNJaUFJaUlBaUlnTlRlUG1IU1QwMkRWd1plQ0dWOFV6L3dDWE1BUi8ya2ZDMVRYMUF5UXV1TnYwalpmVU9OWVhTWTFodFJoOWZIbnA1MkZyaHNSMklQUWpkZk5YSGZDbUljSlZ6b2F3R1drZTgvaHFvRFNRZGoyY08zd3FNa2ZrMytObVNYcXlwbnFzMFlqSnU4dHNTQm92SndjMGh6aGUrMWlvYlpiRURzTEtWRzhPRzZwcEd1YTJlc0xpMDNjMEUrcW1SdjJBMWQwVUVudXZha2VjenI3N3JpSlBvdTdHV2xESGtYN3FxbmJ5dERmM3VwMGNvNWJtdmRxNGFBZEZCcXRBQS96ZXE3UkdGcGtLVFRVS1pTMWVtUngwVU55NkM3WEFoY2w2SlZLWmtVRlNOd0xEMFhzOTJadDNhS2xwcWdnWEttTnFXbXdMcms5RmFxTTE0djhBRGMzaEhWbWJEcXFCNzJ1ZkU1dTNZcllLMC80Unp0aHgyU0FhR1dBM3QvRllnaS8zVzMxZEhSZ3lyVkhLSWltVmhFUkFFUkVBUkVRQlFzV3d1aXhqRDVxSEVxZGs5TktMT1k0ZmNkajZxYWlBK1kvRVR3N3J1RUpqVlVwa3E4SGNiTm5JQmZFZXo3ZnZ0N0xEb3BMVzFYMlRQREhQQzZLYU5za2J4bGN4d3VIRHNRdEhlSUhoRE5TT2x4TGhOanBZTDVwS0QrSm5jeG5xUDh1L2EreXJ1TjlHbkZtYTdOWWg5Mml5OXFja3VCRnRGQlk4dGM1cmdXdWFiT0JGaUQ2aFNZNUxCWjJ0TTlCV21pMWFiQVdQMVhsTEk2eHVNelZ6VEZwYmR4MFhXWTNPaE8reFhHaXlIc2p2YUw2STFseTFkbWtYTzJxN0J3YTYybWlpZFBPeGFORnpBNDVyaXdOMTJEZkp2MVhrNXdBMHNQVHV1b05Hd3ZDMnAvNDBvQmNrdkVqVC9RVnZvTDU4OEhvMzFISE5HOXYvQUNvcFpIZTJYTCs3Z3ZvTUxWaiswOG55ZUwwY29pS3d6aEVSQUVSRUFSRVFCRVJBRnd1VVFHRWNiK0d1RDhVdGZVTVkyaXhJalNxaVorcy81Mi94ZSs2MFh4THdianZDOHJtNGxSdU5NRFp0WEY1bzNqdmYrSDJObDlWcnBKSEhJd3NsWTE3SEN4YTRYQlVhbE10akxVSHlBeWNCbzl1NmtaaVd0NmxiNTRrOEplSGNXelM0ZkdjS3FEcmVtQTVaUHF6YjRzdFpZOTRZOFVZSzh5VTlPTVRwckVtU2wxYzMzWVRmNHVxYXhzMlI1TXZzeFBsdVp1ZzFLNlN2a2prTWRRMTBjZ05peDdTMGo2SFZkbUd3UFVxcjEwYXB0TWtOeTJ5OXQxSG1ZTlNGMmFRTmMyMjZ1ZUZ1R01RNHJ4RnRKUU5mSERmOCtxTGJzaUhYWHE3MFJKdG5idFN0czJGNENZTkl5bnI4Ym1hUTJZaUNDL1ZyZFhINXNQb3R1aFE4SXcybHdqREtiRDZKbkxwNmVNTVlQUWRUNjlWTTZyWkswdEhqWkw5NjJjb2lMcEFJaUlBaUlnQ0lpQUlpSUFpSWdDSWlBTGc3SWlBcnNVd1RDc1hpTWVKNGRTMVRUcitiRUhmZFl2VitGUENVNUpaUlRVL3BEVVBBL2RFVVdpYzAxMHptbDhLZUU2ZDRrZFJ6VDZhdG1xSE9hZm9zd3craHBNUHBtMDFEVFJVOERQMHh4TURXajZCRVhVaFZOOXNrb2lMcEFJaUlBaUlnUC8vWikNCjwvY2VudGVyPg0KDQoNCg0KYGBge3J9DQojIFJlZ3Jlc2nDs24gTGluZWFsDQojIEltcG9ydGFyIGxhIGJhc2UgZGUgZGF0b3MgZGUgY3N2DQpkZiA8LSByZWFkLmNzdigiQzpcXFVzZXJzXFxyYW1pclxcRG93bmxvYWRzXFxoZWFydC5jc3YiKQ0KYGBgDQoNCiMgPHNwYW4gc3R5bGU9ImNvbG9yOmJsdWUiPiBFbnRlbmRlciBsYSBiYXNlIGRlIGRhdG9zIDwvc3Bhbj4NCmBgYHtyfQ0Kc3VtbWFyeShkZikNCnN0cihkZikNCmRmJHNleCA8LSBhcy5mYWN0b3IoZGYkc2V4KQ0KZGYkY3AgPC0gYXMuZmFjdG9yKGRmJGNwKQ0KZGYkZmJzIDwtIGFzLmZhY3RvcihkZiRmYnMpDQpkZiRyZXN0ZWNnIDwtIGFzLmZhY3RvcihkZiRyZXN0ZWNnKQ0KZGYkZXhhbmcgPC0gYXMuZmFjdG9yKGRmJGV4YW5nKQ0KZGYkc2xvcGUgPC0gYXMuZmFjdG9yKGRmJHNsb3BlKQ0KZGYkdGhhbCA8LSBhcy5mYWN0b3IoZGYkdGhhbCkNCmRmJHRhcmdldCA8LSBhcy5mYWN0b3IoZGYkdGFyZ2V0KQ0KYGBgDQoNCiMgPHNwYW4gc3R5bGU9ImNvbG9yOmJsdWUiPiBDcmVhciBlbCBtb2RlbG8gPC9zcGFuPg0KYGBge3J9DQptb2RlbG8gPC0gZ2xtKHRhcmdldCB+IC4sIGRhdGE9ZGYsIGZhbWlseT1iaW5vbWlhbCkNCnN1bW1hcnkobW9kZWxvKQ0KYGBgDQoNCiMgPHNwYW4gc3R5bGU9ImNvbG9yOmJsdWUiPiBQcm9iYXIgZWwgbW9kZWxvIDwvc3Bhbj4NCmBgYHtyfQ0KcHJ1ZWJhIDwtIGRhdGEuZnJhbWUoDQogIGFnZSAgICAgID0gYyg1NywgNjUpLA0KICBzZXggICAgICA9IGZhY3RvcihjKDEsIDApLCBsZXZlbHMgPSBsZXZlbHMoZGYkc2V4KSksDQogIGNwICAgICAgID0gZmFjdG9yKGMoMiwgMiksIGxldmVscyA9IGxldmVscyhkZiRjcCkpLA0KICB0cmVzdGJwcyA9IGMoMTI4LCAxNjApLA0KICBjaG9sICAgICA9IGMoMjI5LCAzNjApLA0KICBmYnMgICAgICA9IGZhY3RvcihjKDAsIDApLCBsZXZlbHMgPSBsZXZlbHMoZGYkZmJzKSksDQogIHJlc3RlY2cgID0gZmFjdG9yKGMoMCwgMCksIGxldmVscyA9IGxldmVscyhkZiRyZXN0ZWNnKSksDQogIHRoYWxhY2ggID0gYygxNTAsIDE1MSksDQogIGV4YW5nICAgID0gZmFjdG9yKGMoMCwgMCksIGxldmVscyA9IGxldmVscyhkZiRleGFuZykpLA0KICBvbGRwZWFrICA9IGMoMC40LCAwLjgpLA0KICBzbG9wZSAgICA9IGZhY3RvcihjKDEsIDEpLCBsZXZlbHMgPSBsZXZlbHMoZGYkc2xvcGUpKSwNCiAgY2EgICAgICAgPSBjKDAsIDApLA0KICB0aGFsICAgICA9IGZhY3RvcihjKDIsIDIpLCBsZXZlbHMgPSBsZXZlbHMoZGYkdGhhbCkpDQopDQoNCnByb2JhYmlsaWRhZCA8LSBwcmVkaWN0KG1vZGVsbywgbmV3ZGF0YSA9IHBydWViYSwgdHlwZSA9ICJyZXNwb25zZSIpDQpjYmluZChwcnVlYmEsIFByb2JhYmlsaWRhZF9UYXJnZXQxID0gcHJvYmFiaWxpZGFkKQ0KYGBgDQoNCiMgPHNwYW4gc3R5bGU9ImNvbG9yOmJsdWUiPiBDb25jbHVzaW9uZXMgPC9zcGFuPg0KUGFyYSBhbWJvcyBwYWNpZW50ZXMsIGVsIG1vZGVsbyBwcmVkaWNlIHVuYSBhbHRhIHByb2JhYmlsaWRhZCBkZSBwZXJ0ZW5lY2VyIGEgbGEgY2xhc2Ugb2JqZXRpdm8gKHRhcmdldD0xKSwgY29uIDAuODUgeSAwLjg3IHJlc3BlY3RpdmFtZW50ZS4gRWwgc2VndW5kbyBwZXJmaWwgcHJlc2VudGEgdW5hIHByb2JhYmlsaWRhZCBsaWdlcmFtZW50ZSBtYXlvciwgcG9yIGxvIHF1ZSwgc2Vnw7puIGVsIG1vZGVsbywgc2UgYXNvY2lhIG3DoXMgZnVlcnRlbWVudGUgY29uIGxhIGNsYXNlIHRhcmdldD0xLiANCg0KDQoNCg0KDQo=