forked from mahmoodlab/CLAM
-
Notifications
You must be signed in to change notification settings - Fork 0
/
hyperparameter_optimization.py
152 lines (120 loc) · 5.62 KB
/
hyperparameter_optimization.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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
import argparse
import optuna
import subprocess
import os
from datetime import datetime, date
from tensorboard.backend.event_processing.event_accumulator import EventAccumulator
""" before running : check available GPU , set model_type, set task(in main.py default=),
adjust weights according to class per sample distribution
study can be run with multiple instances if study_name is set to existing name """
gpu = 0
model = 'clam_sb'
size = '256'
model_size = 'picoLayer'
# [ 'big','small','miniLayer','microLayer','nanoLayer', 'picoLayer']
def get_date_time():
curr_date = "".join(date.today().strftime("%d/%m").split("/")) + date.today().strftime("%Y")[2:]
curr_time = "".join(str(datetime.now()).split(" ")[1].split(".")[0].split(":"))
return curr_date, curr_time
def parse_metrics(log_dir, weights=(0.8, 0.2)):
best_weighted_mean = 0
best_trial = None
event_files = []
for dirpath, _, filenames in os.walk(log_dir):
for filename in filenames:
if filename.startswith('events.out'):
event_files.append(os.path.join(dirpath, filename))
for idx, event_file in enumerate(event_files):
event_accumulator = EventAccumulator(event_file)
event_accumulator.Reload()
# Get the scalar values for each class
class_0_acc = event_accumulator.Scalars('final/val_class_0_acc')
class_1_acc = event_accumulator.Scalars('final/val_class_1_acc')
if class_0_acc and class_1_acc:
# Assuming you want to use the latest values
latest_class_0_acc = max(scalar.value for scalar in class_0_acc)
latest_class_1_acc = max(scalar.value for scalar in class_1_acc)
# Compute the weighted mean
weighted_mean = (weights[0] * latest_class_0_acc +
weights[1] * latest_class_1_acc)
# Update the best weighted mean
if weighted_mean > best_weighted_mean:
best_weighted_mean = weighted_mean
best_trial = idx
return best_weighted_mean
def run_training_script(args):
"""Run the training script with the specified arguments."""
command = [
"python", os.path.join(os.path.dirname(__file__), "main.py"),
"--lr", str(args.lr),
"--reg", str(args.reg),
"--opt", args.opt,
"--drop_out", str(args.drop_out),
"--bag_loss", args.bag_loss,
"--model_type", args.model_type,
"--exp_code", args.exp_code,
"--model_size", str(args.model_size),
"--bag_weight", str(args.bag_weight),
"--B", str(args.B),
"--task", str(args.task),
"--log_data",
"--weighted_sample",
"--early_stopping",
"--results_dir", f"/mnt/EncryptedDisk2/BreastData/Studies/CLAM/results/{size}",
]
if args.inst_loss:
command.extend([
"--inst_loss", args.inst_loss,
])
if str(args.no_inst_cluster):
command.extend([
"--no_inst_cluster",
])
# Run the script
env = os.environ.copy()
env["CUDA_VISIBLE_DEVICES"] = f"{gpu}"
print("Running: {}".format(command))
result = subprocess.run(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, env=env)
if result.returncode != 0:
print(f"Script failed with error: {result.stderr}")
else:
print(f"Script executed successfully: {result.stdout}")
def objective(trial):
# Define the search space
lr = trial.suggest_float('lr', 1e-5, 1e-3, log=True)
reg = trial.suggest_float('reg', 1e-6, 1e-4, log=True)
opt = trial.suggest_categorical('opt', ['adam', 'sgd'])
drop_out = trial.suggest_float('drop_out', 0.0, 0.5,step=0.05)
bag_loss = trial.suggest_categorical('bag_loss', ['svm', 'ce'])
# model_size = trial.suggest_categorical('model_size', [ 'big','small','miniLayer','microLayer','nanoLayer', 'picoLayer']) # big, small, mini128, miniLayer, microLayer,'nanoLayer', 'picoLayer'
if size == '256':
task = 'biomarker_ER_256'
if size == '1024':
task = 'biomarker_ER_1024'
no_inst_cluster = trial.suggest_categorical('no_inst_cluster', [True, False])
inst_loss = trial.suggest_categorical('inst_loss', ['svm', 'ce', None])
bag_weight = trial.suggest_float('bag_weight', 0.5, 1.0,step=0.05)
B = trial.suggest_categorical('B', [4, 8, 16, 32, 64,128,256])
curr_date, curr_time = get_date_time()
exp_code = (model + curr_date + "_" + curr_time)
# Setup args namespace
args = argparse.Namespace(lr=lr, reg=reg, opt=opt, drop_out=drop_out, bag_loss=bag_loss,
model_type=model, model_size=model_size, no_inst_cluster=no_inst_cluster,
inst_loss=inst_loss, bag_weight=bag_weight, B=B, exp_code=exp_code, task=task)
# Define the log directory for TensorBoard logs
log_dir = f"/mnt/EncryptedDisk2/BreastData/Studies/CLAM/results/{size}/{exp_code}_s1"
run_training_script(args)
metric = parse_metrics(log_dir)
if metric is None:
return float('inf') # Return a high value if metric extraction fails
return metric
if __name__ == "__main__":
curr_date, _ = get_date_time()
study = optuna.create_study(direction="maximize", storage="sqlite:///example.db",study_name=(model+'_'+size+'_'+model_size+'_'+ curr_date), load_if_exists=True)
# study = optuna.create_study(direction="maximize", storage="sqlite:///example.db",study_name='', load_if_exists=True)
study.optimize(objective, n_trials=70)
# Print the best found parameters
print("Best trial:")
trial = study.best_trial
print(f"Value: {trial.value}")
print("Best hyperparameters: ", trial.params)