Spaces:
Sleeping
Sleeping
Delete Mini-Projeto3-Deploy.ipynb
Browse files- Mini-Projeto3-Deploy.ipynb +0 -375
Mini-Projeto3-Deploy.ipynb
DELETED
@@ -1,375 +0,0 @@
|
|
1 |
-
{
|
2 |
-
"cells": [
|
3 |
-
{
|
4 |
-
"cell_type": "markdown",
|
5 |
-
"metadata": {},
|
6 |
-
"source": [
|
7 |
-
"# <font color='blue'>Data Science Academy</font>\n",
|
8 |
-
"# <font color='blue'>Deep Learning Para Aplicações de IA com PyTorch e Lightning</font>\n",
|
9 |
-
"\n",
|
10 |
-
"## <font color='blue'>Mini-Projeto 3 - Deploy</font>\n",
|
11 |
-
"## <font color='blue'>Fine-Tuning de Modelo LLM Para Tarefa Específica e Deploy de Web App com Gradio</font>"
|
12 |
-
]
|
13 |
-
},
|
14 |
-
{
|
15 |
-
"cell_type": "markdown",
|
16 |
-
"metadata": {},
|
17 |
-
"source": [
|
18 |
-
""
|
19 |
-
]
|
20 |
-
},
|
21 |
-
{
|
22 |
-
"cell_type": "markdown",
|
23 |
-
"metadata": {},
|
24 |
-
"source": [
|
25 |
-
"## Instalando e Carregando os Pacotes"
|
26 |
-
]
|
27 |
-
},
|
28 |
-
{
|
29 |
-
"cell_type": "code",
|
30 |
-
"execution_count": 1,
|
31 |
-
"metadata": {},
|
32 |
-
"outputs": [
|
33 |
-
{
|
34 |
-
"name": "stdout",
|
35 |
-
"output_type": "stream",
|
36 |
-
"text": [
|
37 |
-
"Versão da Linguagem Python Usada Neste Jupyter Notebook: 3.10.9\n"
|
38 |
-
]
|
39 |
-
}
|
40 |
-
],
|
41 |
-
"source": [
|
42 |
-
"# Versão da Linguagem Python\n",
|
43 |
-
"from platform import python_version\n",
|
44 |
-
"print('Versão da Linguagem Python Usada Neste Jupyter Notebook:', python_version())"
|
45 |
-
]
|
46 |
-
},
|
47 |
-
{
|
48 |
-
"cell_type": "code",
|
49 |
-
"execution_count": 2,
|
50 |
-
"metadata": {},
|
51 |
-
"outputs": [],
|
52 |
-
"source": [
|
53 |
-
"# Para atualizar um pacote, execute o comando abaixo no terminal ou prompt de comando:\n",
|
54 |
-
"# pip install -U nome_pacote\n",
|
55 |
-
"\n",
|
56 |
-
"# Para instalar a versão exata de um pacote, execute o comando abaixo no terminal ou prompt de comando:\n",
|
57 |
-
"# !pip install nome_pacote==versão_desejada\n",
|
58 |
-
"\n",
|
59 |
-
"# Depois de instalar ou atualizar o pacote, reinicie o jupyter notebook.\n",
|
60 |
-
"\n",
|
61 |
-
"# Instala o pacote watermark. \n",
|
62 |
-
"# Esse pacote é usado para gravar as versões de outros pacotes usados neste jupyter notebook.\n",
|
63 |
-
"!pip install -q -U watermark"
|
64 |
-
]
|
65 |
-
},
|
66 |
-
{
|
67 |
-
"cell_type": "code",
|
68 |
-
"execution_count": 3,
|
69 |
-
"metadata": {},
|
70 |
-
"outputs": [
|
71 |
-
{
|
72 |
-
"name": "stdout",
|
73 |
-
"output_type": "stream",
|
74 |
-
"text": [
|
75 |
-
"env: TF_CPP_MIN_LOG_LEVEL=3\n"
|
76 |
-
]
|
77 |
-
}
|
78 |
-
],
|
79 |
-
"source": [
|
80 |
-
"%env TF_CPP_MIN_LOG_LEVEL=3"
|
81 |
-
]
|
82 |
-
},
|
83 |
-
{
|
84 |
-
"cell_type": "code",
|
85 |
-
"execution_count": 4,
|
86 |
-
"metadata": {},
|
87 |
-
"outputs": [],
|
88 |
-
"source": [
|
89 |
-
"# https://pypi.org/project/gradio/\n",
|
90 |
-
"!pip install -q gradio"
|
91 |
-
]
|
92 |
-
},
|
93 |
-
{
|
94 |
-
"cell_type": "code",
|
95 |
-
"execution_count": 5,
|
96 |
-
"metadata": {},
|
97 |
-
"outputs": [],
|
98 |
-
"source": [
|
99 |
-
"# Imports\n",
|
100 |
-
"import torch\n",
|
101 |
-
"import gradio as gr\n",
|
102 |
-
"from transformers import AutoModelForCausalLM"
|
103 |
-
]
|
104 |
-
},
|
105 |
-
{
|
106 |
-
"cell_type": "code",
|
107 |
-
"execution_count": 6,
|
108 |
-
"metadata": {},
|
109 |
-
"outputs": [
|
110 |
-
{
|
111 |
-
"name": "stdout",
|
112 |
-
"output_type": "stream",
|
113 |
-
"text": [
|
114 |
-
"Author: Data Science Academy\n",
|
115 |
-
"\n",
|
116 |
-
"torch : 2.0.1\n",
|
117 |
-
"gradio: 3.39.0\n",
|
118 |
-
"\n"
|
119 |
-
]
|
120 |
-
}
|
121 |
-
],
|
122 |
-
"source": [
|
123 |
-
"# Versões dos pacotes usados neste jupyter notebook\n",
|
124 |
-
"%reload_ext watermark\n",
|
125 |
-
"%watermark -a \"Data Science Academy\" --iversions"
|
126 |
-
]
|
127 |
-
},
|
128 |
-
{
|
129 |
-
"cell_type": "code",
|
130 |
-
"execution_count": 7,
|
131 |
-
"metadata": {},
|
132 |
-
"outputs": [],
|
133 |
-
"source": [
|
134 |
-
"# Carrega o modelo\n",
|
135 |
-
"modelo_llm = AutoModelForCausalLM.from_pretrained(\"modelos/modelo_final\")"
|
136 |
-
]
|
137 |
-
},
|
138 |
-
{
|
139 |
-
"cell_type": "code",
|
140 |
-
"execution_count": 8,
|
141 |
-
"metadata": {},
|
142 |
-
"outputs": [],
|
143 |
-
"source": [
|
144 |
-
"# Definindo uma classe chamada NumberTokenizer, que é usada para tokenizar os números\n",
|
145 |
-
"class DSATokenizer:\n",
|
146 |
-
" \n",
|
147 |
-
" # Método construtor da classe, que é executado quando um objeto dessa classe é criado\n",
|
148 |
-
" def __init__(self, numbers_qty = 10):\n",
|
149 |
-
" \n",
|
150 |
-
" # Lista de tokens possíveis que o tokenizador pode encontrar\n",
|
151 |
-
" vocab = ['+', '=', '-1', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9']\n",
|
152 |
-
" \n",
|
153 |
-
" # Definindo a quantidade de números que o tokenizador pode lidar\n",
|
154 |
-
" self.numbers_qty = numbers_qty\n",
|
155 |
-
" \n",
|
156 |
-
" # Definindo o token de preenchimento (padding)\n",
|
157 |
-
" self.pad_token = '-1'\n",
|
158 |
-
" \n",
|
159 |
-
" # Criando um dicionário que mapeia cada token para um índice único\n",
|
160 |
-
" self.encoder = {str(v):i for i,v in enumerate(vocab)}\n",
|
161 |
-
" \n",
|
162 |
-
" # Criando um dicionário que mapeia cada índice único de volta ao token correspondente\n",
|
163 |
-
" self.decoder = {i:str(v) for i,v in enumerate(vocab)}\n",
|
164 |
-
" \n",
|
165 |
-
" # Obtendo o índice do token de preenchimento no encoder\n",
|
166 |
-
" self.pad_token_id = self.encoder[self.pad_token]\n",
|
167 |
-
"\n",
|
168 |
-
" # Método para decodificar uma lista de IDs de token de volta para uma string\n",
|
169 |
-
" def decode(self, token_ids):\n",
|
170 |
-
" return ' '.join(self.decoder[t] for t in token_ids)\n",
|
171 |
-
"\n",
|
172 |
-
" # Método que é chamado quando o objeto da classe é invocado como uma função\n",
|
173 |
-
" def __call__(self, text):\n",
|
174 |
-
" # Dividindo o texto em tokens individuais e retornando uma lista dos IDs correspondentes\n",
|
175 |
-
" return [self.encoder[t] for t in text.split()]"
|
176 |
-
]
|
177 |
-
},
|
178 |
-
{
|
179 |
-
"cell_type": "code",
|
180 |
-
"execution_count": 9,
|
181 |
-
"metadata": {},
|
182 |
-
"outputs": [],
|
183 |
-
"source": [
|
184 |
-
"# Cria o objeto\n",
|
185 |
-
"tokenizer = DSATokenizer(13)"
|
186 |
-
]
|
187 |
-
},
|
188 |
-
{
|
189 |
-
"cell_type": "code",
|
190 |
-
"execution_count": 10,
|
191 |
-
"metadata": {},
|
192 |
-
"outputs": [],
|
193 |
-
"source": [
|
194 |
-
"# Definindo a função gera_solution com três parâmetros: input, solution_length e model\n",
|
195 |
-
"def faz_previsao(entrada, solution_length = 6, model = modelo_llm):\n",
|
196 |
-
"\n",
|
197 |
-
" # Colocando o modelo em modo de avaliação. \n",
|
198 |
-
" model.eval()\n",
|
199 |
-
"\n",
|
200 |
-
" # Convertendo a entrada (string) em tensor utilizando o tokenizer. \n",
|
201 |
-
" # O tensor é uma estrutura de dados que o modelo de aprendizado de máquina pode processar.\n",
|
202 |
-
" entrada = torch.tensor(tokenizer(entrada))\n",
|
203 |
-
"\n",
|
204 |
-
" # Iniciando uma lista vazia para armazenar a solução\n",
|
205 |
-
" solution = []\n",
|
206 |
-
"\n",
|
207 |
-
" # Loop que gera a solução de comprimento solution_length\n",
|
208 |
-
" for i in range(solution_length):\n",
|
209 |
-
"\n",
|
210 |
-
" # Alimentando a entrada atual ao modelo e obtendo a saída\n",
|
211 |
-
" saida = model(entrada)\n",
|
212 |
-
"\n",
|
213 |
-
" # Pegando o índice do maior valor no último conjunto de logits (log-odds) da saída, \n",
|
214 |
-
" # que é a previsão do modelo para o próximo token\n",
|
215 |
-
" predicted = saida.logits[-1].argmax()\n",
|
216 |
-
"\n",
|
217 |
-
" # Concatenando a previsão atual com a entrada atual. \n",
|
218 |
-
" # Isso servirá como a nova entrada para a próxima iteração.\n",
|
219 |
-
" entrada = torch.cat((entrada, predicted.unsqueeze(0)), dim = 0)\n",
|
220 |
-
"\n",
|
221 |
-
" # Adicionando a previsão atual à lista de soluções e convertendo o tensor em um número Python padrão\n",
|
222 |
-
" solution.append(predicted.cpu().item())\n",
|
223 |
-
"\n",
|
224 |
-
" # Decodificando a lista de soluções para obter a string de saída e retornando-a\n",
|
225 |
-
" return tokenizer.decode(solution)"
|
226 |
-
]
|
227 |
-
},
|
228 |
-
{
|
229 |
-
"cell_type": "code",
|
230 |
-
"execution_count": 11,
|
231 |
-
"metadata": {},
|
232 |
-
"outputs": [
|
233 |
-
{
|
234 |
-
"data": {
|
235 |
-
"text/plain": [
|
236 |
-
"'0 8'"
|
237 |
-
]
|
238 |
-
},
|
239 |
-
"execution_count": 11,
|
240 |
-
"metadata": {},
|
241 |
-
"output_type": "execute_result"
|
242 |
-
}
|
243 |
-
],
|
244 |
-
"source": [
|
245 |
-
"# Testa a função\n",
|
246 |
-
"faz_previsao('3 + 5 =', solution_length = 2)"
|
247 |
-
]
|
248 |
-
},
|
249 |
-
{
|
250 |
-
"cell_type": "code",
|
251 |
-
"execution_count": 12,
|
252 |
-
"metadata": {},
|
253 |
-
"outputs": [],
|
254 |
-
"source": [
|
255 |
-
"# Função para retornar a função que faz a previsão\n",
|
256 |
-
"def funcsolve(entrada):\n",
|
257 |
-
" return faz_previsao(entrada, solution_length = 2)"
|
258 |
-
]
|
259 |
-
},
|
260 |
-
{
|
261 |
-
"cell_type": "code",
|
262 |
-
"execution_count": 13,
|
263 |
-
"metadata": {},
|
264 |
-
"outputs": [],
|
265 |
-
"source": [
|
266 |
-
"# Cria a web app\n",
|
267 |
-
"webapp = gr.Interface(fn = funcsolve, \n",
|
268 |
-
" inputs = [gr.Textbox(label = \"Dados de Entrada\", \n",
|
269 |
-
" lines = 1, \n",
|
270 |
-
" info = \"Os dados devem estar na forma: '1 + 2 =' com um único espaço entre cada caractere e apenas números de um dígito são permitidos.\")],\n",
|
271 |
-
" outputs = [gr.Textbox(label = \"Resultado (Previsão do Modelo)\", lines = 1)],\n",
|
272 |
-
" title = \"Deploy de LLM Após o Fine-Tuning\",\n",
|
273 |
-
" description = \"Digite os dados de entrada e clique no botão Submit para o modelo fazer a previsão.\",\n",
|
274 |
-
" examples = [\"5 + 3 =\", \"2 + 9 =\"]) "
|
275 |
-
]
|
276 |
-
},
|
277 |
-
{
|
278 |
-
"cell_type": "code",
|
279 |
-
"execution_count": 14,
|
280 |
-
"metadata": {},
|
281 |
-
"outputs": [
|
282 |
-
{
|
283 |
-
"name": "stdout",
|
284 |
-
"output_type": "stream",
|
285 |
-
"text": [
|
286 |
-
"Running on local URL: http://127.0.0.1:7860\n",
|
287 |
-
"\n",
|
288 |
-
"To create a public link, set `share=True` in `launch()`.\n"
|
289 |
-
]
|
290 |
-
},
|
291 |
-
{
|
292 |
-
"data": {
|
293 |
-
"text/html": [
|
294 |
-
"<div><iframe src=\"http://127.0.0.1:7860/\" width=\"100%\" height=\"500\" allow=\"autoplay; camera; microphone; clipboard-read; clipboard-write;\" frameborder=\"0\" allowfullscreen></iframe></div>"
|
295 |
-
],
|
296 |
-
"text/plain": [
|
297 |
-
"<IPython.core.display.HTML object>"
|
298 |
-
]
|
299 |
-
},
|
300 |
-
"metadata": {},
|
301 |
-
"output_type": "display_data"
|
302 |
-
},
|
303 |
-
{
|
304 |
-
"data": {
|
305 |
-
"text/plain": []
|
306 |
-
},
|
307 |
-
"execution_count": 14,
|
308 |
-
"metadata": {},
|
309 |
-
"output_type": "execute_result"
|
310 |
-
},
|
311 |
-
{
|
312 |
-
"name": "stderr",
|
313 |
-
"output_type": "stream",
|
314 |
-
"text": [
|
315 |
-
"Traceback (most recent call last):\n",
|
316 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/gradio/routes.py\", line 442, in run_predict\n",
|
317 |
-
" output = await app.get_blocks().process_api(\n",
|
318 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/gradio/blocks.py\", line 1392, in process_api\n",
|
319 |
-
" result = await self.call_function(\n",
|
320 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/gradio/blocks.py\", line 1097, in call_function\n",
|
321 |
-
" prediction = await anyio.to_thread.run_sync(\n",
|
322 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/anyio/to_thread.py\", line 28, in run_sync\n",
|
323 |
-
" return await get_asynclib().run_sync_in_worker_thread(func, *args, cancellable=cancellable,\n",
|
324 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/anyio/_backends/_asyncio.py\", line 818, in run_sync_in_worker_thread\n",
|
325 |
-
" return await future\n",
|
326 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/anyio/_backends/_asyncio.py\", line 754, in run\n",
|
327 |
-
" result = context.run(func, *args)\n",
|
328 |
-
" File \"/Users/dmpm/anaconda3/lib/python3.10/site-packages/gradio/utils.py\", line 703, in wrapper\n",
|
329 |
-
" response = f(*args, **kwargs)\n",
|
330 |
-
" File \"/var/folders/dc/lqrc3k5j4438r150cbrdr_000000gn/T/ipykernel_5062/1694334565.py\", line 3, in funcsolve\n",
|
331 |
-
" return faz_previsao(entrada, solution_length = 2)\n",
|
332 |
-
" File \"/var/folders/dc/lqrc3k5j4438r150cbrdr_000000gn/T/ipykernel_5062/2965097493.py\", line 9, in faz_previsao\n",
|
333 |
-
" entrada = torch.tensor(tokenizer(entrada))\n",
|
334 |
-
" File \"/var/folders/dc/lqrc3k5j4438r150cbrdr_000000gn/T/ipykernel_5062/3426772234.py\", line 32, in __call__\n",
|
335 |
-
" return [self.encoder[t] for t in text.split()]\n",
|
336 |
-
" File \"/var/folders/dc/lqrc3k5j4438r150cbrdr_000000gn/T/ipykernel_5062/3426772234.py\", line 32, in <listcomp>\n",
|
337 |
-
" return [self.encoder[t] for t in text.split()]\n",
|
338 |
-
"KeyError: '122220'\n"
|
339 |
-
]
|
340 |
-
}
|
341 |
-
],
|
342 |
-
"source": [
|
343 |
-
"webapp.launch()"
|
344 |
-
]
|
345 |
-
},
|
346 |
-
{
|
347 |
-
"cell_type": "markdown",
|
348 |
-
"metadata": {},
|
349 |
-
"source": [
|
350 |
-
"# Fim"
|
351 |
-
]
|
352 |
-
}
|
353 |
-
],
|
354 |
-
"metadata": {
|
355 |
-
"kernelspec": {
|
356 |
-
"display_name": "Python 3 (ipykernel)",
|
357 |
-
"language": "python",
|
358 |
-
"name": "python3"
|
359 |
-
},
|
360 |
-
"language_info": {
|
361 |
-
"codemirror_mode": {
|
362 |
-
"name": "ipython",
|
363 |
-
"version": 3
|
364 |
-
},
|
365 |
-
"file_extension": ".py",
|
366 |
-
"mimetype": "text/x-python",
|
367 |
-
"name": "python",
|
368 |
-
"nbconvert_exporter": "python",
|
369 |
-
"pygments_lexer": "ipython3",
|
370 |
-
"version": "3.10.9"
|
371 |
-
}
|
372 |
-
},
|
373 |
-
"nbformat": 4,
|
374 |
-
"nbformat_minor": 2
|
375 |
-
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|