-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathshowcase_experiments.py
208 lines (182 loc) · 11.5 KB
/
showcase_experiments.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
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
import argparse
import scipy as sp
import os
import pandas as pd
from experiment import Experiment
from utils import *
LOGISTIC_HYPERPARAMS = {
'sgd': {'start': 4e-3, 'end': 4e2, 'num': 10},
'svrg': {'start': 4e-3, 'end': 4e2, 'num': 10, 'update_freq': {'snapshot': (1, 'epochs')}},
'saga': {'start': 4e-3, 'end': 4e2, 'num': 10},
'lkatyusha': {'start': 2.5e-3, 'end': 2.5e-1, 'num': 10},
'slbfgs': {'start': 4e-5, 'end': 4e0, 'num': 10, 'update_freq': {'precond': (1, 'epochs'), 'snapshot': (1, 'epochs')}, 'Mem': 10},
'sketchysgd': {'update_freq': {'precond': (1, 'epochs')}, 'rank': 10, 'rho': 1e-3},
'sketchysvrg': {'update_freq': {'precond': (1, 'epochs'), 'snapshot': (1, 'epochs')}, 'rank': 10, 'rho': 1e-3},
'sketchysaga': {'update_freq': {'precond': (1, 'epochs')}, 'rank': 10, 'rho': 1e-3},
'sketchykatyusha': {'update_freq': {'precond': (1, 'epochs')}, 'rank': 10, 'rho': 1e-3}
}
LS_HYPERPARAMS = {
'sgd': {'start': 1e-3, 'end': 1e2, 'num': 10},
'svrg': {'start': 1e-3, 'end': 1e2, 'num': 10, 'update_freq': {'snapshot': (1, 'epochs')}},
'saga': {'start': 1e-3, 'end': 1e2, 'num': 10},
'lkatyusha': {'start': 1e-2, 'end': 1e0, 'num': 10},
'slbfgs': {'start': 1e-5, 'end': 1e0, 'num': 10, 'update_freq': {'precond': (1, 'epochs'), 'snapshot': (1, 'epochs')}, 'Mem': 10},
'sketchysgd': {'rank': 10, 'rho': 1e-3},
'sketchysvrg': {'update_freq': {'snapshot': (1, 'epochs')}, 'rank': 10, 'rho': 1e-3},
'sketchysaga': {'rank': 10, 'rho': 1e-3},
'sketchykatyusha': {'rank': 10, 'rho': 1e-3}
}
BATCH_SIZE = 4096 # Minibatch size for stochastic gradients
def get_grid_search_list(start, end, num):
return [start * (end / start) ** (i / (num - 1)) for i in range(num)]
def get_experiment_data(dataset, data_source, problem_type, rescale, rf_params_set):
rf_params = None
if dataset in list(rf_params_set.keys()):
rf_params = rf_params_set[dataset]
if data_source == 'libsvm':
data = load_preprocessed_data(dataset, problem_type, rescale, rf_params)
elif data_source == 'openml':
data = load_preprocessed_data_openml(dataset, rescale, rf_params)
elif data_source == 'yelp':
data = {}
data['Atr'] = sp.sparse.load_npz(os.path.join(DATA_DIR, YELP_DATA_FILES['train']))
data['Atst'] = sp.sparse.load_npz(os.path.join(DATA_DIR, YELP_DATA_FILES['test']))
data['btr'] = np.load(os.path.join(DATA_DIR, YELP_DATA_FILES['train_labels']))
data['btst'] = np.load(os.path.join(DATA_DIR, YELP_DATA_FILES['test_labels']))
return data
# Get all the experiments for a given dataset, model, and optimizer
def get_experiments(data, model_type, model_params, opt, precond_type, hyperparams, auto_lr, max_epochs, time_budget, bg, bh):
if opt in ['sgd', 'svrg', 'saga', 'lkatyusha', 'slbfgs']:
# Get the grid search list for the learning rate
hp_list = get_grid_search_list(hyperparams['start'], hyperparams['end'], hyperparams['num'])
# Get the update frequency
update_freq = hyperparams['update_freq'] if 'update_freq' in list(hyperparams.keys()) else None
if opt == 'sgd':
params_list = [{'eta': lr} for lr in hp_list]
elif opt == 'saga':
params_list = [{}] if auto_lr else [{'eta': lr} for lr in hp_list]
elif opt == 'svrg':
params_list = [{'update_freq': update_freq.copy()}] if auto_lr else [{'eta': lr, 'update_freq': update_freq.copy()} for lr in hp_list]
elif opt == 'lkatyusha':
params_list = [{'mu': model_params['mu'], 'bg': bg}] if auto_lr else [{'mu': model_params['mu'], 'L': L, 'bg': bg} for L in hp_list]
elif opt == 'slbfgs':
params_list = [{'eta': lr, 'update_freq': update_freq.copy(), 'Mem': hyperparams['Mem'], 'bh': bh} for lr in hp_list]
# Get the experiments
experiments = [Experiment(data, model_type, model_params, opt, opt_params, time_budget = time_budget) for opt_params in params_list]
elif opt in ['sketchysgd', 'sketchysvrg', 'sketchysaga', 'sketchykatyusha']:
# Get the update frequency
if model_type == 'logistic':
update_freq = hyperparams['update_freq']
elif model_type == 'least_squares' and opt != 'sketchysvrg':
update_freq = {'precond': (max_epochs * 2, 'epochs')} # Ensure the preconditioner is held fixed for the entire run
elif model_type == 'least_squares' and opt == 'sketchysvrg':
update_freq = hyperparams['update_freq']
update_freq['precond'] = (max_epochs * 2, 'epochs')
if opt in ['sketchysgd', 'sketchysvrg', 'sketchysaga']:
opt_params = {'precond_type': precond_type, 'update_freq': update_freq.copy(), 'rank': hyperparams['rank'], 'rho': hyperparams['rho'], 'bh': bh}
elif opt in ['sketchykatyusha']:
opt_params = {'precond_type': precond_type, 'update_freq': update_freq.copy(), 'rank': hyperparams['rank'], 'rho': hyperparams['rho'], 'bh': bh, 'bg': bg, 'mu': model_params['mu']}
experiments = [Experiment(data, model_type, model_params, opt, opt_params, time_budget = time_budget)]
return experiments
# Writes results to a csv file
def write_as_dataframe(result, directory, opt_name, opt_params, r_seed, np_seed):
df = pd.DataFrame.from_dict(result)
if opt_name in ['sgd', 'saga', 'svrg', 'slbfgs']:
if 'eta' in list(opt_params.keys()): # Account for experiments where SAGA/SVRG/L-Katyusha use the default learning rate
csv_name = 'lr_'+str(opt_params['eta'])
else:
csv_name = 'auto'
elif opt_name == 'lkatyusha':
if 'L' in list(opt_params.keys()):
csv_name = 'L_'+str(opt_params['L'])
else:
csv_name = 'auto'
else:
csv_name = 'auto'
csv_name += '_seed_'+str(r_seed)+'_'+str(np_seed)+'.csv'
if not os.path.exists(directory):
os.makedirs(directory)
file_name = os.path.join(directory, csv_name)
df.to_csv(file_name)
def main():
# Get arguments from command line
parser = argparse.ArgumentParser(description = "Run experiments for showcase plots. We solve problems to the time budget specified by the user. \
Hyperparameters are selected according to LOGISTIC_HYPERPARAMS/LS_HYPERPARAMS at the top of the file.\n \
Datasets are automatically normalized/standardized (if applicable) and random features are applied if applicable (with a small exception for 'acsincome'). \
Random seeds are set according to SEEDS in constants.py.")
parser.add_argument('--data', type = str, required = True, help = "Name of a dataset, i.e., a key in LOGISTIC_DATA_FILES/LS_DATA_FILES/LS_DATA_FILES_OPENML in constants.py, or 'yelp'.") # Dataset
parser.add_argument('--problem', type = str, required = True, help = "Type of problem: either 'logistic' or 'least_squares'") # Problem type
parser.add_argument('--opt', type = str, required = True, help = "Optimization method: one of 'sgd', 'svrg', 'saga', 'lkatyusha', 'slbfgs', 'sketchysgd', 'sketchysvrg', 'sketchysaga', or 'sketchykatyusha'") # Optimizer
parser.add_argument('--precond', type = str, default = None, help = "Preconditioner type: one of 'diagonal', 'nystrom', 'sassn', 'lessn', or 'ssn' (default is None)")
parser.add_argument('--time-budget', type = float, required = True, help = 'Time budget in seconds') # Time budget
parser.add_argument('--epochs', type = int, required = True, help = 'Number of epochs to run the optimizer. This can be used if we want to stop earlier. If we want to run to the full time, this quantity can be set to a large integer.') # Number of epochs to run
parser.add_argument('--mu', type = float, required = False, default = 1e-2, help = 'Unscaled regularization parameter (default is 1e-2)') # Regularization parameter
parser.add_argument('--auto_lr', default=False, action='store_true', help = 'Whether to use the default learning rate for SAGA (default is False)') # Whether to use the default learning rate for SAGA/SVRG/L-Katyusha
parser.add_argument('--dest', type = str, required = True, help = 'Directory to save results') # Directory to save results
# Extract arguments
args = parser.parse_args()
dataset = args.data
problem_type = args.problem
opt = args.opt
precond_type = args.precond
time_budget = args.time_budget
epochs = args.epochs
mu_unscaled = args.mu
auto_lr = args.auto_lr
results_dest = os.path.abspath(args.dest)
# Check that auto_lr is only used with SAGA/SVRG/L-Katyusha
if auto_lr and opt not in ['saga', 'svrg', 'lkatyusha']:
raise ValueError("The automatic learning rate argument can only be used with SAGA/SVRG/L-Katyusha")
# If we are using a "sketchy" optimizer, make sure a preconditioner is specified
if opt.startswith('sketchy') and precond_type is None:
raise ValueError("Must specify a preconditioner for sketchy optimizers")
if not opt.startswith('sketchy'):
directory = os.path.join(results_dest, dataset, opt) # Location where results will be saved
else:
directory = os.path.join(results_dest, dataset, opt, precond_type) # Location where results will be saved for "sketchy" optimizers
# Print key parameters
print(f"Dataset: {dataset}")
print(f"Problem type: {problem_type}")
print(f"Optimization method: {opt}")
print(f"Preconditioner type: {precond_type}")
print(f"Time budget (in seconds): {time_budget}")
print(f"Number of epochs: {epochs}")
print(f"Automatic learning rate: {auto_lr}") if opt in ['saga', 'svrg', 'lkatyusha'] else None
print(f"Unscaled regularization parameter: {mu_unscaled}")
print(f"Results directory: {directory}\n")
# Ensure normalization/standardization occurs + set seeds for reproducibility
rescale = True
set_random_seeds(**SEEDS)
if problem_type == 'logistic':
rf_params_set = LOGISTIC_RAND_FEAT_PARAMS
hyperparam_set = LOGISTIC_HYPERPARAMS
elif problem_type == 'least_squares':
rf_params_set = LS_RAND_FEAT_PARAMS
hyperparam_set = LS_HYPERPARAMS
# Get data (special exception for 'acsincome' dataset)
# Furthermore, apply random features if applicable
if dataset != 'acsincome':
if dataset in (list(LOGISTIC_DATA_FILES.keys()) or dataset in list(LS_DATA_FILES.keys())):
data_source = 'libsvm'
elif dataset in list(LS_DATA_FILES_OPENML.keys()):
data_source = 'openml'
elif dataset == 'yelp':
data_source = 'yelp'
data = get_experiment_data(dataset, data_source, problem_type, rescale, rf_params_set)
else: # Do custom preprocessing for 'acsincome' dataset
data = load_preprocessed_data_openml('acsincome', standardize = True,
rf_params = None, filter_outliers = True)
data['Atr'], data['Atst'] = rand_features(7500, data['Atr'].shape[1],
1, data['Atr'], data['Atst'])
# Get experiments
hyperparams = hyperparam_set[opt] # Get hyperparameters
ntr = data['Atr'].shape[0]
model_params = {'mu': mu_unscaled / ntr}
bh = int(ntr ** (0.5)) # Hessian batch size
experiments = get_experiments(data, problem_type, model_params, opt, precond_type, hyperparams, auto_lr, epochs, time_budget, BATCH_SIZE, bh)
# Run experiments and write results to .csv files
for experiment in experiments:
result = experiment.run(epochs, BATCH_SIZE)
write_as_dataframe(result, directory, opt, experiment.opt_params, **SEEDS)
if __name__ == '__main__':
main()