#file.choose()

#install.packages("tidyverse")
library (tidyverse)
## ── Attaching core tidyverse packages ──────────────────────── tidyverse 2.0.0 ──
## ✔ dplyr     1.1.4     ✔ readr     2.1.5
## ✔ forcats   1.0.0     ✔ stringr   1.5.1
## ✔ ggplot2   3.5.1     ✔ tibble    3.2.1
## ✔ lubridate 1.9.4     ✔ tidyr     1.3.1
## ✔ purrr     1.0.4     
## ── Conflicts ────────────────────────────────────────── tidyverse_conflicts() ──
## ✖ dplyr::filter() masks stats::filter()
## ✖ dplyr::lag()    masks stats::lag()
## ℹ Use the conflicted package (<http://conflicted.r-lib.org/>) to force all conflicts to become errors
#install.packages("dplyr")
library (dplyr)
#install.packages("lubridate")
library (lubridate)
#install.packages("Matrix")
library (Matrix)
## 
## Adjuntando el paquete: 'Matrix'
## 
## The following objects are masked from 'package:tidyr':
## 
##     expand, pack, unpack
#install.packages("arules")
library (arules)
## 
## Adjuntando el paquete: 'arules'
## 
## The following object is masked from 'package:dplyr':
## 
##     recode
## 
## The following objects are masked from 'package:base':
## 
##     abbreviate, write
#install.packages("arulesViz")
library (arulesViz)
#install.packages("datasets")
library (datasets)
#install.packages("plyr")
library (plyr)
## ------------------------------------------------------------------------------
## You have loaded plyr after dplyr - this is likely to cause problems.
## If you need functions from both plyr and dplyr, please load plyr first, then dplyr:
## library(plyr); library(dplyr)
## ------------------------------------------------------------------------------
## 
## Adjuntando el paquete: 'plyr'
## 
## The following objects are masked from 'package:dplyr':
## 
##     arrange, count, desc, failwith, id, mutate, rename, summarise,
##     summarize
## 
## The following object is masked from 'package:purrr':
## 
##     compact

importar base de datos

#file.choose()
bd <- read.csv("C:\\Users\\lechu\\OneDrive\\Desktop\\abarrotes.csv")

entender la base de datos

summary(bd)
##  vcClaveTienda        DescGiro         Codigo.Barras            PLU        
##  Length:200625      Length:200625      Min.   :8.347e+05   Min.   : 1.00   
##  Class :character   Class :character   1st Qu.:7.501e+12   1st Qu.: 1.00   
##  Mode  :character   Mode  :character   Median :7.501e+12   Median : 1.00   
##                                        Mean   :5.950e+12   Mean   : 2.11   
##                                        3rd Qu.:7.501e+12   3rd Qu.: 1.00   
##                                        Max.   :1.750e+13   Max.   :30.00   
##                                                            NA's   :199188  
##     Fecha               Hora              Marca            Fabricante       
##  Length:200625      Length:200625      Length:200625      Length:200625     
##  Class :character   Class :character   Class :character   Class :character  
##  Mode  :character   Mode  :character   Mode  :character   Mode  :character  
##                                                                             
##                                                                             
##                                                                             
##                                                                             
##    Producto             Precio          Ult.Costo         Unidades     
##  Length:200625      Min.   :-147.00   Min.   :  0.38   Min.   : 0.200  
##  Class :character   1st Qu.:  11.00   1st Qu.:  8.46   1st Qu.: 1.000  
##  Mode  :character   Median :  16.00   Median : 12.31   Median : 1.000  
##                     Mean   :  19.42   Mean   : 15.31   Mean   : 1.262  
##                     3rd Qu.:  25.00   3rd Qu.: 19.23   3rd Qu.: 1.000  
##                     Max.   :1000.00   Max.   :769.23   Max.   :96.000  
##                                                                        
##     F.Ticket      NombreDepartamento NombreFamilia      NombreCategoria   
##  Min.   :     1   Length:200625      Length:200625      Length:200625     
##  1st Qu.: 33964   Class :character   Class :character   Class :character  
##  Median :105993   Mode  :character   Mode  :character   Mode  :character  
##  Mean   :193990                                                           
##  3rd Qu.:383005                                                           
##  Max.   :450040                                                           
##                                                                           
##     Estado              Mts.2      Tipo.ubicación         Giro          
##  Length:200625      Min.   :47.0   Length:200625      Length:200625     
##  Class :character   1st Qu.:53.0   Class :character   Class :character  
##  Mode  :character   Median :60.0   Mode  :character   Mode  :character  
##                     Mean   :56.6                                        
##                     3rd Qu.:60.0                                        
##                     Max.   :62.0                                        
##                                                                         
##  Hora.inicio        Hora.cierre       
##  Length:200625      Length:200625     
##  Class :character   Class :character  
##  Mode  :character   Mode  :character  
##                                       
##                                       
##                                       
## 
str(bd)
## 'data.frame':    200625 obs. of  22 variables:
##  $ vcClaveTienda     : chr  "MX001" "MX001" "MX001" "MX001" ...
##  $ DescGiro          : chr  "Abarrotes" "Abarrotes" "Abarrotes" "Abarrotes" ...
##  $ Codigo.Barras     : num  7.5e+12 7.5e+12 7.5e+12 7.5e+12 7.5e+12 ...
##  $ PLU               : int  NA NA NA NA NA NA NA NA NA NA ...
##  $ Fecha             : chr  "19/06/2020" "19/06/2020" "19/06/2020" "19/06/2020" ...
##  $ Hora              : chr  "08:16:21" "08:23:33" "08:24:33" "08:24:33" ...
##  $ Marca             : chr  "NUTRI LECHE" "DAN UP" "BIMBO" "PEPSI" ...
##  $ Fabricante        : chr  "MEXILAC" "DANONE DE MEXICO" "GRUPO BIMBO" "PEPSI-COLA MEXICANA" ...
##  $ Producto          : chr  "Nutri Leche 1 Litro" "DANUP STRAWBERRY P/BEBER 350GR NAL" "Rebanadas Bimbo 2Pz" "Pepsi N.R. 400Ml" ...
##  $ Precio            : num  16 14 5 8 19.5 16 14 5 8 19.5 ...
##  $ Ult.Costo         : num  12.3 14 5 8 15 ...
##  $ Unidades          : num  1 1 1 1 1 1 1 1 1 1 ...
##  $ F.Ticket          : int  1 2 3 3 4 1 2 3 3 4 ...
##  $ NombreDepartamento: chr  "Abarrotes" "Abarrotes" "Abarrotes" "Abarrotes" ...
##  $ NombreFamilia     : chr  "Lacteos y Refrigerados" "Lacteos y Refrigerados" "Pan y Tortilla" "Bebidas" ...
##  $ NombreCategoria   : chr  "Leche" "Yogurt" "Pan Dulce Empaquetado" "Refrescos Plástico (N.R.)" ...
##  $ Estado            : chr  "Nuevo León" "Nuevo León" "Nuevo León" "Nuevo León" ...
##  $ Mts.2             : int  60 60 60 60 60 60 60 60 60 60 ...
##  $ Tipo.ubicación    : chr  "Esquina" "Esquina" "Esquina" "Esquina" ...
##  $ Giro              : chr  "Abarrotes" "Abarrotes" "Abarrotes" "Abarrotes" ...
##  $ Hora.inicio       : chr  "08:00" "08:00" "08:00" "08:00" ...
##  $ Hora.cierre       : chr  "22:00" "22:00" "22:00" "22:00" ...
head(bd, n=10)
##    vcClaveTienda  DescGiro Codigo.Barras PLU      Fecha     Hora
## 1          MX001 Abarrotes  7.501021e+12  NA 19/06/2020 08:16:21
## 2          MX001 Abarrotes  7.501032e+12  NA 19/06/2020 08:23:33
## 3          MX001 Abarrotes  7.501000e+12  NA 19/06/2020 08:24:33
## 4          MX001 Abarrotes  7.501031e+12  NA 19/06/2020 08:24:33
## 5          MX001 Abarrotes  7.501026e+12  NA 19/06/2020 08:26:28
## 6          MX001 Abarrotes  7.501021e+12  NA 19/06/2020 08:16:21
## 7          MX001 Abarrotes  7.501032e+12  NA 19/06/2020 08:23:33
## 8          MX001 Abarrotes  7.501000e+12  NA 19/06/2020 08:24:33
## 9          MX001 Abarrotes  7.501031e+12  NA 19/06/2020 08:24:33
## 10         MX001 Abarrotes  7.501026e+12  NA 19/06/2020 08:26:28
##                         Marca                 Fabricante
## 1                 NUTRI LECHE                    MEXILAC
## 2                      DAN UP           DANONE DE MEXICO
## 3                       BIMBO                GRUPO BIMBO
## 4                       PEPSI        PEPSI-COLA MEXICANA
## 5  BLANCA NIEVES (DETERGENTE) FABRICA DE JABON LA CORONA
## 6                 NUTRI LECHE                    MEXILAC
## 7                      DAN UP           DANONE DE MEXICO
## 8                       BIMBO                GRUPO BIMBO
## 9                       PEPSI        PEPSI-COLA MEXICANA
## 10 BLANCA NIEVES (DETERGENTE) FABRICA DE JABON LA CORONA
##                              Producto Precio Ult.Costo Unidades F.Ticket
## 1                 Nutri Leche 1 Litro   16.0     12.31        1        1
## 2  DANUP STRAWBERRY P/BEBER 350GR NAL   14.0     14.00        1        2
## 3                 Rebanadas Bimbo 2Pz    5.0      5.00        1        3
## 4                    Pepsi N.R. 400Ml    8.0      8.00        1        3
## 5       Detergente Blanca Nieves 500G   19.5     15.00        1        4
## 6                 Nutri Leche 1 Litro   16.0     12.31        1        1
## 7  DANUP STRAWBERRY P/BEBER 350GR NAL   14.0     14.00        1        2
## 8                 Rebanadas Bimbo 2Pz    5.0      5.00        1        3
## 9                    Pepsi N.R. 400Ml    8.0      8.00        1        3
## 10      Detergente Blanca Nieves 500G   19.5     15.00        1        4
##    NombreDepartamento          NombreFamilia           NombreCategoria
## 1           Abarrotes Lacteos y Refrigerados                     Leche
## 2           Abarrotes Lacteos y Refrigerados                    Yogurt
## 3           Abarrotes         Pan y Tortilla     Pan Dulce Empaquetado
## 4           Abarrotes                Bebidas Refrescos Plástico (N.R.)
## 5           Abarrotes     Limpieza del Hogar                Lavandería
## 6           Abarrotes Lacteos y Refrigerados                     Leche
## 7           Abarrotes Lacteos y Refrigerados                    Yogurt
## 8           Abarrotes         Pan y Tortilla     Pan Dulce Empaquetado
## 9           Abarrotes                Bebidas Refrescos Plástico (N.R.)
## 10          Abarrotes     Limpieza del Hogar                Lavandería
##        Estado Mts.2 Tipo.ubicación      Giro Hora.inicio Hora.cierre
## 1  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 2  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 3  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 4  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 5  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 6  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 7  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 8  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 9  Nuevo León    60        Esquina Abarrotes       08:00       22:00
## 10 Nuevo León    60        Esquina Abarrotes       08:00       22:00
tail(bd, n=10)
##        vcClaveTienda DescGiro Codigo.Barras PLU      Fecha     Hora
## 200616         MX005 Depósito   7.62221e+12  NA 07/08/2020 19:30:13
## 200617         MX005 Depósito   7.62221e+12  NA 25/07/2020 18:42:24
## 200618         MX005 Depósito   7.62221e+12  NA 18/07/2020 22:45:58
## 200619         MX005 Depósito   7.62221e+12  NA 12/07/2020 00:36:34
## 200620         MX005 Depósito   7.62221e+12  NA 12/07/2020 01:08:25
## 200621         MX005 Depósito   7.62221e+12  NA 23/10/2020 22:17:37
## 200622         MX005 Depósito   7.62221e+12  NA 10/10/2020 20:30:20
## 200623         MX005 Depósito   7.62221e+12  NA 10/10/2020 22:40:43
## 200624         MX005 Depósito   7.62221e+12  NA 27/06/2020 22:30:19
## 200625         MX005 Depósito   7.62221e+12  NA 26/06/2020 23:43:34
##                    Marca    Fabricante                          Producto Precio
## 200616 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200617 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200618 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200619 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200620 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200621 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200622 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200623 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200624 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
## 200625 TRIDENT XTRA CARE CADBURY ADAMS Trident Xtracare Freshmint 16.32G      9
##        Ult.Costo Unidades F.Ticket NombreDepartamento NombreFamilia
## 200616      6.92        1   106411          Abarrotes      Dulcería
## 200617      6.92        1   104693          Abarrotes      Dulcería
## 200618      6.92        1   103856          Abarrotes      Dulcería
## 200619      6.92        1   103087          Abarrotes      Dulcería
## 200620      6.92        1   103100          Abarrotes      Dulcería
## 200621      6.92        1   116598          Abarrotes      Dulcería
## 200622      6.92        1   114886          Abarrotes      Dulcería
## 200623      6.92        1   114955          Abarrotes      Dulcería
## 200624      6.92        1   101121          Abarrotes      Dulcería
## 200625      6.92        1   100879          Abarrotes      Dulcería
##        NombreCategoria       Estado Mts.2 Tipo.ubicación       Giro Hora.inicio
## 200616 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200617 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200618 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200619 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200620 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200621 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200622 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200623 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200624 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
## 200625 Gomas de Mazcar Quintana Roo    58        Esquina Mini súper       08:00
##        Hora.cierre
## 200616       21:00
## 200617       21:00
## 200618       21:00
## 200619       21:00
## 200620       21:00
## 200621       21:00
## 200622       21:00
## 200623       21:00
## 200624       21:00
## 200625       21:00
#plyr::count(bd, DescGiro, sort=TRUE)
#plyr::count(bd, Marca, sort=TRUE)
#plyr::count(bd, Fecha, sort=TRUE)
#plyr::count(bd, Hora, sort=TRUE)
#plyr::count(bd, Fabricante, sort=TRUE)
#plyr::count(bd, Producto, sort=TRUE)
#plyr::count(bd, NombreDepartamento, sort=TRUE)
#plyr::count(bd, NombreFamilia, sort=TRUE)
#plyr::count(bd, NombreCategoria, sort=TRUE)
#plyr::count(bd, Estado, sort=TRUE)
#plyr::count(bd, Tipo.ubicación, sort=TRUE)
#plyr::count(bd, Giro, sort=TRUE)

Limpiar base de datos

Tecnica 1. Remover valores irrelevantes

#Eliminar Columnas
bd1 <- bd
bd1 <- subset(bd1, select = -c(PLU, Codigo.Barras))

#Eliminar renglones
bd2 <- bd1
bd2 <- bd2[bd2$Precio > 0, ]
summary(bd1$Precio)
##    Min. 1st Qu.  Median    Mean 3rd Qu.    Max. 
## -147.00   11.00   16.00   19.42   25.00 1000.00
summary(bd2$Precio)
##    Min. 1st Qu.  Median    Mean 3rd Qu.    Max. 
##    0.50   11.00   16.00   19.45   25.00 1000.00

Tecnica 2. Remover valores duplicados

#cuantos valores duplicados tenemos?
sum(duplicated(bd2))
## [1] 5
#cuales son los renglones duplicados?
bd2[duplicated(bd2),]
##    vcClaveTienda  DescGiro      Fecha     Hora                      Marca
## 6          MX001 Abarrotes 19/06/2020 08:16:21                NUTRI LECHE
## 7          MX001 Abarrotes 19/06/2020 08:23:33                     DAN UP
## 8          MX001 Abarrotes 19/06/2020 08:24:33                      BIMBO
## 9          MX001 Abarrotes 19/06/2020 08:24:33                      PEPSI
## 10         MX001 Abarrotes 19/06/2020 08:26:28 BLANCA NIEVES (DETERGENTE)
##                    Fabricante                           Producto Precio
## 6                     MEXILAC                Nutri Leche 1 Litro   16.0
## 7            DANONE DE MEXICO DANUP STRAWBERRY P/BEBER 350GR NAL   14.0
## 8                 GRUPO BIMBO                Rebanadas Bimbo 2Pz    5.0
## 9         PEPSI-COLA MEXICANA                   Pepsi N.R. 400Ml    8.0
## 10 FABRICA DE JABON LA CORONA      Detergente Blanca Nieves 500G   19.5
##    Ult.Costo Unidades F.Ticket NombreDepartamento          NombreFamilia
## 6      12.31        1        1          Abarrotes Lacteos y Refrigerados
## 7      14.00        1        2          Abarrotes Lacteos y Refrigerados
## 8       5.00        1        3          Abarrotes         Pan y Tortilla
## 9       8.00        1        3          Abarrotes                Bebidas
## 10     15.00        1        4          Abarrotes     Limpieza del Hogar
##              NombreCategoria     Estado Mts.2 Tipo.ubicación      Giro
## 6                      Leche Nuevo León    60        Esquina Abarrotes
## 7                     Yogurt Nuevo León    60        Esquina Abarrotes
## 8      Pan Dulce Empaquetado Nuevo León    60        Esquina Abarrotes
## 9  Refrescos Plástico (N.R.) Nuevo León    60        Esquina Abarrotes
## 10                Lavandería Nuevo León    60        Esquina Abarrotes
##    Hora.inicio Hora.cierre
## 6        08:00       22:00
## 7        08:00       22:00
## 8        08:00       22:00
## 9        08:00       22:00
## 10       08:00       22:00
#Eliminar renglones duplicados
bd3 <- bd2
bd3 <- distinct(bd3)

Tecnica 3. Eliminar errores tipograficos similares

bd4 <- bd3
bd4$Unidades <- ceiling(bd4$Unidades)

Tecnica 4. Convertir tipos de datos

#convertir caracter a entero
bd5 <- bd4
bd5$Hora <- substr(bd5$Hora, start = 1, stop = 2)
bd5$Hora <- as.integer(bd5$Hora)

Tecnica 5. Reemplazar valores faltantes

#cuantos NAs tengo en la base de datos?
sum(is.na(bd5))
## [1] 0
#cuantos NAs tengo ppr variable?
sapply(bd, function(x) sum(is.na(x)))
##      vcClaveTienda           DescGiro      Codigo.Barras                PLU 
##                  0                  0                  0             199188 
##              Fecha               Hora              Marca         Fabricante 
##                  0                  0                  0                  0 
##           Producto             Precio          Ult.Costo           Unidades 
##                  0                  0                  0                  0 
##           F.Ticket NombreDepartamento      NombreFamilia    NombreCategoria 
##                  0                  0                  0                  0 
##             Estado              Mts.2     Tipo.ubicación               Giro 
##                  0                  0                  0                  0 
##        Hora.inicio        Hora.cierre 
##                  0                  0
#opcion 1. borrar todos los NAs de una tabla
#bd100 <- na.omit(bd)

#opcion 2. reemplazar los NAs con 0
# bd101 <- bd
# bd101[is.na(bd101)]<-0

#opcion 3. reemplazar los NAs con el promedio
# bd102 <- bd
# bd102$PLU[is.na(bd$PLU)]<-mean(bd102$PLU, na.rm=TRUE)

Tecnica 5. Correciones por metodos estadisticos

boxplot(bd5$Precio, horizontal=TRUE)

boxplot(bd5$Unidades, horizontal=TRUE)

# Agregar columnas

# Agregar dia de la semana
bd5$Dia_Semana <- wday(bd5$Fecha)
# Agregar el subtotal 
bd5$Subtotal <- bd5$Precio * bd5$Unidades

#Exportar la base de datos

#write.csv(bd5, file="abarrotes_limpi.csv", row.names = FALSE)

#Generar Market Basket Analysis

#Ordenar de menor a mayor la columna ticket
bd6 <- bd5
bd6 <- bd6[order(bd6$F.Ticket), ]
#generar el canasto
basket <- ddply(bd6,c("F.Ticket"), function(bd6)paste(bd6$Marca,collapse = ","))

#eliminar columna ticket
basket$F.Ticket <- NULL

#renombrar nombre de columna marca
colnames(basket) <- c("Marca")

#exportar basket
write.csv(basket, file="basket.csv", row.names = FALSE)

#importar transacciones
#file.choose()
tr <- read.transactions("C:\\Users\\lechu\\OneDrive\\Desktop\\basket.csv", format = "basket", sep = ",")
reglas.asociacion <- apriori(tr, parameter = list(supp=00.001, conf=0.2, maxlen=10))
## Apriori
## 
## Parameter specification:
##  confidence minval smax arem  aval originalSupport maxtime support minlen
##         0.2    0.1    1 none FALSE            TRUE       5   0.001      1
##  maxlen target  ext
##      10  rules TRUE
## 
## Algorithmic control:
##  filter tree heap memopt load sort verbose
##     0.1 TRUE TRUE  FALSE TRUE    2    TRUE
## 
## Absolute minimum support count: 115 
## 
## set item appearances ...[0 item(s)] done [0.00s].
## set transactions ...[28710 item(s), 115031 transaction(s)] done [0.08s].
## sorting and recoding items ... [119 item(s)] done [0.00s].
## creating transaction tree ... done [0.02s].
## checking subsets of size 1 done [0.01s].
## writing ... [0 rule(s)] done [0.00s].
## creating S4 object  ... done [0.01s].
summary(reglas.asociacion)
## set of 0 rules
inspect(reglas.asociacion)

#ordenar reglas de asociacion
reglas.asociacion <- sort(reglas.asociacion, by="confidence", decreasing = TRUE)

#ordenar
top10reglas <- head (reglas.asociacion, n=10, by="confidence")
#plot (top10reglas, method="graph", engine="htmlwidget")
#file.choose()

##CONCLUSIONES El uso de RStudio nos mostró lo importante que es limpiar los datos antes de analizarlos. Al trabajar con una tienda de abarrotes, vimos que muchas veces la información tiene errores o valores faltantes. Si no se corrigen, los resultados pueden ser incorrectos. Aprendimos a usar funciones en R para eliminar datos nulos, corregir formatos y organizar la información.

También analizamos patrones de compra entre productos. Notamos que ciertos artículos se compran juntos con frecuencia, lo que es útil para estrategias de venta. Por ejemplo, si el café y las galletas se venden juntos, la tienda podría colocarlos cerca para incentivar más compras.

Trabajar con RStudio mejoró nuestras habilidades de programación con datos reales. Al principio, los comandos pueden parecer complicados, pero con práctica se vuelven más fáciles. Además, R nos permite visualizar datos con gráficos y tablas para entender mejor la información.

En general, este análisis nos mostró cómo los datos ayudan a tomar mejores decisiones en los negocios.

LS0tDQp0aXRsZTogIk1hcmtldCBCYXNrZXQgQW5hbHlzaXMiDQphdXRob3I6ICJKYXZpZXIgTGVjaHVnYSBBMDE1NzE1MTMiDQpkYXRlOiAiMDMvMTkvMjUiDQpvdXRwdXQ6IA0KICBodG1sX2RvY3VtZW50Og0KICAgIHRvYzogVFJVRQ0KICAgIHRvY19mbG9hdDogVFJVRQ0KICAgIGNvZGVfZG93bmxvYWQ6IFRSVUUNCiAgICB0aGVtZTogInlldGkiDQotLS0NCmBgYHtyfQ0KI2ZpbGUuY2hvb3NlKCkNCmBgYA0KDQohW10oQzpcXFVzZXJzXFxsZWNodVxcT25lRHJpdmVcXERlc2t0b3BcXGRlc3BlbnNhLWRpYnVqYWRhLW1hbm8tZGlmZXJlbnRlcy1hbGltZW50b3NfMjMtMjE0ODcyNDU4OC5hdmlmKQ0KDQpgYGB7cn0NCiNpbnN0YWxsLnBhY2thZ2VzKCJ0aWR5dmVyc2UiKQ0KbGlicmFyeSAodGlkeXZlcnNlKQ0KI2luc3RhbGwucGFja2FnZXMoImRwbHlyIikNCmxpYnJhcnkgKGRwbHlyKQ0KI2luc3RhbGwucGFja2FnZXMoImx1YnJpZGF0ZSIpDQpsaWJyYXJ5IChsdWJyaWRhdGUpDQojaW5zdGFsbC5wYWNrYWdlcygiTWF0cml4IikNCmxpYnJhcnkgKE1hdHJpeCkNCiNpbnN0YWxsLnBhY2thZ2VzKCJhcnVsZXMiKQ0KbGlicmFyeSAoYXJ1bGVzKQ0KI2luc3RhbGwucGFja2FnZXMoImFydWxlc1ZpeiIpDQpsaWJyYXJ5IChhcnVsZXNWaXopDQojaW5zdGFsbC5wYWNrYWdlcygiZGF0YXNldHMiKQ0KbGlicmFyeSAoZGF0YXNldHMpDQojaW5zdGFsbC5wYWNrYWdlcygicGx5ciIpDQpsaWJyYXJ5IChwbHlyKQ0KYGBgDQoNCiMgaW1wb3J0YXIgYmFzZSBkZSBkYXRvcw0KYGBge3J9DQojZmlsZS5jaG9vc2UoKQ0KYmQgPC0gcmVhZC5jc3YoIkM6XFxVc2Vyc1xcbGVjaHVcXE9uZURyaXZlXFxEZXNrdG9wXFxhYmFycm90ZXMuY3N2IikNCmBgYA0KDQojIGVudGVuZGVyIGxhIGJhc2UgZGUgZGF0b3MNCmBgYHtyfQ0Kc3VtbWFyeShiZCkNCnN0cihiZCkNCmhlYWQoYmQsIG49MTApDQp0YWlsKGJkLCBuPTEwKQ0KI3BseXI6OmNvdW50KGJkLCBEZXNjR2lybywgc29ydD1UUlVFKQ0KI3BseXI6OmNvdW50KGJkLCBNYXJjYSwgc29ydD1UUlVFKQ0KI3BseXI6OmNvdW50KGJkLCBGZWNoYSwgc29ydD1UUlVFKQ0KI3BseXI6OmNvdW50KGJkLCBIb3JhLCBzb3J0PVRSVUUpDQojcGx5cjo6Y291bnQoYmQsIEZhYnJpY2FudGUsIHNvcnQ9VFJVRSkNCiNwbHlyOjpjb3VudChiZCwgUHJvZHVjdG8sIHNvcnQ9VFJVRSkNCiNwbHlyOjpjb3VudChiZCwgTm9tYnJlRGVwYXJ0YW1lbnRvLCBzb3J0PVRSVUUpDQojcGx5cjo6Y291bnQoYmQsIE5vbWJyZUZhbWlsaWEsIHNvcnQ9VFJVRSkNCiNwbHlyOjpjb3VudChiZCwgTm9tYnJlQ2F0ZWdvcmlhLCBzb3J0PVRSVUUpDQojcGx5cjo6Y291bnQoYmQsIEVzdGFkbywgc29ydD1UUlVFKQ0KI3BseXI6OmNvdW50KGJkLCBUaXBvLnViaWNhY2nDs24sIHNvcnQ9VFJVRSkNCiNwbHlyOjpjb3VudChiZCwgR2lybywgc29ydD1UUlVFKQ0KYGBgDQoNCiMgTGltcGlhciBiYXNlIGRlIGRhdG9zDQoNCiMjIFRlY25pY2EgMS4gUmVtb3ZlciB2YWxvcmVzIGlycmVsZXZhbnRlcw0KYGBge3J9DQojRWxpbWluYXIgQ29sdW1uYXMNCmJkMSA8LSBiZA0KYmQxIDwtIHN1YnNldChiZDEsIHNlbGVjdCA9IC1jKFBMVSwgQ29kaWdvLkJhcnJhcykpDQoNCiNFbGltaW5hciByZW5nbG9uZXMNCmJkMiA8LSBiZDENCmJkMiA8LSBiZDJbYmQyJFByZWNpbyA+IDAsIF0NCnN1bW1hcnkoYmQxJFByZWNpbykNCnN1bW1hcnkoYmQyJFByZWNpbykNCmBgYA0KDQojIyBUZWNuaWNhIDIuIFJlbW92ZXIgdmFsb3JlcyBkdXBsaWNhZG9zDQpgYGB7cn0NCiNjdWFudG9zIHZhbG9yZXMgZHVwbGljYWRvcyB0ZW5lbW9zPw0Kc3VtKGR1cGxpY2F0ZWQoYmQyKSkNCiNjdWFsZXMgc29uIGxvcyByZW5nbG9uZXMgZHVwbGljYWRvcz8NCmJkMltkdXBsaWNhdGVkKGJkMiksXQ0KI0VsaW1pbmFyIHJlbmdsb25lcyBkdXBsaWNhZG9zDQpiZDMgPC0gYmQyDQpiZDMgPC0gZGlzdGluY3QoYmQzKQ0KYGBgDQojIyBUZWNuaWNhIDMuIEVsaW1pbmFyIGVycm9yZXMgdGlwb2dyYWZpY29zIHNpbWlsYXJlcw0KYGBge3J9DQpiZDQgPC0gYmQzDQpiZDQkVW5pZGFkZXMgPC0gY2VpbGluZyhiZDQkVW5pZGFkZXMpDQpgYGANCiMjIFRlY25pY2EgNC4gQ29udmVydGlyIHRpcG9zIGRlIGRhdG9zDQpgYGB7cn0NCiNjb252ZXJ0aXIgY2FyYWN0ZXIgYSBlbnRlcm8NCmJkNSA8LSBiZDQNCmJkNSRIb3JhIDwtIHN1YnN0cihiZDUkSG9yYSwgc3RhcnQgPSAxLCBzdG9wID0gMikNCmJkNSRIb3JhIDwtIGFzLmludGVnZXIoYmQ1JEhvcmEpDQpgYGANCiMjIFRlY25pY2EgNS4gUmVlbXBsYXphciB2YWxvcmVzIGZhbHRhbnRlcw0KYGBge3J9DQojY3VhbnRvcyBOQXMgdGVuZ28gZW4gbGEgYmFzZSBkZSBkYXRvcz8NCnN1bShpcy5uYShiZDUpKQ0KI2N1YW50b3MgTkFzIHRlbmdvIHBwciB2YXJpYWJsZT8NCnNhcHBseShiZCwgZnVuY3Rpb24oeCkgc3VtKGlzLm5hKHgpKSkNCg0KI29wY2lvbiAxLiBib3JyYXIgdG9kb3MgbG9zIE5BcyBkZSB1bmEgdGFibGENCiNiZDEwMCA8LSBuYS5vbWl0KGJkKQ0KDQojb3BjaW9uIDIuIHJlZW1wbGF6YXIgbG9zIE5BcyBjb24gMA0KIyBiZDEwMSA8LSBiZA0KIyBiZDEwMVtpcy5uYShiZDEwMSldPC0wDQoNCiNvcGNpb24gMy4gcmVlbXBsYXphciBsb3MgTkFzIGNvbiBlbCBwcm9tZWRpbw0KIyBiZDEwMiA8LSBiZA0KIyBiZDEwMiRQTFVbaXMubmEoYmQkUExVKV08LW1lYW4oYmQxMDIkUExVLCBuYS5ybT1UUlVFKQ0KYGBgDQojIyBUZWNuaWNhIDUuIENvcnJlY2lvbmVzIHBvciBtZXRvZG9zIGVzdGFkaXN0aWNvcw0KYGBge3J9DQpib3hwbG90KGJkNSRQcmVjaW8sIGhvcml6b250YWw9VFJVRSkNCmJveHBsb3QoYmQ1JFVuaWRhZGVzLCBob3Jpem9udGFsPVRSVUUpDQpgYGANCiMgQWdyZWdhciBjb2x1bW5hcw0KYGBge3J9DQojIEFncmVnYXIgZGlhIGRlIGxhIHNlbWFuYQ0KYmQ1JERpYV9TZW1hbmEgPC0gd2RheShiZDUkRmVjaGEpDQojIEFncmVnYXIgZWwgc3VidG90YWwgDQpiZDUkU3VidG90YWwgPC0gYmQ1JFByZWNpbyAqIGJkNSRVbmlkYWRlcw0KYGBgDQoNCiNFeHBvcnRhciBsYSBiYXNlIGRlIGRhdG9zDQpgYGB7cn0NCiN3cml0ZS5jc3YoYmQ1LCBmaWxlPSJhYmFycm90ZXNfbGltcGkuY3N2Iiwgcm93Lm5hbWVzID0gRkFMU0UpDQpgYGANCg0KI0dlbmVyYXIgTWFya2V0IEJhc2tldCBBbmFseXNpcw0KYGBge3J9DQojT3JkZW5hciBkZSBtZW5vciBhIG1heW9yIGxhIGNvbHVtbmEgdGlja2V0DQpiZDYgPC0gYmQ1DQpiZDYgPC0gYmQ2W29yZGVyKGJkNiRGLlRpY2tldCksIF0NCiNnZW5lcmFyIGVsIGNhbmFzdG8NCmJhc2tldCA8LSBkZHBseShiZDYsYygiRi5UaWNrZXQiKSwgZnVuY3Rpb24oYmQ2KXBhc3RlKGJkNiRNYXJjYSxjb2xsYXBzZSA9ICIsIikpDQoNCiNlbGltaW5hciBjb2x1bW5hIHRpY2tldA0KYmFza2V0JEYuVGlja2V0IDwtIE5VTEwNCg0KI3Jlbm9tYnJhciBub21icmUgZGUgY29sdW1uYSBtYXJjYQ0KY29sbmFtZXMoYmFza2V0KSA8LSBjKCJNYXJjYSIpDQoNCiNleHBvcnRhciBiYXNrZXQNCndyaXRlLmNzdihiYXNrZXQsIGZpbGU9ImJhc2tldC5jc3YiLCByb3cubmFtZXMgPSBGQUxTRSkNCg0KI2ltcG9ydGFyIHRyYW5zYWNjaW9uZXMNCiNmaWxlLmNob29zZSgpDQp0ciA8LSByZWFkLnRyYW5zYWN0aW9ucygiQzpcXFVzZXJzXFxsZWNodVxcT25lRHJpdmVcXERlc2t0b3BcXGJhc2tldC5jc3YiLCBmb3JtYXQgPSAiYmFza2V0Iiwgc2VwID0gIiwiKQ0KcmVnbGFzLmFzb2NpYWNpb24gPC0gYXByaW9yaSh0ciwgcGFyYW1ldGVyID0gbGlzdChzdXBwPTAwLjAwMSwgY29uZj0wLjIsIG1heGxlbj0xMCkpDQpzdW1tYXJ5KHJlZ2xhcy5hc29jaWFjaW9uKQ0KaW5zcGVjdChyZWdsYXMuYXNvY2lhY2lvbikNCg0KI29yZGVuYXIgcmVnbGFzIGRlIGFzb2NpYWNpb24NCnJlZ2xhcy5hc29jaWFjaW9uIDwtIHNvcnQocmVnbGFzLmFzb2NpYWNpb24sIGJ5PSJjb25maWRlbmNlIiwgZGVjcmVhc2luZyA9IFRSVUUpDQoNCiNvcmRlbmFyDQp0b3AxMHJlZ2xhcyA8LSBoZWFkIChyZWdsYXMuYXNvY2lhY2lvbiwgbj0xMCwgYnk9ImNvbmZpZGVuY2UiKQ0KI3Bsb3QgKHRvcDEwcmVnbGFzLCBtZXRob2Q9ImdyYXBoIiwgZW5naW5lPSJodG1sd2lkZ2V0IikNCmBgYA0KDQpgYGB7cn0NCg0KI2ZpbGUuY2hvb3NlKCkNCmBgYA0KIyNDT05DTFVTSU9ORVMNCkVsIHVzbyBkZSBSU3R1ZGlvIG5vcyBtb3N0csOzIGxvIGltcG9ydGFudGUgcXVlIGVzIGxpbXBpYXIgbG9zIGRhdG9zIGFudGVzIGRlIGFuYWxpemFybG9zLiBBbCB0cmFiYWphciBjb24gdW5hIHRpZW5kYSBkZSBhYmFycm90ZXMsIHZpbW9zIHF1ZSBtdWNoYXMgdmVjZXMgbGEgaW5mb3JtYWNpw7NuIHRpZW5lIGVycm9yZXMgbyB2YWxvcmVzIGZhbHRhbnRlcy4gU2kgbm8gc2UgY29ycmlnZW4sIGxvcyByZXN1bHRhZG9zIHB1ZWRlbiBzZXIgaW5jb3JyZWN0b3MuIEFwcmVuZGltb3MgYSB1c2FyIGZ1bmNpb25lcyBlbiBSIHBhcmEgZWxpbWluYXIgZGF0b3MgbnVsb3MsIGNvcnJlZ2lyIGZvcm1hdG9zIHkgb3JnYW5pemFyIGxhIGluZm9ybWFjacOzbi4NCg0KVGFtYmnDqW4gYW5hbGl6YW1vcyBwYXRyb25lcyBkZSBjb21wcmEgZW50cmUgcHJvZHVjdG9zLiBOb3RhbW9zIHF1ZSBjaWVydG9zIGFydMOtY3Vsb3Mgc2UgY29tcHJhbiBqdW50b3MgY29uIGZyZWN1ZW5jaWEsIGxvIHF1ZSBlcyDDunRpbCBwYXJhIGVzdHJhdGVnaWFzIGRlIHZlbnRhLiBQb3IgZWplbXBsbywgc2kgZWwgY2Fmw6kgeSBsYXMgZ2FsbGV0YXMgc2UgdmVuZGVuIGp1bnRvcywgbGEgdGllbmRhIHBvZHLDrWEgY29sb2NhcmxvcyBjZXJjYSBwYXJhIGluY2VudGl2YXIgbcOhcyBjb21wcmFzLg0KDQpUcmFiYWphciBjb24gUlN0dWRpbyBtZWpvcsOzIG51ZXN0cmFzIGhhYmlsaWRhZGVzIGRlIHByb2dyYW1hY2nDs24gY29uIGRhdG9zIHJlYWxlcy4gQWwgcHJpbmNpcGlvLCBsb3MgY29tYW5kb3MgcHVlZGVuIHBhcmVjZXIgY29tcGxpY2Fkb3MsIHBlcm8gY29uIHByw6FjdGljYSBzZSB2dWVsdmVuIG3DoXMgZsOhY2lsZXMuIEFkZW3DoXMsIFIgbm9zIHBlcm1pdGUgdmlzdWFsaXphciBkYXRvcyBjb24gZ3LDoWZpY29zIHkgdGFibGFzIHBhcmEgZW50ZW5kZXIgbWVqb3IgbGEgaW5mb3JtYWNpw7NuLg0KDQpFbiBnZW5lcmFsLCBlc3RlIGFuw6FsaXNpcyBub3MgbW9zdHLDsyBjw7NtbyBsb3MgZGF0b3MgYXl1ZGFuIGEgdG9tYXIgbWVqb3JlcyBkZWNpc2lvbmVzIGVuIGxvcyBuZWdvY2lvcy4NCg0KIVtdKEM6XFxVc2Vyc1xcbGVjaHVcXE9uZURyaXZlXFxEZXNrdG9wXFxib290Y2FtcDEuUE5HKQ0KIVtdKEM6XFxVc2Vyc1xcbGVjaHVcXE9uZURyaXZlXFxEZXNrdG9wXFxib290Y2FtcDIucG5nKQ0KDQoNCg0K