Spaces:
Running
Running
greos
commited on
Commit
•
3d75b57
1
Parent(s):
bbe87fb
Don't ignore the model file
Browse filesDon't need the old ipynb file
- .gitignore +1 -5
- house_room.ipynb +0 -946
- model/model.pkl +3 -0
.gitignore
CHANGED
@@ -95,8 +95,4 @@ mkl_config.ini
|
|
95 |
dmypy.json
|
96 |
|
97 |
# Pyre type checker
|
98 |
-
.pyre/
|
99 |
-
|
100 |
-
# Deployment
|
101 |
-
model/
|
102 |
-
*.pkl
|
|
|
95 |
dmypy.json
|
96 |
|
97 |
# Pyre type checker
|
98 |
+
.pyre/
|
|
|
|
|
|
|
|
house_room.ipynb
DELETED
@@ -1,946 +0,0 @@
|
|
1 |
-
{
|
2 |
-
"cells": [
|
3 |
-
{
|
4 |
-
"cell_type": "code",
|
5 |
-
"execution_count": 9,
|
6 |
-
"metadata": {},
|
7 |
-
"outputs": [],
|
8 |
-
"source": [
|
9 |
-
"!pip install -Uqq fastai"
|
10 |
-
]
|
11 |
-
},
|
12 |
-
{
|
13 |
-
"cell_type": "code",
|
14 |
-
"execution_count": 13,
|
15 |
-
"metadata": {},
|
16 |
-
"outputs": [
|
17 |
-
{
|
18 |
-
"ename": "AttributeError",
|
19 |
-
"evalue": "module 'torch' has no attribute 'device'",
|
20 |
-
"output_type": "error",
|
21 |
-
"traceback": [
|
22 |
-
"\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
|
23 |
-
"\u001b[0;31mAttributeError\u001b[0m Traceback (most recent call last)",
|
24 |
-
"Cell \u001b[0;32mIn[13], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mfastai\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mvision\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mall\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;241m*\u001b[39m\n\u001b[1;32m 2\u001b[0m \u001b[38;5;66;03m#import gradio as gr\u001b[39;00m\n\u001b[1;32m 3\u001b[0m \n\u001b[1;32m 4\u001b[0m \u001b[38;5;66;03m# def house_room(x): return x[0].isupper()\u001b[39;00m\n",
|
25 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/fastai/vision/all.py:1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m models\n\u001b[1;32m 2\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mbasics\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;241m*\u001b[39m\n\u001b[1;32m 3\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mcallback\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mall\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;241m*\u001b[39m\n",
|
26 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/fastai/vision/models/__init__.py:1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m xresnet\n\u001b[1;32m 2\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m unet\n\u001b[1;32m 3\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mtvm\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;241m*\u001b[39m\n",
|
27 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/fastai/vision/models/xresnet.py:5\u001b[0m\n\u001b[1;32m 1\u001b[0m \u001b[38;5;66;03m# AUTOGENERATED! DO NOT EDIT! File to edit: ../../../nbs/11_vision.models.xresnet.ipynb.\u001b[39;00m\n\u001b[1;32m 2\u001b[0m \n\u001b[1;32m 3\u001b[0m \u001b[38;5;66;03m# %% ../../../nbs/11_vision.models.xresnet.ipynb 2\u001b[39;00m\n\u001b[1;32m 4\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m__future__\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m annotations\n\u001b[0;32m----> 5\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mtorch_basics\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;241m*\u001b[39m\n\u001b[1;32m 6\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m: \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mtorchvision\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mmodels\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mutils\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m load_state_dict_from_url\n\u001b[1;32m 7\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m \u001b[38;5;167;01mModuleNotFoundError\u001b[39;00m: \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mhub\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m load_state_dict_from_url\n",
|
28 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/fastai/torch_basics.py:1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m multiprocessing\n\u001b[1;32m 2\u001b[0m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;21;01mplatform\u001b[39;00m\u001b[38;5;241m,\u001b[39m\u001b[38;5;21;01mos\u001b[39;00m\n\u001b[1;32m 3\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m platform\u001b[38;5;241m.\u001b[39msystem()\u001b[38;5;241m==\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mDarwin\u001b[39m\u001b[38;5;124m'\u001b[39m:\n\u001b[1;32m 4\u001b[0m \u001b[38;5;66;03m# Python 3.8 changed to 'spawn' but that doesn't work with PyTorch DataLoader w n_workers>0\u001b[39;00m\n",
|
29 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/torch/multiprocessing/__init__.py:20\u001b[0m\n\u001b[1;32m 17\u001b[0m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;21;01msys\u001b[39;00m\n\u001b[1;32m 19\u001b[0m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m\n\u001b[0;32m---> 20\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mreductions\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m init_reductions\n\u001b[1;32m 22\u001b[0m __all__ \u001b[38;5;241m=\u001b[39m [\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mset_sharing_strategy\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mget_sharing_strategy\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mget_all_sharing_strategies\u001b[39m\u001b[38;5;124m\"\u001b[39m]\n\u001b[1;32m 25\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mmultiprocessing\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;241m*\u001b[39m \u001b[38;5;66;03m# noqa: F403\u001b[39;00m\n",
|
30 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/torch/multiprocessing/reductions.py:9\u001b[0m\n\u001b[1;32m 6\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mtyping\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m Union\n\u001b[1;32m 8\u001b[0m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m\n\u001b[0;32m----> 9\u001b[0m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mutils\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mhooks\u001b[39;00m\n\u001b[1;32m 10\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01m_namedtensor_internals\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m check_serializing_named_tensor\n\u001b[1;32m 12\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m 13\u001b[0m \u001b[38;5;66;03m# Early load resource_sharer to prevent a partially initialized instance\u001b[39;00m\n\u001b[1;32m 14\u001b[0m \u001b[38;5;66;03m# from being inherited in a forked child process. The reduce_storage method\u001b[39;00m\n\u001b[1;32m 15\u001b[0m \u001b[38;5;66;03m# requires this module indirectly through DupFd(). The built-in mp.Queue\u001b[39;00m\n\u001b[1;32m 16\u001b[0m \u001b[38;5;66;03m# class pickles arguments in a background thread which may overlap with the\u001b[39;00m\n\u001b[1;32m 17\u001b[0m \u001b[38;5;66;03m# fork.\u001b[39;00m\n",
|
31 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/torch/utils/__init__.py:6\u001b[0m\n\u001b[1;32m 4\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mthroughput_benchmark\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m ThroughputBenchmark\n\u001b[1;32m 5\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mcpp_backtrace\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m get_cpp_backtrace\n\u001b[0;32m----> 6\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mbackend_registration\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m rename_privateuse1_backend, generate_methods_for_privateuse1_backend\n\u001b[1;32m 7\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m deterministic\n\u001b[1;32m 8\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01m.\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m collect_env\n",
|
32 |
-
"File \u001b[0;32m~/projects/app/python/fast-ai-course/projects/house_room_classifier/.env/lib/python3.11/site-packages/torch/utils/backend_registration.py:95\u001b[0m\n\u001b[1;32m 91\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mhasattr\u001b[39m(module, attr):\n\u001b[1;32m 92\u001b[0m \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mRuntimeError\u001b[39;00m(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mThe custom device module of \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mmodule\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m has already been registered with \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mattr\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m\"\u001b[39m)\n\u001b[0;32m---> 95\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m_normalization_device\u001b[39m(custom_backend_name: \u001b[38;5;28mstr\u001b[39m, device: Optional[Union[\u001b[38;5;28mint\u001b[39m, \u001b[38;5;28mstr\u001b[39m, \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mdevice\u001b[49m]] \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m \u001b[38;5;28mint\u001b[39m:\n\u001b[1;32m 96\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m_get_current_device_index\u001b[39m():\n\u001b[1;32m 97\u001b[0m _get_device_index \u001b[38;5;241m=\u001b[39m \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mcurrent_device\u001b[39m\u001b[38;5;124m\"\u001b[39m\n",
|
33 |
-
"\u001b[0;31mAttributeError\u001b[0m: module 'torch' has no attribute 'device'"
|
34 |
-
]
|
35 |
-
}
|
36 |
-
],
|
37 |
-
"source": [
|
38 |
-
"from fastai.vision.all import *\n",
|
39 |
-
"import gradio as gr\n",
|
40 |
-
"\n",
|
41 |
-
"def house_room(x): return x[0].isupper()"
|
42 |
-
]
|
43 |
-
},
|
44 |
-
{
|
45 |
-
"cell_type": "code",
|
46 |
-
"execution_count": 44,
|
47 |
-
"metadata": {},
|
48 |
-
"outputs": [],
|
49 |
-
"source": [
|
50 |
-
"#/export\n",
|
51 |
-
"learn_inf = load_learner(path/'export.pkl')"
|
52 |
-
]
|
53 |
-
},
|
54 |
-
{
|
55 |
-
"cell_type": "markdown",
|
56 |
-
"metadata": {},
|
57 |
-
"source": [
|
58 |
-
"When we're doing inference, we're generally just getting predictions for one image at a time. To do this, pass a filename to `predict`:"
|
59 |
-
]
|
60 |
-
},
|
61 |
-
{
|
62 |
-
"cell_type": "code",
|
63 |
-
"execution_count": 72,
|
64 |
-
"metadata": {},
|
65 |
-
"outputs": [
|
66 |
-
{
|
67 |
-
"data": {
|
68 |
-
"text/html": [
|
69 |
-
"\n",
|
70 |
-
"<style>\n",
|
71 |
-
" /* Turns off some styling */\n",
|
72 |
-
" progress {\n",
|
73 |
-
" /* gets rid of default border in Firefox and Opera. */\n",
|
74 |
-
" border: none;\n",
|
75 |
-
" /* Needs to be in here for Safari polyfill so background images work as expected. */\n",
|
76 |
-
" background-size: auto;\n",
|
77 |
-
" }\n",
|
78 |
-
" progress:not([value]), progress:not([value])::-webkit-progress-bar {\n",
|
79 |
-
" background: repeating-linear-gradient(45deg, #7e7e7e, #7e7e7e 10px, #5c5c5c 10px, #5c5c5c 20px);\n",
|
80 |
-
" }\n",
|
81 |
-
" .progress-bar-interrupted, .progress-bar-interrupted::-webkit-progress-bar {\n",
|
82 |
-
" background: #F44336;\n",
|
83 |
-
" }\n",
|
84 |
-
"</style>\n"
|
85 |
-
],
|
86 |
-
"text/plain": [
|
87 |
-
"<IPython.core.display.HTML object>"
|
88 |
-
]
|
89 |
-
},
|
90 |
-
"metadata": {},
|
91 |
-
"output_type": "display_data"
|
92 |
-
},
|
93 |
-
{
|
94 |
-
"data": {
|
95 |
-
"text/html": [],
|
96 |
-
"text/plain": [
|
97 |
-
"<IPython.core.display.HTML object>"
|
98 |
-
]
|
99 |
-
},
|
100 |
-
"metadata": {},
|
101 |
-
"output_type": "display_data"
|
102 |
-
},
|
103 |
-
{
|
104 |
-
"data": {
|
105 |
-
"text/html": [
|
106 |
-
"\n",
|
107 |
-
"<style>\n",
|
108 |
-
" /* Turns off some styling */\n",
|
109 |
-
" progress {\n",
|
110 |
-
" /* gets rid of default border in Firefox and Opera. */\n",
|
111 |
-
" border: none;\n",
|
112 |
-
" /* Needs to be in here for Safari polyfill so background images work as expected. */\n",
|
113 |
-
" background-size: auto;\n",
|
114 |
-
" }\n",
|
115 |
-
" progress:not([value]), progress:not([value])::-webkit-progress-bar {\n",
|
116 |
-
" background: repeating-linear-gradient(45deg, #7e7e7e, #7e7e7e 10px, #5c5c5c 10px, #5c5c5c 20px);\n",
|
117 |
-
" }\n",
|
118 |
-
" .progress-bar-interrupted, .progress-bar-interrupted::-webkit-progress-bar {\n",
|
119 |
-
" background: #F44336;\n",
|
120 |
-
" }\n",
|
121 |
-
"</style>\n"
|
122 |
-
],
|
123 |
-
"text/plain": [
|
124 |
-
"<IPython.core.display.HTML object>"
|
125 |
-
]
|
126 |
-
},
|
127 |
-
"metadata": {},
|
128 |
-
"output_type": "display_data"
|
129 |
-
},
|
130 |
-
{
|
131 |
-
"data": {
|
132 |
-
"text/html": [],
|
133 |
-
"text/plain": [
|
134 |
-
"<IPython.core.display.HTML object>"
|
135 |
-
]
|
136 |
-
},
|
137 |
-
"metadata": {},
|
138 |
-
"output_type": "display_data"
|
139 |
-
},
|
140 |
-
{
|
141 |
-
"name": "stdout",
|
142 |
-
"output_type": "stream",
|
143 |
-
"text": [
|
144 |
-
"CPU times: user 210 ms, sys: 12.9 ms, total: 223 ms\n",
|
145 |
-
"Wall time: 39 ms\n"
|
146 |
-
]
|
147 |
-
},
|
148 |
-
{
|
149 |
-
"data": {
|
150 |
-
"text/plain": [
|
151 |
-
"('Livingroom',\n",
|
152 |
-
" tensor(4),\n",
|
153 |
-
" tensor([4.0148e-04, 1.8107e-01, 4.3362e-05, 5.1020e-04, 8.1798e-01]))"
|
154 |
-
]
|
155 |
-
},
|
156 |
-
"execution_count": 72,
|
157 |
-
"metadata": {},
|
158 |
-
"output_type": "execute_result"
|
159 |
-
}
|
160 |
-
],
|
161 |
-
"source": [
|
162 |
-
"im = \"PXL_20240421_110831643.jpg\"\n",
|
163 |
-
"learn_inf.predict(im)\n",
|
164 |
-
"%time learn.predict(im)"
|
165 |
-
]
|
166 |
-
},
|
167 |
-
{
|
168 |
-
"cell_type": "markdown",
|
169 |
-
"metadata": {},
|
170 |
-
"source": [
|
171 |
-
"This has returned three things: the predicted category in the same format you originally provided (in this case that's a string), the index of the predicted category, and the probabilities of each category. The last two are based on the order of categories in the *vocab* of the `DataLoaders`; that is, the stored list of all possible categories. At inference time, you can access the `DataLoaders` as an attribute of the `Learner`:"
|
172 |
-
]
|
173 |
-
},
|
174 |
-
{
|
175 |
-
"cell_type": "code",
|
176 |
-
"execution_count": 48,
|
177 |
-
"metadata": {},
|
178 |
-
"outputs": [
|
179 |
-
{
|
180 |
-
"data": {
|
181 |
-
"text/plain": [
|
182 |
-
"['Bathroom', 'Bedroom', 'Dinning', 'Kitchen', 'Livingroom']"
|
183 |
-
]
|
184 |
-
},
|
185 |
-
"execution_count": 48,
|
186 |
-
"metadata": {},
|
187 |
-
"output_type": "execute_result"
|
188 |
-
}
|
189 |
-
],
|
190 |
-
"source": [
|
191 |
-
"learn_inf.dls.vocab"
|
192 |
-
]
|
193 |
-
},
|
194 |
-
{
|
195 |
-
"cell_type": "code",
|
196 |
-
"execution_count": 73,
|
197 |
-
"metadata": {},
|
198 |
-
"outputs": [],
|
199 |
-
"source": [
|
200 |
-
"#/export\n",
|
201 |
-
"categories = learn_inf.dls.vocab\n",
|
202 |
-
"def classify_image(img):\n",
|
203 |
-
" pred,idx,probs = learn.predict(img)\n",
|
204 |
-
" return dict(zip(categories, map(float,probs)))"
|
205 |
-
]
|
206 |
-
},
|
207 |
-
{
|
208 |
-
"cell_type": "code",
|
209 |
-
"execution_count": 74,
|
210 |
-
"metadata": {},
|
211 |
-
"outputs": [
|
212 |
-
{
|
213 |
-
"data": {
|
214 |
-
"text/html": [
|
215 |
-
"\n",
|
216 |
-
"<style>\n",
|
217 |
-
" /* Turns off some styling */\n",
|
218 |
-
" progress {\n",
|
219 |
-
" /* gets rid of default border in Firefox and Opera. */\n",
|
220 |
-
" border: none;\n",
|
221 |
-
" /* Needs to be in here for Safari polyfill so background images work as expected. */\n",
|
222 |
-
" background-size: auto;\n",
|
223 |
-
" }\n",
|
224 |
-
" progress:not([value]), progress:not([value])::-webkit-progress-bar {\n",
|
225 |
-
" background: repeating-linear-gradient(45deg, #7e7e7e, #7e7e7e 10px, #5c5c5c 10px, #5c5c5c 20px);\n",
|
226 |
-
" }\n",
|
227 |
-
" .progress-bar-interrupted, .progress-bar-interrupted::-webkit-progress-bar {\n",
|
228 |
-
" background: #F44336;\n",
|
229 |
-
" }\n",
|
230 |
-
"</style>\n"
|
231 |
-
],
|
232 |
-
"text/plain": [
|
233 |
-
"<IPython.core.display.HTML object>"
|
234 |
-
]
|
235 |
-
},
|
236 |
-
"metadata": {},
|
237 |
-
"output_type": "display_data"
|
238 |
-
},
|
239 |
-
{
|
240 |
-
"data": {
|
241 |
-
"text/html": [],
|
242 |
-
"text/plain": [
|
243 |
-
"<IPython.core.display.HTML object>"
|
244 |
-
]
|
245 |
-
},
|
246 |
-
"metadata": {},
|
247 |
-
"output_type": "display_data"
|
248 |
-
},
|
249 |
-
{
|
250 |
-
"data": {
|
251 |
-
"text/plain": [
|
252 |
-
"{'Bathroom': 0.0004014758742414415,\n",
|
253 |
-
" 'Bedroom': 0.18106792867183685,\n",
|
254 |
-
" 'Dinning': 4.3361960706533864e-05,\n",
|
255 |
-
" 'Kitchen': 0.0005102035356685519,\n",
|
256 |
-
" 'Livingroom': 0.8179770112037659}"
|
257 |
-
]
|
258 |
-
},
|
259 |
-
"execution_count": 74,
|
260 |
-
"metadata": {},
|
261 |
-
"output_type": "execute_result"
|
262 |
-
}
|
263 |
-
],
|
264 |
-
"source": [
|
265 |
-
"#/export\n",
|
266 |
-
"classify_image(im)"
|
267 |
-
]
|
268 |
-
},
|
269 |
-
{
|
270 |
-
"cell_type": "code",
|
271 |
-
"execution_count": 77,
|
272 |
-
"metadata": {},
|
273 |
-
"outputs": [
|
274 |
-
{
|
275 |
-
"ename": "NameError",
|
276 |
-
"evalue": "name 'gr' is not defined",
|
277 |
-
"output_type": "error",
|
278 |
-
"traceback": [
|
279 |
-
"\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
|
280 |
-
"\u001b[0;31mNameError\u001b[0m Traceback (most recent call last)",
|
281 |
-
"Cell \u001b[0;32mIn[77], line 2\u001b[0m\n\u001b[1;32m 1\u001b[0m \u001b[38;5;66;03m#/export\u001b[39;00m\n\u001b[0;32m----> 2\u001b[0m image \u001b[38;5;241m=\u001b[39m \u001b[43mgr\u001b[49m\u001b[38;5;241m.\u001b[39minputs\u001b[38;5;241m.\u001b[39mImage(shape\u001b[38;5;241m=\u001b[39m(\u001b[38;5;241m192\u001b[39m,\u001b[38;5;241m192\u001b[39m))\n\u001b[1;32m 3\u001b[0m label \u001b[38;5;241m=\u001b[39m gr\u001b[38;5;241m.\u001b[39moutputs\u001b[38;5;241m.\u001b[39mLabel()\n\u001b[1;32m 4\u001b[0m examples \u001b[38;5;241m=\u001b[39m [\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mbathroom.jpg\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mbedroom.jpg\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mdining.jpg\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mkitchen.jpg\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlivingroom.jpg\u001b[39m\u001b[38;5;124m'\u001b[39m]\n",
|
282 |
-
"\u001b[0;31mNameError\u001b[0m: name 'gr' is not defined"
|
283 |
-
]
|
284 |
-
}
|
285 |
-
],
|
286 |
-
"source": [
|
287 |
-
"#/export\n",
|
288 |
-
"image = gr.inputs.Image(shape=(192,192))\n",
|
289 |
-
"label = gr.outputs.Label()\n",
|
290 |
-
"examples = ['bathroom.jpg', 'bedroom.jpg', 'dining.jpg', 'kitchen.jpg', 'livingroom.jpg']\n",
|
291 |
-
"\n",
|
292 |
-
"intf = gr.Interface(fn=classify_image, inputs=image, outputs=label, examples=examples)\n",
|
293 |
-
"intf.launch(inline=False)"
|
294 |
-
]
|
295 |
-
},
|
296 |
-
{
|
297 |
-
"cell_type": "markdown",
|
298 |
-
"metadata": {},
|
299 |
-
"source": [
|
300 |
-
"### Creating a Notebook App from the Model"
|
301 |
-
]
|
302 |
-
},
|
303 |
-
{
|
304 |
-
"cell_type": "markdown",
|
305 |
-
"metadata": {},
|
306 |
-
"source": [
|
307 |
-
"To use our model in an application, we can simply treat the `predict` method as a regular function. Therefore, creating an app from the model can be done using any of the myriad of frameworks and techniques available to application developers.\n",
|
308 |
-
"\n",
|
309 |
-
"However, most data scientists are not familiar with the world of web application development. So let's try using something that you do, at this point, know: it turns out that we can create a complete working web application using nothing but Jupyter notebooks! The two things we need to make this happen are:\n",
|
310 |
-
"\n",
|
311 |
-
"- IPython widgets (ipywidgets)\n",
|
312 |
-
"- Voilà\n",
|
313 |
-
"\n",
|
314 |
-
"*IPython widgets* are GUI components that bring together JavaScript and Python functionality in a web browser, and can be created and used within a Jupyter notebook. For instance, the image cleaner that we saw earlier in this chapter is entirely written with IPython widgets. However, we don't want to require users of our application to run Jupyter themselves.\n",
|
315 |
-
"\n",
|
316 |
-
"That is why *Voilà* exists. It is a system for making applications consisting of IPython widgets available to end users, without them having to use Jupyter at all. Voilà is taking advantage of the fact that a notebook _already is_ a kind of web application, just a rather complex one that depends on another web application: Jupyter itself. Essentially, it helps us automatically convert the complex web application we've already implicitly made (the notebook) into a simpler, easier-to-deploy web application, which functions like a normal web application rather than like a notebook.\n",
|
317 |
-
"\n",
|
318 |
-
"But we still have the advantage of developing in a notebook, so with ipywidgets, we can build up our GUI step by step. We will use this approach to create a simple image classifier. First, we need a file upload widget:"
|
319 |
-
]
|
320 |
-
},
|
321 |
-
{
|
322 |
-
"cell_type": "code",
|
323 |
-
"execution_count": 25,
|
324 |
-
"metadata": {},
|
325 |
-
"outputs": [
|
326 |
-
{
|
327 |
-
"data": {
|
328 |
-
"application/vnd.jupyter.widget-view+json": {
|
329 |
-
"model_id": "9c0397bc309c4503be967f16039029bd",
|
330 |
-
"version_major": 2,
|
331 |
-
"version_minor": 0
|
332 |
-
},
|
333 |
-
"text/plain": [
|
334 |
-
"FileUpload(value={}, description='Upload')"
|
335 |
-
]
|
336 |
-
},
|
337 |
-
"metadata": {},
|
338 |
-
"output_type": "display_data"
|
339 |
-
}
|
340 |
-
],
|
341 |
-
"source": [
|
342 |
-
"#hide_output\n",
|
343 |
-
"btn_upload = widgets.FileUpload()\n",
|
344 |
-
"btn_upload"
|
345 |
-
]
|
346 |
-
},
|
347 |
-
{
|
348 |
-
"cell_type": "markdown",
|
349 |
-
"metadata": {},
|
350 |
-
"source": [
|
351 |
-
"<img alt=\"An upload button\" width=\"159\" src=\"images/att_00008.png\">\n",
|
352 |
-
"\n",
|
353 |
-
"Now we can grab the image:"
|
354 |
-
]
|
355 |
-
},
|
356 |
-
{
|
357 |
-
"cell_type": "code",
|
358 |
-
"execution_count": 26,
|
359 |
-
"metadata": {
|
360 |
-
"hide_input": true
|
361 |
-
},
|
362 |
-
"outputs": [],
|
363 |
-
"source": [
|
364 |
-
"#hide\n",
|
365 |
-
"# For the book, we can't actually click an upload button, so we fake it\n",
|
366 |
-
"btn_upload = SimpleNamespace(data = ['images/grizzly.jpg'])"
|
367 |
-
]
|
368 |
-
},
|
369 |
-
{
|
370 |
-
"cell_type": "code",
|
371 |
-
"execution_count": 27,
|
372 |
-
"metadata": {},
|
373 |
-
"outputs": [],
|
374 |
-
"source": [
|
375 |
-
"img = PILImage.create(btn_upload.data[-1])"
|
376 |
-
]
|
377 |
-
},
|
378 |
-
{
|
379 |
-
"cell_type": "markdown",
|
380 |
-
"metadata": {},
|
381 |
-
"source": [
|
382 |
-
"<img alt=\"Output widget representing the image\" width=\"117\" src=\"images/att_00009.png\">"
|
383 |
-
]
|
384 |
-
},
|
385 |
-
{
|
386 |
-
"cell_type": "markdown",
|
387 |
-
"metadata": {},
|
388 |
-
"source": [
|
389 |
-
"We can use an `Output` widget to display it:"
|
390 |
-
]
|
391 |
-
},
|
392 |
-
{
|
393 |
-
"cell_type": "code",
|
394 |
-
"execution_count": 28,
|
395 |
-
"metadata": {},
|
396 |
-
"outputs": [
|
397 |
-
{
|
398 |
-
"data": {
|
399 |
-
"application/vnd.jupyter.widget-view+json": {
|
400 |
-
"model_id": "08dd6699b63a4af0a71a941384d0741c",
|
401 |
-
"version_major": 2,
|
402 |
-
"version_minor": 0
|
403 |
-
},
|
404 |
-
"text/plain": [
|
405 |
-
"Output()"
|
406 |
-
]
|
407 |
-
},
|
408 |
-
"metadata": {},
|
409 |
-
"output_type": "display_data"
|
410 |
-
}
|
411 |
-
],
|
412 |
-
"source": [
|
413 |
-
"#hide_output\n",
|
414 |
-
"out_pl = widgets.Output()\n",
|
415 |
-
"out_pl.clear_output()\n",
|
416 |
-
"with out_pl: display(img.to_thumb(128,128))\n",
|
417 |
-
"out_pl"
|
418 |
-
]
|
419 |
-
},
|
420 |
-
{
|
421 |
-
"cell_type": "markdown",
|
422 |
-
"metadata": {},
|
423 |
-
"source": [
|
424 |
-
"<img alt=\"Output widget representing the image\" width=\"117\" src=\"images/att_00009.png\">\n",
|
425 |
-
"\n",
|
426 |
-
"Then we can get our predictions:"
|
427 |
-
]
|
428 |
-
},
|
429 |
-
{
|
430 |
-
"cell_type": "code",
|
431 |
-
"execution_count": 29,
|
432 |
-
"metadata": {},
|
433 |
-
"outputs": [
|
434 |
-
{
|
435 |
-
"data": {
|
436 |
-
"text/html": [
|
437 |
-
"\n",
|
438 |
-
"<style>\n",
|
439 |
-
" /* Turns off some styling */\n",
|
440 |
-
" progress {\n",
|
441 |
-
" /* gets rid of default border in Firefox and Opera. */\n",
|
442 |
-
" border: none;\n",
|
443 |
-
" /* Needs to be in here for Safari polyfill so background images work as expected. */\n",
|
444 |
-
" background-size: auto;\n",
|
445 |
-
" }\n",
|
446 |
-
" progress:not([value]), progress:not([value])::-webkit-progress-bar {\n",
|
447 |
-
" background: repeating-linear-gradient(45deg, #7e7e7e, #7e7e7e 10px, #5c5c5c 10px, #5c5c5c 20px);\n",
|
448 |
-
" }\n",
|
449 |
-
" .progress-bar-interrupted, .progress-bar-interrupted::-webkit-progress-bar {\n",
|
450 |
-
" background: #F44336;\n",
|
451 |
-
" }\n",
|
452 |
-
"</style>\n"
|
453 |
-
],
|
454 |
-
"text/plain": [
|
455 |
-
"<IPython.core.display.HTML object>"
|
456 |
-
]
|
457 |
-
},
|
458 |
-
"metadata": {},
|
459 |
-
"output_type": "display_data"
|
460 |
-
},
|
461 |
-
{
|
462 |
-
"data": {
|
463 |
-
"text/html": [],
|
464 |
-
"text/plain": [
|
465 |
-
"<IPython.core.display.HTML object>"
|
466 |
-
]
|
467 |
-
},
|
468 |
-
"metadata": {},
|
469 |
-
"output_type": "display_data"
|
470 |
-
}
|
471 |
-
],
|
472 |
-
"source": [
|
473 |
-
"pred,pred_idx,probs = learn_inf.predict(img)"
|
474 |
-
]
|
475 |
-
},
|
476 |
-
{
|
477 |
-
"cell_type": "markdown",
|
478 |
-
"metadata": {},
|
479 |
-
"source": [
|
480 |
-
"and use a `Label` to display them:"
|
481 |
-
]
|
482 |
-
},
|
483 |
-
{
|
484 |
-
"cell_type": "code",
|
485 |
-
"execution_count": 30,
|
486 |
-
"metadata": {},
|
487 |
-
"outputs": [
|
488 |
-
{
|
489 |
-
"data": {
|
490 |
-
"application/vnd.jupyter.widget-view+json": {
|
491 |
-
"model_id": "a5a4fc8d858745d8831644b7ddec31d8",
|
492 |
-
"version_major": 2,
|
493 |
-
"version_minor": 0
|
494 |
-
},
|
495 |
-
"text/plain": [
|
496 |
-
"Label(value='Prediction: grizzly; Probability: 0.9999')"
|
497 |
-
]
|
498 |
-
},
|
499 |
-
"metadata": {},
|
500 |
-
"output_type": "display_data"
|
501 |
-
}
|
502 |
-
],
|
503 |
-
"source": [
|
504 |
-
"#hide_output\n",
|
505 |
-
"lbl_pred = widgets.Label()\n",
|
506 |
-
"lbl_pred.value = f'Prediction: {pred}; Probability: {probs[pred_idx]:.04f}'\n",
|
507 |
-
"lbl_pred"
|
508 |
-
]
|
509 |
-
},
|
510 |
-
{
|
511 |
-
"cell_type": "markdown",
|
512 |
-
"metadata": {},
|
513 |
-
"source": [
|
514 |
-
"`Prediction: grizzly; Probability: 1.0000`\n",
|
515 |
-
"\n",
|
516 |
-
"We'll need a button to do the classification. It looks exactly like the upload button:"
|
517 |
-
]
|
518 |
-
},
|
519 |
-
{
|
520 |
-
"cell_type": "code",
|
521 |
-
"execution_count": 33,
|
522 |
-
"metadata": {},
|
523 |
-
"outputs": [
|
524 |
-
{
|
525 |
-
"data": {
|
526 |
-
"application/vnd.jupyter.widget-view+json": {
|
527 |
-
"model_id": "0af1f4ea49ef463987deb9848c37251e",
|
528 |
-
"version_major": 2,
|
529 |
-
"version_minor": 0
|
530 |
-
},
|
531 |
-
"text/plain": [
|
532 |
-
"Button(description='Classify', style=ButtonStyle())"
|
533 |
-
]
|
534 |
-
},
|
535 |
-
"metadata": {},
|
536 |
-
"output_type": "display_data"
|
537 |
-
}
|
538 |
-
],
|
539 |
-
"source": [
|
540 |
-
"#hide_output\n",
|
541 |
-
"btn_run = widgets.Button(description='Classify')\n",
|
542 |
-
"btn_run"
|
543 |
-
]
|
544 |
-
},
|
545 |
-
{
|
546 |
-
"cell_type": "markdown",
|
547 |
-
"metadata": {},
|
548 |
-
"source": [
|
549 |
-
"We'll also need a *click event handler*; that is, a function that will be called when it's pressed. We can just copy over the lines of code from above:"
|
550 |
-
]
|
551 |
-
},
|
552 |
-
{
|
553 |
-
"cell_type": "code",
|
554 |
-
"execution_count": 34,
|
555 |
-
"metadata": {},
|
556 |
-
"outputs": [],
|
557 |
-
"source": [
|
558 |
-
"def on_click_classify(change):\n",
|
559 |
-
" img = PILImage.create(btn_upload.data[-1])\n",
|
560 |
-
" out_pl.clear_output()\n",
|
561 |
-
" with out_pl: display(img.to_thumb(128,128))\n",
|
562 |
-
" pred,pred_idx,probs = learn_inf.predict(img)\n",
|
563 |
-
" lbl_pred.value = f'Prediction: {pred}; Probability: {probs[pred_idx]:.04f}'\n",
|
564 |
-
"\n",
|
565 |
-
"btn_run.on_click(on_click_classify)"
|
566 |
-
]
|
567 |
-
},
|
568 |
-
{
|
569 |
-
"cell_type": "markdown",
|
570 |
-
"metadata": {},
|
571 |
-
"source": [
|
572 |
-
"You can test the button now by pressing it, and you should see the image and predictions update automatically!\n",
|
573 |
-
"\n",
|
574 |
-
"We can now put them all in a vertical box (`VBox`) to complete our GUI:"
|
575 |
-
]
|
576 |
-
},
|
577 |
-
{
|
578 |
-
"cell_type": "code",
|
579 |
-
"execution_count": 35,
|
580 |
-
"metadata": {},
|
581 |
-
"outputs": [],
|
582 |
-
"source": [
|
583 |
-
"#hide\n",
|
584 |
-
"#Putting back btn_upload to a widget for next cell\n",
|
585 |
-
"btn_upload = widgets.FileUpload()"
|
586 |
-
]
|
587 |
-
},
|
588 |
-
{
|
589 |
-
"cell_type": "code",
|
590 |
-
"execution_count": 36,
|
591 |
-
"metadata": {},
|
592 |
-
"outputs": [
|
593 |
-
{
|
594 |
-
"data": {
|
595 |
-
"application/vnd.jupyter.widget-view+json": {
|
596 |
-
"model_id": "5e86351c73224e6d8de42753c55ac94b",
|
597 |
-
"version_major": 2,
|
598 |
-
"version_minor": 0
|
599 |
-
},
|
600 |
-
"text/plain": [
|
601 |
-
"VBox(children=(Label(value='Select your bear!'), FileUpload(value={}, description='Upload'), Button(descriptio…"
|
602 |
-
]
|
603 |
-
},
|
604 |
-
"metadata": {},
|
605 |
-
"output_type": "display_data"
|
606 |
-
}
|
607 |
-
],
|
608 |
-
"source": [
|
609 |
-
"#hide_output\n",
|
610 |
-
"VBox([widgets.Label('Select your bear!'), \n",
|
611 |
-
" btn_upload, btn_run, out_pl, lbl_pred])"
|
612 |
-
]
|
613 |
-
},
|
614 |
-
{
|
615 |
-
"cell_type": "markdown",
|
616 |
-
"metadata": {},
|
617 |
-
"source": [
|
618 |
-
"<img alt=\"The whole widget\" width=\"233\" src=\"images/att_00011.png\">"
|
619 |
-
]
|
620 |
-
},
|
621 |
-
{
|
622 |
-
"cell_type": "markdown",
|
623 |
-
"metadata": {},
|
624 |
-
"source": [
|
625 |
-
"We have written all the code necessary for our app. The next step is to convert it into something we can deploy."
|
626 |
-
]
|
627 |
-
},
|
628 |
-
{
|
629 |
-
"cell_type": "markdown",
|
630 |
-
"metadata": {},
|
631 |
-
"source": [
|
632 |
-
"### Turning Your Notebook into a Real App"
|
633 |
-
]
|
634 |
-
},
|
635 |
-
{
|
636 |
-
"cell_type": "code",
|
637 |
-
"execution_count": 37,
|
638 |
-
"metadata": {},
|
639 |
-
"outputs": [],
|
640 |
-
"source": [
|
641 |
-
"#hide\n",
|
642 |
-
"# !pip install voila\n",
|
643 |
-
"# !jupyter serverextension enable --sys-prefix voila "
|
644 |
-
]
|
645 |
-
},
|
646 |
-
{
|
647 |
-
"cell_type": "markdown",
|
648 |
-
"metadata": {},
|
649 |
-
"source": [
|
650 |
-
"Now that we have everything working in this Jupyter notebook, we can create our application. To do this, start a new notebook and add to it only the code needed to create and show the widgets that you need, and markdown for any text that you want to appear. Have a look at the *bear_classifier* notebook in the book's repo to see the simple notebook application we created.\n",
|
651 |
-
"\n",
|
652 |
-
"Next, install Voilà if you haven't already, by copying these lines into a notebook cell and executing it:\n",
|
653 |
-
"\n",
|
654 |
-
" !pip install voila\n",
|
655 |
-
" !jupyter serverextension enable --sys-prefix voila\n",
|
656 |
-
"\n",
|
657 |
-
"Cells that begin with a `!` do not contain Python code, but instead contain code that is passed to your shell (bash, Windows PowerShell, etc.). If you are comfortable using the command line, which we'll discuss more later in this book, you can of course simply type these two lines (without the `!` prefix) directly into your terminal. In this case, the first line installs the `voila` library and application, and the second connects it to your existing Jupyter notebook.\n",
|
658 |
-
"\n",
|
659 |
-
"Voilà runs Jupyter notebooks just like the Jupyter notebook server you are using now does, but it also does something very important: it removes all of the cell inputs, and only shows output (including ipywidgets), along with your markdown cells. So what's left is a web application! To view your notebook as a Voilà web application, replace the word \"notebooks\" in your browser's URL with: \"voila/render\". You will see the same content as your notebook, but without any of the code cells.\n",
|
660 |
-
"\n",
|
661 |
-
"Of course, you don't need to use Voilà or ipywidgets. Your model is just a function you can call (`pred,pred_idx,probs = learn.predict(img)`), so you can use it with any framework, hosted on any platform. And you can take something you've prototyped in ipywidgets and Voilà and later convert it into a regular web application. We're showing you this approach in the book because we think it's a great way for data scientists and other folks that aren't web development experts to create applications from their models.\n",
|
662 |
-
"\n",
|
663 |
-
"We have our app, now let's deploy it!"
|
664 |
-
]
|
665 |
-
},
|
666 |
-
{
|
667 |
-
"cell_type": "markdown",
|
668 |
-
"metadata": {},
|
669 |
-
"source": [
|
670 |
-
"### Deploying your app"
|
671 |
-
]
|
672 |
-
},
|
673 |
-
{
|
674 |
-
"cell_type": "markdown",
|
675 |
-
"metadata": {},
|
676 |
-
"source": [
|
677 |
-
"As you now know, you need a GPU to train nearly any useful deep learning model. So, do you need a GPU to use that model in production? No! You almost certainly *do not need a GPU to serve your model in production*. There are a few reasons for this:\n",
|
678 |
-
"\n",
|
679 |
-
"- As we've seen, GPUs are only useful when they do lots of identical work in parallel. If you're doing (say) image classification, then you'll normally be classifying just one user's image at a time, and there isn't normally enough work to do in a single image to keep a GPU busy for long enough for it to be very efficient. So, a CPU will often be more cost-effective.\n",
|
680 |
-
"- An alternative could be to wait for a few users to submit their images, and then batch them up and process them all at once on a GPU. But then you're asking your users to wait, rather than getting answers straight away! And you need a high-volume site for this to be workable. If you do need this functionality, you can use a tool such as Microsoft's [ONNX Runtime](https://github.com/microsoft/onnxruntime), or [AWS Sagemaker](https://aws.amazon.com/sagemaker/)\n",
|
681 |
-
"- The complexities of dealing with GPU inference are significant. In particular, the GPU's memory will need careful manual management, and you'll need a careful queueing system to ensure you only process one batch at a time.\n",
|
682 |
-
"- There's a lot more market competition in CPU than GPU servers, as a result of which there are much cheaper options available for CPU servers.\n",
|
683 |
-
"\n",
|
684 |
-
"Because of the complexity of GPU serving, many systems have sprung up to try to automate this. However, managing and running these systems is also complex, and generally requires compiling your model into a different form that's specialized for that system. It's typically preferable to avoid dealing with this complexity until/unless your app gets popular enough that it makes clear financial sense for you to do so."
|
685 |
-
]
|
686 |
-
},
|
687 |
-
{
|
688 |
-
"cell_type": "markdown",
|
689 |
-
"metadata": {},
|
690 |
-
"source": [
|
691 |
-
"For at least the initial prototype of your application, and for any hobby projects that you want to show off, you can easily host them for free. The best place and the best way to do this will vary over time, so check the [book's website](https://book.fast.ai/) for the most up-to-date recommendations. As we're writing this book in early 2020 the simplest (and free!) approach is to use [Binder](https://mybinder.org/). To publish your web app on Binder, you follow these steps:\n",
|
692 |
-
"\n",
|
693 |
-
"1. Add your notebook to a [GitHub repository](http://github.com/).\n",
|
694 |
-
"2. Paste the URL of that repo into Binder's URL, as shown in <<deploy-binder>>.\n",
|
695 |
-
"3. Change the File dropdown to instead select URL.\n",
|
696 |
-
"4. In the \"URL to open\" field, enter `/voila/render/name.ipynb` (replacing `name` with the name of for your notebook).\n",
|
697 |
-
"5. Click the clickboard button at the bottom right to copy the URL and paste it somewhere safe. \n",
|
698 |
-
"6. Click Launch."
|
699 |
-
]
|
700 |
-
},
|
701 |
-
{
|
702 |
-
"cell_type": "markdown",
|
703 |
-
"metadata": {},
|
704 |
-
"source": [
|
705 |
-
"<img alt=\"Deploying to Binder\" width=\"800\" caption=\"Deploying to Binder\" id=\"deploy-binder\" src=\"images/att_00001.png\">"
|
706 |
-
]
|
707 |
-
},
|
708 |
-
{
|
709 |
-
"cell_type": "markdown",
|
710 |
-
"metadata": {},
|
711 |
-
"source": [
|
712 |
-
"The first time you do this, Binder will take around 5 minutes to build your site. Behind the scenes, it is finding a virtual machine that can run your app, allocating storage, collecting the files needed for Jupyter, for your notebook, and for presenting your notebook as a web application.\n",
|
713 |
-
"\n",
|
714 |
-
"Finally, once it has started the app running, it will navigate your browser to your new web app. You can share the URL you copied to allow others to access your app as well.\n",
|
715 |
-
"\n",
|
716 |
-
"For other (both free and paid) options for deploying your web app, be sure to take a look at the [book's website](https://book.fast.ai/)."
|
717 |
-
]
|
718 |
-
},
|
719 |
-
{
|
720 |
-
"cell_type": "markdown",
|
721 |
-
"metadata": {},
|
722 |
-
"source": [
|
723 |
-
"You may well want to deploy your application onto mobile devices, or edge devices such as a Raspberry Pi. There are a lot of libraries and frameworks that allow you to integrate a model directly into a mobile application. However, these approaches tend to require a lot of extra steps and boilerplate, and do not always support all the PyTorch and fastai layers that your model might use. In addition, the work you do will depend on what kind of mobile devices you are targeting for deployment—you might need to do some work to run on iOS devices, different work to run on newer Android devices, different work for older Android devices, etc. Instead, we recommend wherever possible that you deploy the model itself to a server, and have your mobile or edge application connect to it as a web service.\n",
|
724 |
-
"\n",
|
725 |
-
"There are quite a few upsides to this approach. The initial installation is easier, because you only have to deploy a small GUI application, which connects to the server to do all the heavy lifting. More importantly perhaps, upgrades of that core logic can happen on your server, rather than needing to be distributed to all of your users. Your server will have a lot more memory and processing capacity than most edge devices, and it is far easier to scale those resources if your model becomes more demanding. The hardware that you will have on a server is also going to be more standard and more easily supported by fastai and PyTorch, so you don't have to compile your model into a different form.\n",
|
726 |
-
"\n",
|
727 |
-
"There are downsides too, of course. Your application will require a network connection, and there will be some latency each time the model is called. (It takes a while for a neural network model to run anyway, so this additional network latency may not make a big difference to your users in practice. In fact, since you can use better hardware on the server, the overall latency may even be less than if it were running locally!) Also, if your application uses sensitive data then your users may be concerned about an approach which sends that data to a remote server, so sometimes privacy considerations will mean that you need to run the model on the edge device (it may be possible to avoid this by having an *on-premise* server, such as inside a company's firewall). Managing the complexity and scaling the server can create additional overhead too, whereas if your model runs on the edge devices then each user is bringing their own compute resources, which leads to easier scaling with an increasing number of users (also known as *horizontal scaling*)."
|
728 |
-
]
|
729 |
-
},
|
730 |
-
{
|
731 |
-
"cell_type": "markdown",
|
732 |
-
"metadata": {},
|
733 |
-
"source": [
|
734 |
-
"> A: I've had a chance to see up close how the mobile ML landscape is changing in my work. We offer an iPhone app that depends on computer vision, and for years we ran our own computer vision models in the cloud. This was the only way to do it then since those models needed significant memory and compute resources and took minutes to process inputs. This approach required building not only the models (fun!) but also the infrastructure to ensure a certain number of \"compute worker machines\" were absolutely always running (scary), that more machines would automatically come online if traffic increased, that there was stable storage for large inputs and outputs, that the iOS app could know and tell the user how their job was doing, etc. Nowadays Apple provides APIs for converting models to run efficiently on device and most iOS devices have dedicated ML hardware, so that's the strategy we use for our newer models. It's still not easy but in our case it's worth it, for a faster user experience and to worry less about servers. What works for you will depend, realistically, on the user experience you're trying to create and what you personally find is easy to do. If you really know how to run servers, do it. If you really know how to build native mobile apps, do that. There are many roads up the hill.\n",
|
735 |
-
"\n",
|
736 |
-
"Overall, we'd recommend using a simple CPU-based server approach where possible, for as long as you can get away with it. If you're lucky enough to have a very successful application, then you'll be able to justify the investment in more complex deployment approaches at that time.\n",
|
737 |
-
"\n",
|
738 |
-
"Congratulations, you have successfully built a deep learning model and deployed it! Now is a good time to take a pause and think about what could go wrong."
|
739 |
-
]
|
740 |
-
},
|
741 |
-
{
|
742 |
-
"cell_type": "markdown",
|
743 |
-
"metadata": {},
|
744 |
-
"source": [
|
745 |
-
"## How to Avoid Disaster"
|
746 |
-
]
|
747 |
-
},
|
748 |
-
{
|
749 |
-
"cell_type": "markdown",
|
750 |
-
"metadata": {},
|
751 |
-
"source": [
|
752 |
-
"In practice, a deep learning model will be just one piece of a much bigger system. As we discussed at the start of this chapter, a data product requires thinking about the entire end-to-end process, from conception to use in production. In this book, we can't hope to cover all the complexity of managing deployed data products, such as managing multiple versions of models, A/B testing, canarying, refreshing the data (should we just grow and grow our datasets all the time, or should we regularly remove some of the old data?), handling data labeling, monitoring all this, detecting model rot, and so forth. In this section we will give an overview of some of the most important issues to consider; for a more detailed discussion of deployment issues we refer to you to the excellent [Building Machine Learning Powered Applications](http://shop.oreilly.com/product/0636920215912.do) by Emmanuel Ameisen (O'Reilly)\n",
|
753 |
-
"\n",
|
754 |
-
"One of the biggest issues to consider is that understanding and testing the behavior of a deep learning model is much more difficult than with most other code you write. With normal software development you can analyze the exact steps that the software is taking, and carefully study which of these steps match the desired behavior that you are trying to create. But with a neural network the behavior emerges from the model's attempt to match the training data, rather than being exactly defined.\n",
|
755 |
-
"\n",
|
756 |
-
"This can result in disaster! For instance, let's say we really were rolling out a bear detection system that will be attached to video cameras around campsites in national parks, and will warn campers of incoming room. If we used a model trained with the dataset we downloaded there would be all kinds of problems in practice, such as:\n",
|
757 |
-
"\n",
|
758 |
-
"- Working with video data instead of images\n",
|
759 |
-
"- Handling nighttime images, which may not appear in this dataset\n",
|
760 |
-
"- Dealing with low-resolution camera images\n",
|
761 |
-
"- Ensuring results are returned fast enough to be useful in practice\n",
|
762 |
-
"- Recognizing room in positions that are rarely seen in photos that people post online (for example from behind, partially covered by bushes, or when a long way away from the camera)"
|
763 |
-
]
|
764 |
-
},
|
765 |
-
{
|
766 |
-
"cell_type": "markdown",
|
767 |
-
"metadata": {},
|
768 |
-
"source": [
|
769 |
-
"A big part of the issue is that the kinds of photos that people are most likely to upload to the internet are the kinds of photos that do a good job of clearly and artistically displaying their subject matter—which isn't the kind of input this system is going to be getting. So, we may need to do a lot of our own data collection and labelling to create a useful system.\n",
|
770 |
-
"\n",
|
771 |
-
"This is just one example of the more general problem of *out-of-domain* data. That is to say, there may be data that our model sees in production which is very different to what it saw during training. There isn't really a complete technical solution to this problem; instead, we have to be careful about our approach to rolling out the technology.\n",
|
772 |
-
"\n",
|
773 |
-
"There are other reasons we need to be careful too. One very common problem is *domain shift*, where the type of data that our model sees changes over time. For instance, an insurance company may use a deep learning model as part of its pricing and risk algorithm, but over time the types of customers that the company attracts, and the types of risks they represent, may change so much that the original training data is no longer relevant.\n",
|
774 |
-
"\n",
|
775 |
-
"Out-of-domain data and domain shift are examples of a larger problem: that you can never fully understand the entire behaviour of your neural network. They have far too many parameters to be able to analytically understand all of their possible behaviors. This is the natural downside of their best feature—their flexibility, which enables them to solve complex problems where we may not even be able to fully specify our preferred solution approaches. The good news, however, is that there are ways to mitigate these risks using a carefully thought-out process. The details of this will vary depending on the details of the problem you are solving, but we will attempt to lay out here a high-level approach, summarized in <<deploy_process>>, which we hope will provide useful guidance."
|
776 |
-
]
|
777 |
-
},
|
778 |
-
{
|
779 |
-
"cell_type": "markdown",
|
780 |
-
"metadata": {},
|
781 |
-
"source": [
|
782 |
-
"<img alt=\"Deployment process\" width=\"500\" caption=\"Deployment process\" id=\"deploy_process\" src=\"images/att_00061.png\">"
|
783 |
-
]
|
784 |
-
},
|
785 |
-
{
|
786 |
-
"cell_type": "markdown",
|
787 |
-
"metadata": {},
|
788 |
-
"source": [
|
789 |
-
"Where possible, the first step is to use an entirely manual process, with your deep learning model approach running in parallel but not being used directly to drive any actions. The humans involved in the manual process should look at the deep learning outputs and check whether they make sense. For instance, with our bear classifier a park ranger could have a screen displaying video feeds from all the cameras, with any possible bear sightings simply highlighted in red. The park ranger would still be expected to be just as alert as before the model was deployed; the model is simply helping to check for problems at this point.\n",
|
790 |
-
"\n",
|
791 |
-
"The second step is to try to limit the scope of the model, and have it carefully supervised by people. For instance, do a small geographically and time-constrained trial of the model-driven approach. Rather than rolling our bear classifier out in every national park throughout the country, we could pick a single observation post, for a one-week period, and have a park ranger check each alert before it goes out.\n",
|
792 |
-
"\n",
|
793 |
-
"Then, gradually increase the scope of your rollout. As you do so, ensure that you have really good reporting systems in place, to make sure that you are aware of any significant changes to the actions being taken compared to your manual process. For instance, if the number of bear alerts doubles or halves after rollout of the new system in some location, we should be very concerned. Try to think about all the ways in which your system could go wrong, and then think about what measure or report or picture could reflect that problem, and ensure that your regular reporting includes that information."
|
794 |
-
]
|
795 |
-
},
|
796 |
-
{
|
797 |
-
"cell_type": "markdown",
|
798 |
-
"metadata": {},
|
799 |
-
"source": [
|
800 |
-
"> J: I started a company 20 years ago called _Optimal Decisions_ that used machine learning and optimization to help giant insurance companies set their pricing, impacting tens of billions of dollars of risks. We used the approaches described here to manage the potential downsides of something going wrong. Also, before we worked with our clients to put anything in production, we tried to simulate the impact by testing the end-to-end system on their previous year's data. It was always quite a nerve-wracking process, putting these new algorithms into production, but every rollout was successful."
|
801 |
-
]
|
802 |
-
},
|
803 |
-
{
|
804 |
-
"cell_type": "markdown",
|
805 |
-
"metadata": {},
|
806 |
-
"source": [
|
807 |
-
"### Unforeseen Consequences and Feedback Loops"
|
808 |
-
]
|
809 |
-
},
|
810 |
-
{
|
811 |
-
"cell_type": "markdown",
|
812 |
-
"metadata": {},
|
813 |
-
"source": [
|
814 |
-
"One of the biggest challenges in rolling out a model is that your model may change the behaviour of the system it is a part of. For instance, consider a \"predictive policing\" algorithm that predicts more crime in certain neighborhoods, causing more police officers to be sent to those neighborhoods, which can result in more crimes being recorded in those neighborhoods, and so on. In the Royal Statistical Society paper [\"To Predict and Serve?\"](https://rss.onlinelibrary.wiley.com/doi/full/10.1111/j.1740-9713.2016.00960.x), Kristian Lum and William Isaac observe that: \"predictive policing is aptly named: it is predicting future policing, not future crime.\"\n",
|
815 |
-
"\n",
|
816 |
-
"Part of the issue in this case is that in the presence of bias (which we'll discuss in depth in the next chapter), *feedback loops* can result in negative implications of that bias getting worse and worse. For instance, there are concerns that this is already happening in the US, where there is significant bias in arrest rates on racial grounds. [According to the ACLU](https://www.aclu.org/issues/smart-justice/sentencing-reform/war-marijuana-black-and-white), \"despite roughly equal usage rates, Blacks are 3.73 times more likely than whites to be arrested for marijuana.\" The impact of this bias, along with the rollout of predictive policing algorithms in many parts of the US, led Bärí Williams to [write in the *New York Times*](https://www.nytimes.com/2017/12/02/opinion/sunday/intelligent-policing-and-my-innocent-children.html): \"The same technology that’s the source of so much excitement in my career is being used in law enforcement in ways that could mean that in the coming years, my son, who is 7 now, is more likely to be profiled or arrested—or worse—for no reason other than his race and where we live.\"\n",
|
817 |
-
"\n",
|
818 |
-
"A helpful exercise prior to rolling out a significant machine learning system is to consider this question: \"What would happen if it went really, really well?\" In other words, what if the predictive power was extremely high, and its ability to influence behavior was extremely significant? In that case, who would be most impacted? What would the most extreme results potentially look like? How would you know what was really going on?\n",
|
819 |
-
"\n",
|
820 |
-
"Such a thought exercise might help you to construct a more careful rollout plan, with ongoing monitoring systems and human oversight. Of course, human oversight isn't useful if it isn't listened to, so make sure that there are reliable and resilient communication channels so that the right people will be aware of issues, and will have the power to fix them."
|
821 |
-
]
|
822 |
-
},
|
823 |
-
{
|
824 |
-
"cell_type": "markdown",
|
825 |
-
"metadata": {},
|
826 |
-
"source": [
|
827 |
-
"## Get Writing!"
|
828 |
-
]
|
829 |
-
},
|
830 |
-
{
|
831 |
-
"cell_type": "markdown",
|
832 |
-
"metadata": {},
|
833 |
-
"source": [
|
834 |
-
"One of the things our students have found most helpful to solidify their understanding of this material is to write it down. There is no better test of your understanding of a topic than attempting to teach it to somebody else. This is helpful even if you never show your writing to anybody—but it's even better if you share it! So we recommend that, if you haven't already, you start a blog. Now that you've completed Chapter 2 and have learned how to train and deploy models, you're well placed to write your first blog post about your deep learning journey. What's surprised you? What opportunities do you see for deep learning in your field? What obstacles do you see?\n",
|
835 |
-
"\n",
|
836 |
-
"Rachel Thomas, cofounder of fast.ai, wrote in the article [\"Why You (Yes, You) Should Blog\"](https://medium.com/@racheltho/why-you-yes-you-should-blog-7d2544ac1045):\n",
|
837 |
-
"\n",
|
838 |
-
"```asciidoc\n",
|
839 |
-
"____\n",
|
840 |
-
"The top advice I would give my younger self would be to start blogging sooner. Here are some reasons to blog:\n",
|
841 |
-
"\n",
|
842 |
-
"* It’s like a resume, only better. I know of a few people who have had blog posts lead to job offers!\n",
|
843 |
-
"* Helps you learn. Organizing knowledge always helps me synthesize my own ideas. One of the tests of whether you understand something is whether you can explain it to someone else. A blog post is a great way to do that.\n",
|
844 |
-
"* I’ve gotten invitations to conferences and invitations to speak from my blog posts. I was invited to the TensorFlow Dev Summit (which was awesome!) for writing a blog post about how I don’t like TensorFlow.\n",
|
845 |
-
"* Meet new people. I’ve met several people who have responded to blog posts I wrote.\n",
|
846 |
-
"* Saves time. Any time you answer a question multiple times through email, you should turn it into a blog post, which makes it easier for you to share the next time someone asks.\n",
|
847 |
-
"____\n",
|
848 |
-
"```\n",
|
849 |
-
"\n",
|
850 |
-
"Perhaps her most important tip is this: \n",
|
851 |
-
"\n",
|
852 |
-
"> : You are best positioned to help people one step behind you. The material is still fresh in your mind. Many experts have forgotten what it was like to be a beginner (or an intermediate) and have forgotten why the topic is hard to understand when you first hear it. The context of your particular background, your particular style, and your knowledge level will give a different twist to what you’re writing about.\n",
|
853 |
-
"\n",
|
854 |
-
"We've provided full details on how to set up a blog in <<appendix_blog>>. If you don't have a blog already, take a look at that now, because we've got a really great approach set up for you to start blogging for free, with no ads—and you can even use Jupyter Notebook!"
|
855 |
-
]
|
856 |
-
},
|
857 |
-
{
|
858 |
-
"cell_type": "markdown",
|
859 |
-
"metadata": {},
|
860 |
-
"source": [
|
861 |
-
"## Questionnaire"
|
862 |
-
]
|
863 |
-
},
|
864 |
-
{
|
865 |
-
"cell_type": "markdown",
|
866 |
-
"metadata": {},
|
867 |
-
"source": [
|
868 |
-
"1. Provide an example of where the bear classification model might work poorly in production, due to structural or style differences in the training data.\n",
|
869 |
-
"1. Where do text models currently have a major deficiency?\n",
|
870 |
-
"1. What are possible negative societal implications of text generation models?\n",
|
871 |
-
"1. In situations where a model might make mistakes, and those mistakes could be harmful, what is a good alternative to automating a process?\n",
|
872 |
-
"1. What kind of tabular data is deep learning particularly good at?\n",
|
873 |
-
"1. What's a key downside of directly using a deep learning model for recommendation systems?\n",
|
874 |
-
"1. What are the steps of the Drivetrain Approach?\n",
|
875 |
-
"1. How do the steps of the Drivetrain Approach map to a recommendation system?\n",
|
876 |
-
"1. Create an image recognition model using data you curate, and deploy it on the web.\n",
|
877 |
-
"1. What is `DataLoaders`?\n",
|
878 |
-
"1. What four things do we need to tell fastai to create `DataLoaders`?\n",
|
879 |
-
"1. What does the `splitter` parameter to `DataBlock` do?\n",
|
880 |
-
"1. How do we ensure a random split always gives the same validation set?\n",
|
881 |
-
"1. What letters are often used to signify the independent and dependent variables?\n",
|
882 |
-
"1. What's the difference between the crop, pad, and squish resize approaches? When might you choose one over the others?\n",
|
883 |
-
"1. What is data augmentation? Why is it needed?\n",
|
884 |
-
"1. What is the difference between `item_tfms` and `batch_tfms`?\n",
|
885 |
-
"1. What is a confusion matrix?\n",
|
886 |
-
"1. What does `export` save?\n",
|
887 |
-
"1. What is it called when we use a model for getting predictions, instead of training?\n",
|
888 |
-
"1. What are IPython widgets?\n",
|
889 |
-
"1. When might you want to use CPU for deployment? When might GPU be better?\n",
|
890 |
-
"1. What are the downsides of deploying your app to a server, instead of to a client (or edge) device such as a phone or PC?\n",
|
891 |
-
"1. What are three examples of problems that could occur when rolling out a bear warning system in practice?\n",
|
892 |
-
"1. What is \"out-of-domain data\"?\n",
|
893 |
-
"1. What is \"domain shift\"?\n",
|
894 |
-
"1. What are the three steps in the deployment process?"
|
895 |
-
]
|
896 |
-
},
|
897 |
-
{
|
898 |
-
"cell_type": "markdown",
|
899 |
-
"metadata": {},
|
900 |
-
"source": [
|
901 |
-
"### Further Research"
|
902 |
-
]
|
903 |
-
},
|
904 |
-
{
|
905 |
-
"cell_type": "markdown",
|
906 |
-
"metadata": {},
|
907 |
-
"source": [
|
908 |
-
"1. Consider how the Drivetrain Approach maps to a project or problem you're interested in.\n",
|
909 |
-
"1. When might it be best to avoid certain types of data augmentation?\n",
|
910 |
-
"1. For a project you're interested in applying deep learning to, consider the thought experiment \"What would happen if it went really, really well?\"\n",
|
911 |
-
"1. Start a blog, and write your first blog post. For instance, write about what you think deep learning might be useful for in a domain you're interested in."
|
912 |
-
]
|
913 |
-
},
|
914 |
-
{
|
915 |
-
"cell_type": "code",
|
916 |
-
"execution_count": null,
|
917 |
-
"metadata": {},
|
918 |
-
"outputs": [],
|
919 |
-
"source": []
|
920 |
-
}
|
921 |
-
],
|
922 |
-
"metadata": {
|
923 |
-
"jupytext": {
|
924 |
-
"split_at_heading": true
|
925 |
-
},
|
926 |
-
"kernelspec": {
|
927 |
-
"display_name": "Python 3 (ipykernel)",
|
928 |
-
"language": "python",
|
929 |
-
"name": "python3"
|
930 |
-
},
|
931 |
-
"language_info": {
|
932 |
-
"codemirror_mode": {
|
933 |
-
"name": "ipython",
|
934 |
-
"version": 3
|
935 |
-
},
|
936 |
-
"file_extension": ".py",
|
937 |
-
"mimetype": "text/x-python",
|
938 |
-
"name": "python",
|
939 |
-
"nbconvert_exporter": "python",
|
940 |
-
"pygments_lexer": "ipython3",
|
941 |
-
"version": "3.11.8"
|
942 |
-
}
|
943 |
-
},
|
944 |
-
"nbformat": 4,
|
945 |
-
"nbformat_minor": 4
|
946 |
-
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
model/model.pkl
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:63c8cb2aa96cba8f79777fa639d3f8c88e48579ef326f2ff91c839f205ade118
|
3 |
+
size 47053054
|