Instalar paquetes y llamar librerías

# install.packages("tidyverse")
library(tidyverse)
# install.packages("dplyr")
library(dplyr)
# install.packages("lubridate")
library(lubridate)
# install.packages("Matrix")
library(Matrix)
# install.packages("arules")
library(arules)
# install.packages("arulesViz")
library(arulesViz)
# install.packages("datasets")
library(datasets)
# install.packages("plyr")
library(plyr)

Importar la base de datos

# file.choose()
bd <- read.csv("/Users/nataliacampos/Downloads/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
# dplyr::count(bd, vcClaveTienda, sort = TRUE)
# dplyr::count(bd, DescGiro, sort = TRUE)
# dplyr::count(bd, Fecha, sort = TRUE)
# dplyr::count(bd, Hora, sort = TRUE)
# dplyr::count(bd, Marca, sort = TRUE)
# dplyr::count(bd, Fabricante, sort = TRUE)
# dplyr::count(bd, Producto, sort = TRUE)
# dplyr::count(bd, NombreDepartamento, sort = TRUE)
# dplyr::count(bd, NombreFamilia, sort = TRUE)
# dplyr::count(bd, NombreCategoria, sort = TRUE)
# dplyr::count(bd, Estado, sort = TRUE)
# dplyr::count(bd, Tipo.ubicación, sort = TRUE)
# dplyr::count(bd, Giro , sort = TRUE)
# dplyr::count(bd, Hora.inicio, sort = TRUE)
# dplyr::count(bd, Hora.cierre, sort = TRUE)

Limpiar la base de datos

Técnica 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

Técnica 2. Remover valores duplicados

# ¿Cuántos renglones duplicados tenemos?
sum(duplicated(bd2))
## [1] 5
# ¿Cuáles 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)

Técnica 3. Eliminar errores tipográficos y similiares

# Cantidades en enteros
bd4 <- bd3
bd4$Unidades <- ceiling(bd4$Unidades)

Técnica 4. Convertir tipos de datos

# Convertir de caracter a fecha
bd5 <- bd4
bd5$Fecha <- as.Date(bd5$Fecha, format = "%d/%m/%Y")
# Convertir de caracter a entero
bd6 <- bd5
bd6$Hora <- substr(bd6$Hora, start = 1, stop = 2)
bd6$Hora <- as.integer(bd6$Hora)

Técnica 5. Reemplazar valores faltantes

# ¿Cuántos NAS tengo en la base de datos?
sum(is.na(bd6))
## [1] 0
sum(is.na(bd))
## [1] 199188
#¿Cuántos NAS tengo por 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
#Opción 1. Borrar todos los NAS de una tabla
# bd100 <- na.omit(bd)

#Opción 2. Reemplazar NAS con ceros
# bd101 <- bd
# bd101[is.n(bd101)] <- 0

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

Técnica 6. Correcciones por métodos estadísticos

boxplot(bd6$Precio, horizontal = TRUE)

boxplot(bd6$Unidades, horizontal = TRUE)

Agregar columnas

# Agregar día de la semana
bd6$Dia_Seamana <- wday(bd6$Fecha)

# Agregar el subtotal de la venta
bd6$Subtotal <- bd6$Precio *bd6$Unidades

Exportar la base de datos

# write.csv(bd6, file = "abarrotes_limpia.csv", row.names = FALSE)

Generar MArket Basket Analysis

bd7 <- bd6
bd7 <- bd7[order(bd7$F.Ticket), ]

# Generar el canasto
basket <- ddply(bd7,c("F.Ticket"), function(bd7)paste(bd7$Marca, collapse = ","))

# Eliminar la columna Ticket
basket$F.Ticket <- NULL

# Renombrar nombre de columna a Marca
colnames(basket) <- c("Marca")

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

# Importar transacciones
# file.choose()
tr <- read.transactions("/Users/nataliacampos/Desktop/basket.csv", format = "basket", sep = ",")

#Generar reglas de asociación
reglas.asociacion <- apriori(tr, parameter = list(supp=0.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.02s].
## sorting and recoding items ... [119 item(s)] done [0.00s].
## creating transaction tree ... done [0.01s].
## checking subsets of size 1 done [0.00s].
## writing ... [0 rule(s)] done [0.00s].
## creating S4 object  ... done [0.00s].
summary(reglas.asociacion)
## set of 0 rules
inspect(reglas.asociacion)

# Ordenar reglas de asociación
reglas.asociacion <- sort(reglas.asociacion, by="confidence", decreasing = TRUE)
top10reglas <- head(reglas.asociacion, n=10, by="confidence")
# plot(top10reglas, method="graph", engine="htlmwidget")

Conclusiones

El análisis de “Market Basket Analysis” se realizó para entender mejor los hábitos de compra de los clientes en tiendas de abarrotes, la idea es descubrir qué productos suelen comprarse con mayor frecuencia, que puede ayudar a estas empresas a optimizar la distribución de sus productos, y mejorar la experiencia de los consumidores. También ayuda a proporcionar información valiosa para tomar decisiones estratégicas basadas en los datos de consumo.

En este análisis se trabajó con la base de datos que tiene información sobre ventas, productos, precios y horarios, lo que ayudó a tener una idea clara del comportamiento de los clientes. Luego se limpió la información para corregir errores y asegurarse de que todo estuviera en orden. Por último, se crearon gráficos y tablas que muestran de forma sencilla cuáles son las combinaciones de productos más comunes y se identificaron asociaciones clave, como la tendencia de los clientes a comprar productos de limpieza como SALVO junto con FABULOSO, o la compra frecuente de COCA COLA junto con la versión tradicional de la bebida, en donde, se generaron gráficos para facilitar la visualización de estas conexiones, para interpretar de mejor manera los resultados y sus implicaciones para la toma de decisiones estratégicas en las tiendas.

Este análisis proporciona información clave que puede ayudar a mejorar la organización de los productos en las tiendas, crear promociones atractivas y ofrecer una mejor experiencia de compra a los clientes. Al final de cuentas, se trata de mejorar la experiencia de los clientes y vender de forma más inteligente dependiendo las preferencias de su mercado.

LS0tCnRpdGxlOiAiTWFya2V0IEJhc2tldCBBbmFseXNpcyIKYXV0aG9yOiAiTmF0YWxpYSBDYW1wb3MgUnVpeiBBMDA4MzU1NjkiCmRhdGU6ICIyMDI1LTAzLTE5IgpvdXRwdXQ6CiAgaHRtbF9kb2N1bWVudDogCiAgICB0b2M6IFRSVUUKICAgIHRvY19mbG9hdDogVFJVRQogICAgY29kZV9kb3dubG9hZDogIFRSVUUKICAgIHRoZW1lOiAieWV0aSIgIAotLS0KCiFbXShodHRwczovL2kucGluaW1nLmNvbS9vcmlnaW5hbHMvMDAvNjEvMzYvMDA2MTM2ZWEzMWI2YTI5YmZhYjg1YmMwMjBiNjliZWEuZ2lmKQoKIyBJbnN0YWxhciBwYXF1ZXRlcyB5IGxsYW1hciBsaWJyZXLDrWFzCmBgYHtyIG1lc3NhZ2U9RkFMU0UsIHdhcm5pbmc9RkFMU0V9CiMgaW5zdGFsbC5wYWNrYWdlcygidGlkeXZlcnNlIikKbGlicmFyeSh0aWR5dmVyc2UpCiMgaW5zdGFsbC5wYWNrYWdlcygiZHBseXIiKQpsaWJyYXJ5KGRwbHlyKQojIGluc3RhbGwucGFja2FnZXMoImx1YnJpZGF0ZSIpCmxpYnJhcnkobHVicmlkYXRlKQojIGluc3RhbGwucGFja2FnZXMoIk1hdHJpeCIpCmxpYnJhcnkoTWF0cml4KQojIGluc3RhbGwucGFja2FnZXMoImFydWxlcyIpCmxpYnJhcnkoYXJ1bGVzKQojIGluc3RhbGwucGFja2FnZXMoImFydWxlc1ZpeiIpCmxpYnJhcnkoYXJ1bGVzVml6KQojIGluc3RhbGwucGFja2FnZXMoImRhdGFzZXRzIikKbGlicmFyeShkYXRhc2V0cykKIyBpbnN0YWxsLnBhY2thZ2VzKCJwbHlyIikKbGlicmFyeShwbHlyKQpgYGAKCiMgSW1wb3J0YXIgbGEgYmFzZSBkZSBkYXRvcwpgYGB7cn0KIyBmaWxlLmNob29zZSgpCmJkIDwtIHJlYWQuY3N2KCIvVXNlcnMvbmF0YWxpYWNhbXBvcy9Eb3dubG9hZHMvYWJhcnJvdGVzLmNzdiIpCmBgYAoKIyBFbnRlbmRlciBsYSBiYXNlIGRlIGRhdG9zCmBgYHtyfQpzdW1tYXJ5KGJkKQpzdHIoYmQpCmhlYWQoYmQsIG49MTApCnRhaWwoYmQsIG49MTApCiMgZHBseXI6OmNvdW50KGJkLCB2Y0NsYXZlVGllbmRhLCBzb3J0ID0gVFJVRSkKIyBkcGx5cjo6Y291bnQoYmQsIERlc2NHaXJvLCBzb3J0ID0gVFJVRSkKIyBkcGx5cjo6Y291bnQoYmQsIEZlY2hhLCBzb3J0ID0gVFJVRSkKIyBkcGx5cjo6Y291bnQoYmQsIEhvcmEsIHNvcnQgPSBUUlVFKQojIGRwbHlyOjpjb3VudChiZCwgTWFyY2EsIHNvcnQgPSBUUlVFKQojIGRwbHlyOjpjb3VudChiZCwgRmFicmljYW50ZSwgc29ydCA9IFRSVUUpCiMgZHBseXI6OmNvdW50KGJkLCBQcm9kdWN0bywgc29ydCA9IFRSVUUpCiMgZHBseXI6OmNvdW50KGJkLCBOb21icmVEZXBhcnRhbWVudG8sIHNvcnQgPSBUUlVFKQojIGRwbHlyOjpjb3VudChiZCwgTm9tYnJlRmFtaWxpYSwgc29ydCA9IFRSVUUpCiMgZHBseXI6OmNvdW50KGJkLCBOb21icmVDYXRlZ29yaWEsIHNvcnQgPSBUUlVFKQojIGRwbHlyOjpjb3VudChiZCwgRXN0YWRvLCBzb3J0ID0gVFJVRSkKIyBkcGx5cjo6Y291bnQoYmQsIFRpcG8udWJpY2FjacOzbiwgc29ydCA9IFRSVUUpCiMgZHBseXI6OmNvdW50KGJkLCBHaXJvICwgc29ydCA9IFRSVUUpCiMgZHBseXI6OmNvdW50KGJkLCBIb3JhLmluaWNpbywgc29ydCA9IFRSVUUpCiMgZHBseXI6OmNvdW50KGJkLCBIb3JhLmNpZXJyZSwgc29ydCA9IFRSVUUpCmBgYAoKIyBMaW1waWFyIGxhIGJhc2UgZGUgZGF0b3MKCiMjIFTDqWNuaWNhIDEuIFJlbW92ZXIgdmFsb3JlcyBpcnJlbGV2YW50ZXMKYGBge3J9CiMgRWxpbWluYXIgY29sdW1uYXMKYmQxIDwtIGJkCmJkMSA8LSBzdWJzZXQoYmQxLCBzZWxlY3QgPSAtYyhQTFUsIENvZGlnby5CYXJyYXMpKQoKIyBFbGltaW5hciByZW5nbG9uZXMKYmQyIDwtIGJkMQpiZDIgPC0gYmQyW2JkMiRQcmVjaW8gPiAwLCBdIApzdW1tYXJ5KGJkMSRQcmVjaW8pCnN1bW1hcnkoYmQyJFByZWNpbykKYGBgCgojIyBUw6ljbmljYSAyLiBSZW1vdmVyIHZhbG9yZXMgZHVwbGljYWRvcwpgYGB7cn0KIyDCv0N1w6FudG9zIHJlbmdsb25lcyBkdXBsaWNhZG9zIHRlbmVtb3M/CnN1bShkdXBsaWNhdGVkKGJkMikpCgojIMK/Q3XDoWxlcyBzb24gbG9zIHJlbmdsb25lcyBkdXBsaWNhZG9zPwpiZDJbZHVwbGljYXRlZChiZDIpLCBdCgojIEVsaW1pbmFyIHJlbmdsb25lcyBkdXBsaWNhZG9zCmJkMyA8LSBiZDIKYmQzIDwtIGRpc3RpbmN0KGJkMykKYGBgCgojIyBUw6ljbmljYSAzLiBFbGltaW5hciBlcnJvcmVzIHRpcG9ncsOhZmljb3MgeSBzaW1pbGlhcmVzCmBgYHtyfQojIENhbnRpZGFkZXMgZW4gZW50ZXJvcwpiZDQgPC0gYmQzCmJkNCRVbmlkYWRlcyA8LSBjZWlsaW5nKGJkNCRVbmlkYWRlcykKYGBgCgojIyBUw6ljbmljYSA0LiBDb252ZXJ0aXIgdGlwb3MgZGUgZGF0b3MKYGBge3J9CiMgQ29udmVydGlyIGRlIGNhcmFjdGVyIGEgZmVjaGEKYmQ1IDwtIGJkNApiZDUkRmVjaGEgPC0gYXMuRGF0ZShiZDUkRmVjaGEsIGZvcm1hdCA9ICIlZC8lbS8lWSIpCiMgQ29udmVydGlyIGRlIGNhcmFjdGVyIGEgZW50ZXJvCmJkNiA8LSBiZDUKYmQ2JEhvcmEgPC0gc3Vic3RyKGJkNiRIb3JhLCBzdGFydCA9IDEsIHN0b3AgPSAyKQpiZDYkSG9yYSA8LSBhcy5pbnRlZ2VyKGJkNiRIb3JhKQpgYGAKCiMjIFTDqWNuaWNhIDUuIFJlZW1wbGF6YXIgdmFsb3JlcyBmYWx0YW50ZXMgCmBgYHtyfQojIMK/Q3XDoW50b3MgTkFTIHRlbmdvIGVuIGxhIGJhc2UgZGUgZGF0b3M/CnN1bShpcy5uYShiZDYpKQpzdW0oaXMubmEoYmQpKQoKI8K/Q3XDoW50b3MgTkFTIHRlbmdvIHBvciB2YXJpYWJsZT8Kc2FwcGx5KGJkLCBmdW5jdGlvbih4KSBzdW0oaXMubmEoeCkpKQoKI09wY2nDs24gMS4gQm9ycmFyIHRvZG9zIGxvcyBOQVMgZGUgdW5hIHRhYmxhCiMgYmQxMDAgPC0gbmEub21pdChiZCkKCiNPcGNpw7NuIDIuIFJlZW1wbGF6YXIgTkFTIGNvbiBjZXJvcwojIGJkMTAxIDwtIGJkCiMgYmQxMDFbaXMubihiZDEwMSldIDwtIDAKCiNPcGNpw7NuIDMuIFJlZW1wbGF6YXIgbG9zIE5BcyBjb24gZWwgcHJvbWVkaW8KIyBiZDEwMiA8LSBiZAojIGJkMTAyJFBMVVtpcy5uYShiZDEwMiRQTFUpIDwtIG1lYW4oYmQxMDIkUExVLCBuYS5ybSA9IFRSVUUpXQpgYGAKCiMjIFTDqWNuaWNhIDYuIENvcnJlY2Npb25lcyBwb3IgbcOpdG9kb3MgZXN0YWTDrXN0aWNvcwpgYGB7cn0KYm94cGxvdChiZDYkUHJlY2lvLCBob3Jpem9udGFsID0gVFJVRSkKYm94cGxvdChiZDYkVW5pZGFkZXMsIGhvcml6b250YWwgPSBUUlVFKQpgYGAKCiMgQWdyZWdhciBjb2x1bW5hcwpgYGB7cn0KIyBBZ3JlZ2FyIGTDrWEgZGUgbGEgc2VtYW5hCmJkNiREaWFfU2VhbWFuYSA8LSB3ZGF5KGJkNiRGZWNoYSkKCiMgQWdyZWdhciBlbCBzdWJ0b3RhbCBkZSBsYSB2ZW50YQpiZDYkU3VidG90YWwgPC0gYmQ2JFByZWNpbyAqYmQ2JFVuaWRhZGVzCmBgYAoKIyBFeHBvcnRhciBsYSBiYXNlIGRlIGRhdG9zCmBgYHtyfQojIHdyaXRlLmNzdihiZDYsIGZpbGUgPSAiYWJhcnJvdGVzX2xpbXBpYS5jc3YiLCByb3cubmFtZXMgPSBGQUxTRSkKYGBgCgojIEdlbmVyYXIgTUFya2V0IEJhc2tldCBBbmFseXNpcwpgYGB7cn0KYmQ3IDwtIGJkNgpiZDcgPC0gYmQ3W29yZGVyKGJkNyRGLlRpY2tldCksIF0KCiMgR2VuZXJhciBlbCBjYW5hc3RvCmJhc2tldCA8LSBkZHBseShiZDcsYygiRi5UaWNrZXQiKSwgZnVuY3Rpb24oYmQ3KXBhc3RlKGJkNyRNYXJjYSwgY29sbGFwc2UgPSAiLCIpKQoKIyBFbGltaW5hciBsYSBjb2x1bW5hIFRpY2tldApiYXNrZXQkRi5UaWNrZXQgPC0gTlVMTAoKIyBSZW5vbWJyYXIgbm9tYnJlIGRlIGNvbHVtbmEgYSBNYXJjYQpjb2xuYW1lcyhiYXNrZXQpIDwtIGMoIk1hcmNhIikKCiMgRXhwb3J0YXIgYmFza2V0CiMgd3JpdGUuY3N2KGJhc2tldCwgZmlsZSA9ICJiYXNrZXQuY3N2Iiwgcm93Lm5hbWVzID0gRkFMU0UpCgojIEltcG9ydGFyIHRyYW5zYWNjaW9uZXMKIyBmaWxlLmNob29zZSgpCnRyIDwtIHJlYWQudHJhbnNhY3Rpb25zKCIvVXNlcnMvbmF0YWxpYWNhbXBvcy9EZXNrdG9wL2Jhc2tldC5jc3YiLCBmb3JtYXQgPSAiYmFza2V0Iiwgc2VwID0gIiwiKQoKI0dlbmVyYXIgcmVnbGFzIGRlIGFzb2NpYWNpw7NuCnJlZ2xhcy5hc29jaWFjaW9uIDwtIGFwcmlvcmkodHIsIHBhcmFtZXRlciA9IGxpc3Qoc3VwcD0wLjAwMSwgY29uZj0wLjIsIG1heGxlbj0xMCkpCnN1bW1hcnkocmVnbGFzLmFzb2NpYWNpb24pCmluc3BlY3QocmVnbGFzLmFzb2NpYWNpb24pCgojIE9yZGVuYXIgcmVnbGFzIGRlIGFzb2NpYWNpw7NuCnJlZ2xhcy5hc29jaWFjaW9uIDwtIHNvcnQocmVnbGFzLmFzb2NpYWNpb24sIGJ5PSJjb25maWRlbmNlIiwgZGVjcmVhc2luZyA9IFRSVUUpCnRvcDEwcmVnbGFzIDwtIGhlYWQocmVnbGFzLmFzb2NpYWNpb24sIG49MTAsIGJ5PSJjb25maWRlbmNlIikKIyBwbG90KHRvcDEwcmVnbGFzLCBtZXRob2Q9ImdyYXBoIiwgZW5naW5lPSJodGxtd2lkZ2V0IikKCmBgYAoKIVtdKC9Vc2Vycy9uYXRhbGlhY2FtcG9zL0Rvd25sb2Fkcy9yZWdsYXMuUE5HKQohW10oL1VzZXJzL25hdGFsaWFjYW1wb3MvRG93bmxvYWRzL2dyYWZpY28ucG5nKQoKCiMgQ29uY2x1c2lvbmVzCgpFbCBhbsOhbGlzaXMgZGUgIk1hcmtldCBCYXNrZXQgQW5hbHlzaXMiIHNlIHJlYWxpesOzIHBhcmEgZW50ZW5kZXIgbWVqb3IgbG9zIGjDoWJpdG9zIGRlIGNvbXByYSBkZSBsb3MgY2xpZW50ZXMgZW4gdGllbmRhcyBkZSBhYmFycm90ZXMsIGxhIGlkZWEgZXMgZGVzY3VicmlyIHF1w6kgcHJvZHVjdG9zIHN1ZWxlbiBjb21wcmFyc2UgY29uIG1heW9yIGZyZWN1ZW5jaWEsIHF1ZSBwdWVkZSBheXVkYXIgYSBlc3RhcyBlbXByZXNhcyBhIG9wdGltaXphciBsYSBkaXN0cmlidWNpw7NuIGRlIHN1cyBwcm9kdWN0b3MsIHkgbWVqb3JhciBsYSBleHBlcmllbmNpYSBkZSBsb3MgY29uc3VtaWRvcmVzLiBUYW1iacOpbiBheXVkYSBhIHByb3BvcmNpb25hciBpbmZvcm1hY2nDs24gdmFsaW9zYSBwYXJhIHRvbWFyIGRlY2lzaW9uZXMgZXN0cmF0w6lnaWNhcyBiYXNhZGFzIGVuIGxvcyBkYXRvcyBkZSBjb25zdW1vLgoKRW4gZXN0ZSBhbsOhbGlzaXMgc2UgdHJhYmFqw7MgY29uIGxhIGJhc2UgZGUgZGF0b3MgcXVlIHRpZW5lIGluZm9ybWFjacOzbiBzb2JyZSAgdmVudGFzLCBwcm9kdWN0b3MsIHByZWNpb3MgeSBob3JhcmlvcywgbG8gcXVlIGF5dWTDsyBhIHRlbmVyIHVuYSBpZGVhIGNsYXJhIGRlbCBjb21wb3J0YW1pZW50byBkZSBsb3MgY2xpZW50ZXMuIEx1ZWdvIHNlIGxpbXBpw7MgbGEgaW5mb3JtYWNpw7NuIHBhcmEgY29ycmVnaXIgZXJyb3JlcyB5IGFzZWd1cmFyc2UgZGUgcXVlIHRvZG8gZXN0dXZpZXJhIGVuIG9yZGVuLiBQb3Igw7psdGltbywgc2UgY3JlYXJvbiBncsOhZmljb3MgeSB0YWJsYXMgcXVlIG11ZXN0cmFuIGRlIGZvcm1hIHNlbmNpbGxhIGN1w6FsZXMgc29uIGxhcyBjb21iaW5hY2lvbmVzIGRlIHByb2R1Y3RvcyBtw6FzIGNvbXVuZXMgeSBzZSBpZGVudGlmaWNhcm9uIGFzb2NpYWNpb25lcyBjbGF2ZSwgY29tbyBsYSB0ZW5kZW5jaWEgZGUgbG9zIGNsaWVudGVzIGEgY29tcHJhciBwcm9kdWN0b3MgZGUgbGltcGllemEgY29tbyBTQUxWTyBqdW50byBjb24gRkFCVUxPU08sIG8gbGEgY29tcHJhIGZyZWN1ZW50ZSBkZSBDT0NBIENPTEEganVudG8gY29uIGxhIHZlcnNpw7NuIHRyYWRpY2lvbmFsIGRlIGxhIGJlYmlkYSwgZW4gZG9uZGUsIHNlIGdlbmVyYXJvbiBncsOhZmljb3MgcGFyYSBmYWNpbGl0YXIgbGEgdmlzdWFsaXphY2nDs24gZGUgZXN0YXMgY29uZXhpb25lcywgcGFyYSBpbnRlcnByZXRhciBkZSBtZWpvciBtYW5lcmEgbG9zIHJlc3VsdGFkb3MgeSBzdXMgaW1wbGljYWNpb25lcyBwYXJhIGxhIHRvbWEgZGUgZGVjaXNpb25lcyBlc3RyYXTDqWdpY2FzIGVuIGxhcyB0aWVuZGFzLgoKRXN0ZSBhbsOhbGlzaXMgcHJvcG9yY2lvbmEgaW5mb3JtYWNpw7NuIGNsYXZlIHF1ZSBwdWVkZSBheXVkYXIgYSBtZWpvcmFyIGxhIG9yZ2FuaXphY2nDs24gZGUgbG9zIHByb2R1Y3RvcyBlbiBsYXMgdGllbmRhcywgY3JlYXIgcHJvbW9jaW9uZXMgYXRyYWN0aXZhcyB5IG9mcmVjZXIgdW5hIG1lam9yIGV4cGVyaWVuY2lhIGRlIGNvbXByYSBhIGxvcyBjbGllbnRlcy4gQWwgZmluYWwgZGUgY3VlbnRhcywgc2UgdHJhdGEgZGUgbWVqb3JhciBsYSBleHBlcmllbmNpYSBkZSBsb3MgY2xpZW50ZXMgeSB2ZW5kZXIgZGUgZm9ybWEgbcOhcyBpbnRlbGlnZW50ZSBkZXBlbmRpZW5kbyBsYXMgcHJlZmVyZW5jaWFzIGRlIHN1IG1lcmNhZG8uCgoKCgoKCg==