-
Notifications
You must be signed in to change notification settings - Fork 180
/
Copy pathfastai_simple.py
107 lines (83 loc) · 3.44 KB
/
fastai_simple.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
"""
Optuna example that optimizes convolutional neural network and data augmentation using fastai V2.
In this example, we optimize the hyperparameters of a convolutional neural network and
data augmentation for hand-written digit recognition in terms of validation accuracy.
The network is implemented by fastai and
evaluated on MNIST dataset. Throughout the training of neural networks, a pruner observes
intermediate results and stops unpromising trials.
Note that this example will take longer than the other examples
as this uses the entire MNIST dataset.
You can run this example as follows, pruning can be turned on and off with the `--pruning`
argument.
$ python fastaiv2_simple.py [--pruning]
"""
import argparse
import optuna
from optuna.integration import FastAIPruningCallback
from fastai.vision.all import accuracy
from fastai.vision.all import aug_transforms
from fastai.vision.all import ImageDataLoaders
from fastai.vision.all import Learner
from fastai.vision.all import SimpleCNN
from fastai.vision.all import untar_data
from fastai.vision.all import URLs
BATCHSIZE = 128
EPOCHS = 10
path = untar_data(URLs.MNIST_SAMPLE)
def objective(trial):
# Data Augmentation.
apply_tfms = trial.suggest_categorical("apply_tfms", [True, False])
if apply_tfms:
# MNIST is a hand-written digit dataset. Thus horizontal and vertical flipping are
# disabled. However, the two flipping will be important when the dataset is CIFAR or
# ImageNet.
tfms = aug_transforms(
do_flip=False,
flip_vert=False,
max_rotate=trial.suggest_int("max_rotate", 0, 45),
max_zoom=trial.suggest_float("max_zoom", 1, 2),
p_affine=trial.suggest_float("p_affine", 0.1, 1.0, step=0.1),
)
data = ImageDataLoaders.from_folder(
path, bs=BATCHSIZE, batch_tfms=tfms if apply_tfms else None
)
n_layers = trial.suggest_int("n_layers", 2, 5)
n_channels = [3]
for i in range(n_layers):
out_channels = trial.suggest_int("n_channels_{}".format(i), 3, 32)
n_channels.append(out_channels)
n_channels.append(2)
model = SimpleCNN(n_channels)
learn = Learner(
data,
model,
metrics=[accuracy],
# You could as FastAIPruningCallback in the fit function.
cbs=FastAIPruningCallback(trial, monitor="accuracy"),
)
# See https://forums.fast.ai/t/how-to-diable-progress-bar-completely/65249/3
# to disable progress bar and logging info.
with learn.no_bar():
with learn.no_logging():
learn.fit(EPOCHS)
return learn.recorder.metrics[0].value.item()
if __name__ == "__main__":
parser = argparse.ArgumentParser(description="fastai V2 example.")
parser.add_argument(
"--pruning",
"-p",
action="store_true",
help="Activate the pruning feature. `MedianPruner` stops unpromising "
"trials at the early stages of training.",
)
args = parser.parse_args()
pruner = optuna.pruners.MedianPruner() if args.pruning else optuna.pruners.NopPruner()
study = optuna.create_study(direction="maximize", pruner=pruner)
study.optimize(objective, n_trials=100, timeout=600)
print("Number of finished trials: {}".format(len(study.trials)))
print("Best trial:")
trial = study.best_trial
print(" Value: {}".format(trial.value))
print(" Params: ")
for key, value in trial.params.items():
print(" {}: {}".format(key, value))