TeorĆ­a

La minería de datos de texto (TM) es el proceso de extraer información útil, patrones o conocimiento de textos no estructurados.

Consta de tres etapas:
1. Obtener datos: El Reconocimiento Ɠptico de CaractĆ©res (OCR) es una tecnologĆ­a que permite convertir imĆ”genes de texto en texto editable. TambiĆ©n es conocido como extracción de texto de imĆ”genes.
2. Explorar datos: Representación grÔfica o visual de los datos para su interpretación. Los métodos mÔs comunes son el AnÔlisis de Sentimientos, la Nube de Palabras y el Topic Modeling.
3. AnƔlisis predictivo: Son las tƩcnicas y modelos estadƭsticos para predecri resultados futuros. Los modelos mƔs usados son el Random Forest, redes neuronales y regresiones.

#install.packages("tidyverse") #Manipulación de datos
library(tidyverse)
#install.packages("tesseract") # OCR
library(tesseract)
#install.packages("magick") # PNG
library(magick)
#install.packages("officer") # Office "Word"
library(officer)
#install.packages("pdftools") # PDF
library(pdftools)
#install.packages("purrr") # PDF
library(purrr) 

Etapa 1. Obtener Datos mediante OCR

De imagen PNG a texto en Word

imagen1 <- image_read("C:\\Users\\Usuario\\Documents\\IA Concentración\\M2\\Text Mining\\imagen1 (1).PNG")
texto1 <- ocr(imagen1)
texto1
## [1] "Linear regression with one variable x is also known as univariate linear regression\nor simple linear regression. Simple linear regression is used to predict a single\noutput from a single input. This is an example of supervised learning, which means\nthat the data is labeled, i.e., the output values are known in the training data. Let us\nfit a line through the data using simple linear regression as shown in Fig. 4.1.\n"
doc1 <- read_docx() #Crea un documento de Word en blanco
doc1 <- doc1 %>% body_add_par(texto1) # Pega el texto en el doc
print(doc1, target="texto1.docx") # Guarda el doc en la compu con un nombre asignado

De imagen PNG en espaƱol a texto en Word

imagen2 <- image_read("C:\\Users\\Usuario\\Documents\\IA Concentración\\M2\\Text Mining\\imagen2.PNG")
tesseract_download("spa")
## Training data already exists. Overwriting C:\Users\Usuario\AppData\Local\tesseract5\tesseract5\tessdata/spa.traineddata
## [1] "C:\\Users\\Usuario\\AppData\\Local\\tesseract5\\tesseract5\\tessdata/spa.traineddata"
texto2 <- ocr(imagen2, engine = tesseract("spa"))
texto2
## [1] "Un importante, y quizĆ” controversial, asunto polĆ­tico es el que se refiere al efecto del salario mĆ­nimo sobre\nlas tasas de desempleo en diversos grupos de trabajadores. Aunque este problema puede ser estudiado con\ndiversos tipos de datos (corte transversal, series de tiempo o datos de panel), suelen usarse las series de\ntiempo para observar los efectos agregados. En la tabla 1.3 se presenta un ejemplo de una base de datos\nde series de tiempo sobre tasas de desempleo y salarios mĆ­nimos.\n"
doc2 <- read_docx() #Crea un documento de Word en blanco
doc2 <- doc2 %>% body_add_par(texto2) # Pega el texto en el doc
#print(doc2, target="texto2.docx") # Guarda el doc en la compu con un nombre asignado

Actividad 1. Eso

#De pdf a texto en word
pdf_eso <- pdf_convert("C:\\Users\\Usuario\\Documents\\IA Concentración\\M2\\Text Mining\\eso.pdf", dpi=600) %>% map(ocr)
## Converting page 1 to eso_1.png... done!
## Converting page 2 to eso_2.png... done!

Transformar pdf a png en espaƱol

imagen3 <- image_read("C:\\Users\\Usuario\\Documents\\IA Concentración\\M2\\Text Mining\\eso_1.PNG")
tesseract_download("spa")
## Training data already exists. Overwriting C:\Users\Usuario\AppData\Local\tesseract5\tesseract5\tessdata/spa.traineddata
## [1] "C:\\Users\\Usuario\\AppData\\Local\\tesseract5\\tesseract5\\tessdata/spa.traineddata"
texto3 <- ocr(imagen3, engine = tesseract("spa"))
texto3
## [1] "I. DESPUƉS DE LA INUNDACIƓN (1957)\nEl terror, que no terminarĆ­a por otros veintiocho aƱos —si es que terminó alguna vez—,\ncomenzó, hasta donde sĆ© o puedo contar, con un barco hecho de una hoja de un diario\nque flotaba a lo largo del arroyo de una calle anegada de lluvia.\nEl barquito cabeceo, se ladeó, volvió a enderezarse en medio de traicioneros remolinos y\ncontinuó su marcha por Witcham Street hacia el semĆ”foro que marcaba la intersección\nde Ć©sta y Jackson. Las tres lentes verticales a los lados del semĆ”foro estaban a oscuras y\ntambiĆ©n todas las casas, en aquella tarde de otoƱo de 1957. LlovĆ­a sin cesar desde hacia\nya una semana y dos dias atrĆ”s habian llegado tambiĆ©n los vientos. Desde entonces, la\nmayor parte de Derry habia quedado sin corriente electrica y aĆŗn seguĆ­a asi.\nUn chiquillo de impermeable amarillo y botas rojas seguĆ­a alegremente al barco de\npapel. La lluvia no habĆ­a cesado, pero al fin estaba amainando. Golpeteaba sobre la\ncapucha amarilla del impermeable sonando a los oidos del niƱo como lluvia sobre el\ntejado de un cobertizo.. un sonido reconfortante, casi acogedor. El niƱo del impermeable\namarillo era George Denbrough. Tenia seis aƱos. William, su hermano, a quien casi todos\nlos niƱos de la escuela primaria de Derry (y hasta los maestros, aunque jamĆ”s habrian\nusado el apodo frente a Ć©l) conocĆ­an como Bill el Tartaja, estaba en su casa pasando los\nrestos de una gripe bastante seria. En ese otoƱo de 1957, ocho meses antes de que\ncomenzasen realmente los horrores y veintiocho aƱos antes del desenlace final, Bill el\nTartaja tenĆ­a diez aƱos.\nEra Bill quien habĆ­a hecho el barquito junto al cual corria George. Lo habia hecho sentado\nen su cama, con la espalda apoyada en un montón de almohadas, mientras la madre\ntocaba Para Elisa en el piano de la sala y la lluvia barrĆ­a incansablemente la ventana de su\ndormitorio.\nA un tercio de manzana, camino de la intersección y del semĆ”foro apagado, Witcham\nStreet estaba cerrada al trĆ”fico por varios toneles de brea y cuatro caballetes color\nnaranja. En cada uno de esos caballetes se leĆ­a: AYUNTAMIENTO DE DERRY —\nDEPARTAMENTO DE OBRAS PUBLICAS. Tras ellos, la lluvia habĆ­a desbordado\nalcantarillas atascadas con ramas, piedras y cĆŗumulos de pegajosas hojas otonales. El\nagua habia ido picando el pavimento al principio, arrancado luego grandes trozos\ncodiciosos; todo esto, hacia el tercer dĆ­a de las lluvias. Hacia el mediodĆ­a de la cuarta\njornada, grandes trozos de pavimento eran arrastrados por la intersección de Jackson y\nWitecham como tempanos de hielo en minlatura. Muchos habitantes de Derry habian\nempezado por entonces a hacer chistes nerviosos sobre el Arca. El Departamento de\nObras PĆŗblicas se las habĆ­a arreglado para mantener ablerta Jackson Street, pero\nWitcham estaba intransitable desde las barreras hasta el centro mismo de la ciudad.\nTodos estaban de acuerdo, sin embargo, en que lo peor habĆ­a pasado. El rio Kenduskeag\nhabia crecido casi hasta sus mĆ”rgenes en los erlales y hasta muy pocos centimetros por\ndebajo de los muros de cemento del canal que constreƱƭa su paso por el centro de la\ncludad. En esos momentos, un grupo de hombres —entre ellos Zack Denbrough, el padre\nde George y de Bill — estaba retirando los sacos de arena que habian lanzado el dĆ­a\nanterior con aterrorizada prisa. Un dĆ­a antes, la inundación y sus costosos daƱos habĆ­an\nparecido casi inevitables. Bien sabia Dios que ya habia ocurrido anteriormente —la\n"
doc3 <- read_docx() #Crea un documento de Word en blanco
doc3 <- doc3 %>% body_add_par(texto3) # Pega el texto en el doc
print(doc3, target="texto3.docx") # Guarda el doc en la compu con un nombre asignado
imagen4 <- image_read("C:\\Users\\Usuario\\Documents\\IA Concentración\\M2\\Text Mining\\eso_2.png")
tesseract_download("spa")
## Training data already exists. Overwriting C:\Users\Usuario\AppData\Local\tesseract5\tesseract5\tessdata/spa.traineddata
## [1] "C:\\Users\\Usuario\\AppData\\Local\\tesseract5\\tesseract5\\tessdata/spa.traineddata"
texto4 <- ocr(imagen4, engine = tesseract("spa"))
texto4
## [1] "inundación de 1931 habia sido un desastre con un costo de millones de dólares y de mas\nde veinte vidas—. De aquello hacia ya mucho tiempo, pero aĆŗn quedaba gente por ahi\nque lo recordaba para asustar al resto. Una de las victimas de la inundación habĆ­a sido\nhallada en Bucksport, a unos cuarenta kilómetros de distancia. Los peces le habĆ­an\ncomido a ese infortunado caballero los ojos, tres dedos, el pene y la mayor parte del pie\nizquierdo. Agarrado por lo que restaba de sus manos, habia aparecido el volante de un\nFord.\n\nAhora, sin embargo, el rio estaba retrocediendo y cuando se elevara la nueva presa\nhidraulica de Bangor, corriente arriba, dejaria de ser una amenaza. Al menos eso decia\nZack Denbrough, que trabajaba en HidroelĆ©ctrica Bangor. En cuanto a los demĆ”s.. bueno,\nlas inundaciones futuras esperarian. Lo importante era salir de Ć©sta, devolver la corriente\nelĆ©ctrica y despuĆ©s olvidarla. En Derry, eso de olvidar la tragedia y el desastre era casi un\narte, tal como Bill Denbrough llegarĆ­a a descubrir con el tiempo.\n\nGeorge se detuvo justo detrĆ”s de las barreras al borde de una profunda grieta que se\nhabĆ­a abierto en la superficie de alquitrĆ”n de Witcham Street. Este barranco discurrĆ­a casi\nexactamente en diagonal. Terminaba al otro extremo de la calle, a unos doce metros de\ndonde Ć©l se encontraba, colina abajo hacia la derecha. Rió en voz alta —el sonido de la\nsolitaria alegrĆ­a infantil salvando metas en aquella tarde gris—, mientras un capricho del\nagua desbordada llevaba su barco de papel hasta unas cataratas a escala formadas por\notra grieta en el pavimento. El agua habĆ­a abierto con su urgencia un canal que corrĆ­a a lo\nlargo de la diagonal y por ello el barco iba de un lado a otro de la calle arrastrado tan\ndeprisa por la corriente que George tuvo que correr para seguirlo. El agua se extendĆ­a\nbajo sus botas, formando lĆ”minas de lodo. Sus hebillas sonaban con un jubiloso tintineo\nmientras George Denbrough corrĆ­a hacia su extraƱa muerte. Y el sentimiento que le\ncolmaba en ese momento era, clara y simplemente, amor hacia su hermano.., amor y\ntambiĆ©n una cierta tristeza porque Bill no podĆ­a estar alli para ver aquello y compartirlo.\nClaro que el tratarĆ­a de describirselo cuando volviese a casa, pero sabia que jamas podrĆ­a\nhacer que Bill lo viese, tal como Bill se lo hubiese hecho ver a el en situación inversa. Bill\ndestacaba en lectura y redacción, pero aun a su edad George tenĆ­a capacidad suficiente\ncomo para comprender que no sólo por eso obtenĆ­a Bill las mejores notas; tampoco era\nel Ćŗnico motivo de que a los maestros les gustaran tanto sus composiciones. La forma de\ncontar era sólo una parte del asunto. Bill sabia ver.\n\nEl barquito casi silbaba a lo largo de aquel canal, sólo una pĆ”gina arrancada de la sección\nde anuncios clasificados del News de Derry, pero George lo imaginaba como una\ntorpedera en una pelĆ­cula de guerra de esas que solĆ­a ver en el Teatro Derry con Bill, en\nlas matinĆ©es de los sabados. Una pelicula de guerra en la que Jonn Wayne luchaba contra\nlos japoneses. La proa del barco de papel levantaba olas a cada lado mientras seguĆ­a su\nprecipitado curso hacia la cuneta del lado izquierdo de la calle. En ese punto, un nuevo\narroyuelo corria sobre la grieta abierta en el pavimento creando un remolino bastante\ngrande. George pensó que el barco volcarĆ­a yendose a pique. Escoró de modo alarmante\npero luego se enderezó, giró y navegó rapidamente hacia la intersección. George lanzó\ngritos de jĆŗbilo y corrió para alcanzarlo. Sobre su cabeza, una torva rĆ”faga de viento\notoƱal hizo silbar los Ć”rboles, casi completamente liberados de su carga de hojas a causa\nde la tormenta, que ese aƱo habĆ­a sido un segador implacable.\n"
doc4 <- read_docx() # Crea un documento de Word en blanco
doc4 <- doc4 %>% body_add_par(texto4) # Pega el texto en el doc
print(doc4, target="texto4.docx") # Guarda el doc en la compu con un nombre asignado

Juntar los dos archivos de texto transformados a un solo docuemnto de word

doc_eso <- read_docx()
doc_eso <- doc_eso %>% body_add_par(texto3) %>% body_add_par(texto4)
print(doc_eso, target="doc_eso.docx")

Etapa 2. Explorar datos mediante anƔlisis de sentimientos

Instalar paquete sy llamra librerĆ­as

#install.packages("xml2", dependencies = TRUE)
#install.packages("NLP", dependencies = TRUE)
#install.packages("tm", dependencies = TRUE)

#install.packages("syuzhet") #AnƔlisis de sentimientos
library(syuzhet)
#install.packages("tm") #MinerĆ­a de texto
library(tm)
## Cargando paquete requerido: NLP
## 
## Adjuntando el paquete: 'NLP'
## The following object is masked from 'package:ggplot2':
## 
##     annotate
#install.packages("wordcloud") #MinerĆ­a de texto
library(wordcloud)
## Cargando paquete requerido: RColorBrewer
#install.packages("RColorBrewer") #MinerĆ­a de texto
library(RColorBrewer)

AnƔlisis de emociones y sentimientos

texto <- pdf_eso
texto_palabras <- get_tokens(texto) # Separar texto por palabras, tokens
emociones <- get_nrc_sentiment(texto_palabras, language = "spanish")
# Alegría, tristeza, ira, miedo, sorpresa, asco, anticipación y confianza
barplot(colSums(prop.table(emociones[,1:8])))

sentimientos <- (emociones$negative*-1)+emociones$positive
simple_plot(sentimientos)

Nube de palabras

palabras <- texto_palabras
palabras <- removeWords(palabras, c(stopwords("spanish"),"hacia", "habia", "habian", "hecho"))
wordcloud(words=palabras, min.freq=2, rot.per = 0, random.order=FALSE )
## Warning in tm_map.SimpleCorpus(corpus, tm::removePunctuation): transformation
## drops documents
## Warning in tm_map.SimpleCorpus(corpus, function(x) tm::removeWords(x,
## tm::stopwords())): transformation drops documents

LS0tDQp0aXRsZTogIlRleHQgTWluaW5nIg0KYXV0aG9yOiAiSWtlciBWaWxsYWZhw7FhIC0gQTAwNTczNzU2Ig0KZGF0ZTogIjIwMjUtMDItMTciDQpvdXRwdXQ6IA0KICBodG1sX2RvY3VtZW50Og0KICAgIHRvYzogdHJ1ZQ0KICAgIHRvY19mbG9hdDogdHJ1ZQ0KICAgIGNvZGVfZG93bmxvYWQ6IHRydWUNCiAgICB0aGVtZTogY29zbW8NCi0tLQ0KDQohW10oQzpcVXNlcnNcVXN1YXJpb1xEb3dubG9hZHNcbWF4cmVzZGVmYXVsdC5qcGcpDQoNCiMgW1Rlb3LDrWFde3N0eWxlPSJjb2xvcjogcmVkOyJ9DQoNCkxhICoqbWluZXLDrWEgZGUgZGF0b3MgZGUgdGV4dG8gKFRNKSoqIGVzIGVsIHByb2Nlc28gZGUgZXh0cmFlciBpbmZvcm1hY2nDs24gw7p0aWwsIHBhdHJvbmVzIG8gY29ub2NpbWllbnRvIGRlIHRleHRvcyBubyBlc3RydWN0dXJhZG9zLg0KDQpDb25zdGEgZGUgdHJlcyBldGFwYXM6XA0KMS4gT2J0ZW5lciBkYXRvczogRWwgKipSZWNvbm9jaW1pZW50byDDk3B0aWNvIGRlIENhcmFjdMOpcmVzIChPQ1IpKiogZXMgdW5hIHRlY25vbG9nw61hIHF1ZSBwZXJtaXRlIGNvbnZlcnRpciBpbcOhZ2VuZXMgZGUgdGV4dG8gZW4gdGV4dG8gZWRpdGFibGUuIFRhbWJpw6luIGVzIGNvbm9jaWRvIGNvbW8gKipleHRyYWNjacOzbiBkZSB0ZXh0byBkZSBpbcOhZ2VuZXMqKi5cDQoyLiBFeHBsb3JhciBkYXRvczogUmVwcmVzZW50YWNpw7NuIGdyw6FmaWNhIG8gdmlzdWFsIGRlIGxvcyBkYXRvcyBwYXJhIHN1IGludGVycHJldGFjacOzbi4gTG9zIG3DqXRvZG9zIG3DoXMgY29tdW5lcyBzb24gZWwgQW7DoWxpc2lzIGRlIFNlbnRpbWllbnRvcywgbGEgTnViZSBkZSBQYWxhYnJhcyB5IGVsIFRvcGljIE1vZGVsaW5nLlwNCjMuIEFuw6FsaXNpcyBwcmVkaWN0aXZvOiBTb24gbGFzIHTDqWNuaWNhcyB5IG1vZGVsb3MgZXN0YWTDrXN0aWNvcyBwYXJhIHByZWRlY3JpIHJlc3VsdGFkb3MgZnV0dXJvcy4gTG9zIG1vZGVsb3MgbcOhcyB1c2Fkb3Mgc29uIGVsIFJhbmRvbSBGb3Jlc3QsIHJlZGVzIG5ldXJvbmFsZXMgeSByZWdyZXNpb25lcy4NCg0KYGBge3IgbWVzc2FnZT1GQUxTRSwgd2FybmluZz1GQUxTRX0NCiNpbnN0YWxsLnBhY2thZ2VzKCJ0aWR5dmVyc2UiKSAjTWFuaXB1bGFjacOzbiBkZSBkYXRvcw0KbGlicmFyeSh0aWR5dmVyc2UpDQojaW5zdGFsbC5wYWNrYWdlcygidGVzc2VyYWN0IikgIyBPQ1INCmxpYnJhcnkodGVzc2VyYWN0KQ0KI2luc3RhbGwucGFja2FnZXMoIm1hZ2ljayIpICMgUE5HDQpsaWJyYXJ5KG1hZ2ljaykNCiNpbnN0YWxsLnBhY2thZ2VzKCJvZmZpY2VyIikgIyBPZmZpY2UgIldvcmQiDQpsaWJyYXJ5KG9mZmljZXIpDQojaW5zdGFsbC5wYWNrYWdlcygicGRmdG9vbHMiKSAjIFBERg0KbGlicmFyeShwZGZ0b29scykNCiNpbnN0YWxsLnBhY2thZ2VzKCJwdXJyciIpICMgUERGDQpsaWJyYXJ5KHB1cnJyKSANCmBgYA0KDQojIFtFdGFwYSAxLiBPYnRlbmVyIERhdG9zIG1lZGlhbnRlIE9DUl17c3R5bGU9ImNvbG9yOiByZWQ7In0NCg0KIyMgW0RlIGltYWdlbiBQTkcgYSB0ZXh0byBlbiBXb3JkXXtzdHlsZT0iY29sb3I6IHJlZDsifQ0KDQpgYGB7cn0NCmltYWdlbjEgPC0gaW1hZ2VfcmVhZCgiQzpcXFVzZXJzXFxVc3VhcmlvXFxEb2N1bWVudHNcXElBIENvbmNlbnRyYWNpw7NuXFxNMlxcVGV4dCBNaW5pbmdcXGltYWdlbjEgKDEpLlBORyIpDQp0ZXh0bzEgPC0gb2NyKGltYWdlbjEpDQp0ZXh0bzENCg0KYGBgDQoNCmBgYHtyfQ0KZG9jMSA8LSByZWFkX2RvY3goKSAjQ3JlYSB1biBkb2N1bWVudG8gZGUgV29yZCBlbiBibGFuY28NCmRvYzEgPC0gZG9jMSAlPiUgYm9keV9hZGRfcGFyKHRleHRvMSkgIyBQZWdhIGVsIHRleHRvIGVuIGVsIGRvYw0KcHJpbnQoZG9jMSwgdGFyZ2V0PSJ0ZXh0bzEuZG9jeCIpICMgR3VhcmRhIGVsIGRvYyBlbiBsYSBjb21wdSBjb24gdW4gbm9tYnJlIGFzaWduYWRvDQpgYGANCg0KIyMgW0RlIGltYWdlbiBQTkcgZW4gZXNwYcOxb2wgYSB0ZXh0byBlbiBXb3JkXXtzdHlsZT0iY29sb3I6IHJlZDsifQ0KDQpgYGB7cn0NCmltYWdlbjIgPC0gaW1hZ2VfcmVhZCgiQzpcXFVzZXJzXFxVc3VhcmlvXFxEb2N1bWVudHNcXElBIENvbmNlbnRyYWNpw7NuXFxNMlxcVGV4dCBNaW5pbmdcXGltYWdlbjIuUE5HIikNCnRlc3NlcmFjdF9kb3dubG9hZCgic3BhIikNCnRleHRvMiA8LSBvY3IoaW1hZ2VuMiwgZW5naW5lID0gdGVzc2VyYWN0KCJzcGEiKSkNCnRleHRvMg0KDQpgYGANCg0KYGBge3J9DQpkb2MyIDwtIHJlYWRfZG9jeCgpICNDcmVhIHVuIGRvY3VtZW50byBkZSBXb3JkIGVuIGJsYW5jbw0KZG9jMiA8LSBkb2MyICU+JSBib2R5X2FkZF9wYXIodGV4dG8yKSAjIFBlZ2EgZWwgdGV4dG8gZW4gZWwgZG9jDQojcHJpbnQoZG9jMiwgdGFyZ2V0PSJ0ZXh0bzIuZG9jeCIpICMgR3VhcmRhIGVsIGRvYyBlbiBsYSBjb21wdSBjb24gdW4gbm9tYnJlIGFzaWduYWRvDQpgYGANCg0KIyBbQWN0aXZpZGFkIDEuIEVzb117c3R5bGU9ImNvbG9yOiByZWQ7In0NCg0KYGBge3J9DQojRGUgcGRmIGEgdGV4dG8gZW4gd29yZA0KcGRmX2VzbyA8LSBwZGZfY29udmVydCgiQzpcXFVzZXJzXFxVc3VhcmlvXFxEb2N1bWVudHNcXElBIENvbmNlbnRyYWNpw7NuXFxNMlxcVGV4dCBNaW5pbmdcXGVzby5wZGYiLCBkcGk9NjAwKSAlPiUgbWFwKG9jcikNCg0KYGBgDQoNCiMjIFtUcmFuc2Zvcm1hciBwZGYgYSBwbmcgZW4gZXNwYcOxb2xde3N0eWxlPSJjb2xvcjogcmVkOyJ9DQoNCmBgYHtyfQ0KaW1hZ2VuMyA8LSBpbWFnZV9yZWFkKCJDOlxcVXNlcnNcXFVzdWFyaW9cXERvY3VtZW50c1xcSUEgQ29uY2VudHJhY2nDs25cXE0yXFxUZXh0IE1pbmluZ1xcZXNvXzEuUE5HIikNCnRlc3NlcmFjdF9kb3dubG9hZCgic3BhIikNCnRleHRvMyA8LSBvY3IoaW1hZ2VuMywgZW5naW5lID0gdGVzc2VyYWN0KCJzcGEiKSkNCnRleHRvMw0KDQpgYGANCg0KYGBge3J9DQpkb2MzIDwtIHJlYWRfZG9jeCgpICNDcmVhIHVuIGRvY3VtZW50byBkZSBXb3JkIGVuIGJsYW5jbw0KZG9jMyA8LSBkb2MzICU+JSBib2R5X2FkZF9wYXIodGV4dG8zKSAjIFBlZ2EgZWwgdGV4dG8gZW4gZWwgZG9jDQpwcmludChkb2MzLCB0YXJnZXQ9InRleHRvMy5kb2N4IikgIyBHdWFyZGEgZWwgZG9jIGVuIGxhIGNvbXB1IGNvbiB1biBub21icmUgYXNpZ25hZG8NCmBgYA0KDQpgYGB7cn0NCg0KaW1hZ2VuNCA8LSBpbWFnZV9yZWFkKCJDOlxcVXNlcnNcXFVzdWFyaW9cXERvY3VtZW50c1xcSUEgQ29uY2VudHJhY2nDs25cXE0yXFxUZXh0IE1pbmluZ1xcZXNvXzIucG5nIikNCnRlc3NlcmFjdF9kb3dubG9hZCgic3BhIikNCnRleHRvNCA8LSBvY3IoaW1hZ2VuNCwgZW5naW5lID0gdGVzc2VyYWN0KCJzcGEiKSkNCnRleHRvNA0KDQpkb2M0IDwtIHJlYWRfZG9jeCgpICMgQ3JlYSB1biBkb2N1bWVudG8gZGUgV29yZCBlbiBibGFuY28NCmRvYzQgPC0gZG9jNCAlPiUgYm9keV9hZGRfcGFyKHRleHRvNCkgIyBQZWdhIGVsIHRleHRvIGVuIGVsIGRvYw0KcHJpbnQoZG9jNCwgdGFyZ2V0PSJ0ZXh0bzQuZG9jeCIpICMgR3VhcmRhIGVsIGRvYyBlbiBsYSBjb21wdSBjb24gdW4gbm9tYnJlIGFzaWduYWRvDQpgYGANCg0KIyMgW0p1bnRhciBsb3MgZG9zIGFyY2hpdm9zIGRlIHRleHRvIHRyYW5zZm9ybWFkb3MgYSB1biBzb2xvIGRvY3VlbW50byBkZSB3b3JkXXtzdHlsZT0iY29sb3I6IHJlZDsifQ0KDQpgYGB7cn0NCmRvY19lc28gPC0gcmVhZF9kb2N4KCkNCmRvY19lc28gPC0gZG9jX2VzbyAlPiUgYm9keV9hZGRfcGFyKHRleHRvMykgJT4lIGJvZHlfYWRkX3Bhcih0ZXh0bzQpDQpwcmludChkb2NfZXNvLCB0YXJnZXQ9ImRvY19lc28uZG9jeCIpDQpgYGANCg0KIyBbRXRhcGEgMi4gRXhwbG9yYXIgZGF0b3MgbWVkaWFudGUgYW7DoWxpc2lzIGRlIHNlbnRpbWllbnRvc117c3R5bGU9ImNvbG9yOiByZWQ7In0NCg0KIyMgW0luc3RhbGFyIHBhcXVldGUgc3kgbGxhbXJhIGxpYnJlcsOtYXNde3N0eWxlPSJjb2xvcjogcmVkOyJ9DQoNCmBgYHtyfQ0KI2luc3RhbGwucGFja2FnZXMoInhtbDIiLCBkZXBlbmRlbmNpZXMgPSBUUlVFKQ0KI2luc3RhbGwucGFja2FnZXMoIk5MUCIsIGRlcGVuZGVuY2llcyA9IFRSVUUpDQojaW5zdGFsbC5wYWNrYWdlcygidG0iLCBkZXBlbmRlbmNpZXMgPSBUUlVFKQ0KDQojaW5zdGFsbC5wYWNrYWdlcygic3l1emhldCIpICNBbsOhbGlzaXMgZGUgc2VudGltaWVudG9zDQpsaWJyYXJ5KHN5dXpoZXQpDQojaW5zdGFsbC5wYWNrYWdlcygidG0iKSAjTWluZXLDrWEgZGUgdGV4dG8NCmxpYnJhcnkodG0pDQojaW5zdGFsbC5wYWNrYWdlcygid29yZGNsb3VkIikgI01pbmVyw61hIGRlIHRleHRvDQpsaWJyYXJ5KHdvcmRjbG91ZCkNCg0KYGBgDQoNCmBgYHtyfQ0KI2luc3RhbGwucGFja2FnZXMoIlJDb2xvckJyZXdlciIpICNNaW5lcsOtYSBkZSB0ZXh0bw0KbGlicmFyeShSQ29sb3JCcmV3ZXIpDQpgYGANCg0KIyMgW0Fuw6FsaXNpcyBkZSBlbW9jaW9uZXMgeSBzZW50aW1pZW50b3Nde3N0eWxlPSJjb2xvcjogcmVkOyJ9DQoNCmBgYHtyfQ0KdGV4dG8gPC0gcGRmX2Vzbw0KdGV4dG9fcGFsYWJyYXMgPC0gZ2V0X3Rva2Vucyh0ZXh0bykgIyBTZXBhcmFyIHRleHRvIHBvciBwYWxhYnJhcywgdG9rZW5zDQplbW9jaW9uZXMgPC0gZ2V0X25yY19zZW50aW1lbnQodGV4dG9fcGFsYWJyYXMsIGxhbmd1YWdlID0gInNwYW5pc2giKQ0KIyBBbGVncsOtYSwgdHJpc3RlemEsIGlyYSwgbWllZG8sIHNvcnByZXNhLCBhc2NvLCBhbnRpY2lwYWNpw7NuIHkgY29uZmlhbnphDQpiYXJwbG90KGNvbFN1bXMocHJvcC50YWJsZShlbW9jaW9uZXNbLDE6OF0pKSkNCnNlbnRpbWllbnRvcyA8LSAoZW1vY2lvbmVzJG5lZ2F0aXZlKi0xKStlbW9jaW9uZXMkcG9zaXRpdmUNCnNpbXBsZV9wbG90KHNlbnRpbWllbnRvcykNCmBgYA0KDQojIyBbTnViZSBkZSBwYWxhYnJhc117c3R5bGU9ImNvbG9yOiByZWQ7In0NCg0KYGBge3J9DQpwYWxhYnJhcyA8LSB0ZXh0b19wYWxhYnJhcw0KcGFsYWJyYXMgPC0gcmVtb3ZlV29yZHMocGFsYWJyYXMsIGMoc3RvcHdvcmRzKCJzcGFuaXNoIiksImhhY2lhIiwgImhhYmlhIiwgImhhYmlhbiIsICJoZWNobyIpKQ0Kd29yZGNsb3VkKHdvcmRzPXBhbGFicmFzLCBtaW4uZnJlcT0yLCByb3QucGVyID0gMCwgcmFuZG9tLm9yZGVyPUZBTFNFICkNCmBgYA0K