-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathged_ppo_single_train.py
447 lines (370 loc) · 18.4 KB
/
ged_ppo_single_train.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
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
import argparse
import torch
from torch import nn
import os
import time
import yaml
import subprocess
import sys
import random
import numpy as np
from torch.multiprocessing import Pool, cpu_count
from copy import deepcopy
from src.ged_ppo_single_model import ActorNet, CriticNet, GraphEncoder
from utils.utils import print_args
from utils.tfboard_helper import TensorboardUtil
from ged_ppo_single_eval import evaluate
from utils.ged_env import GEDenv
class ItemsContainer:
def __init__(self):
self.__reward = []
self.__inp_graph_1 = []
self.__inp_graph_2 = []
self.__ori_k = []
self.__partial_x = []
self.__partial_cost = []
self.__done = []
def append(self, reward, inp_graph_1, inp_graph_2, ori_k, partial_cost, done, partial_x):
self.__reward.append(reward)
self.__inp_graph_1.append(inp_graph_1)
self.__inp_graph_2.append(inp_graph_2)
self.__ori_k.append(ori_k)
self.__partial_cost.append(partial_cost)
self.__done.append(done)
self.__partial_x.append(partial_x)
@property
def reward(self):
return deepcopy(self.__reward)
@property
def inp_graph_1(self):
return deepcopy(self.__inp_graph_1)
@property
def inp_graph_2(self):
return deepcopy(self.__inp_graph_2)
@property
def ori_k(self):
return deepcopy(self.__ori_k)
@property
def partial_cost(self):
return deepcopy(self.__partial_cost)
@property
def done(self):
return deepcopy(self.__done)
@property
def partial_x(self):
return deepcopy(self.__partial_x)
def update(self, idx, reward=None, inp_graph_1=None, inp_graph_2=None, partial_cost=None, done=None, partial_x=None):
if reward is not None:
self.__reward[idx] = reward
if inp_graph_1 is not None:
self.__inp_graph_1[idx] = inp_graph_1
if inp_graph_2 is not None:
self.__inp_graph_2[idx] = inp_graph_2
if partial_cost is not None:
self.__partial_cost[idx] = partial_cost
if done is not None:
self.__done[idx] = done
if partial_x is not None:
self.__partial_x[idx] = partial_x
class Memory:
def __init__(self):
self.actions = []
self.states = []
self.logprobs = []
self.rewards = []
self.is_terminals = []
def clear_memory(self):
del self.actions[:]
del self.states[:]
del self.logprobs[:]
del self.rewards[:]
del self.is_terminals[:]
class ActorCritic(nn.Module):
def __init__(self, node_feature_dim, node_output_size, batch_norm, one_hot_degree, gnn_layers):
super(ActorCritic, self).__init__()
self.state_encoder = GraphEncoder(node_feature_dim, node_output_size, batch_norm, one_hot_degree, gnn_layers)
self.actor_net = ActorNet(node_output_size, batch_norm)
self.value_net = CriticNet(node_output_size, batch_norm)
def forward(self):
raise NotImplementedError
def act(self, inp_graph_1, inp_graph_2, partial_x, memory):
node_feat_1, node_feat_2, graph_feat_1, graph_feat_2 = self.state_encoder(inp_graph_1, inp_graph_2, partial_x)
actions, action_logits, entropy = self.actor_net(node_feat_1, node_feat_2, partial_x)
memory.states.append((inp_graph_1, inp_graph_2, partial_x))
memory.actions.append(actions)
memory.logprobs.append(action_logits)
return actions
def evaluate(self, inp_graph_1, inp_graph_2, partial_x, action):
node_feat_1, node_feat_2, graph_feat_1, graph_feat_2 = self.state_encoder(inp_graph_1, inp_graph_2, partial_x)
_, action_logits, entropy = self.actor_net(node_feat_1, node_feat_2, partial_x, action)
state_value = self.value_net(graph_feat_1, graph_feat_2)
return action_logits, state_value, entropy
class PPO:
def __init__(self, args, device):
self.lr = args.learning_rate
self.betas = args.betas
self.gamma = args.gamma
self.eps_clip = args.eps_clip
self.K_epochs = args.k_epochs
self.device = device
ac_params = args.node_feature_dim, args.node_output_size, args.batch_norm, args.one_hot_degree, args.gnn_layers
self.policy = ActorCritic(*ac_params).to(self.device)
self.optimizer = torch.optim.Adam(
[{'params': self.policy.actor_net.parameters()},
{'params': self.policy.value_net.parameters()},
{'params': self.policy.state_encoder.parameters(), 'lr': self.lr / 10}],
lr=self.lr, betas=self.betas)
if len(args.lr_steps) > 0:
# rescale lr_step value to match the action steps
lr_steps = [step // args.update_timestep for step in args.lr_steps]
self.lr_scheduler = torch.optim.lr_scheduler.MultiStepLR(self.optimizer, lr_steps, gamma=0.1)
else:
self.lr_scheduler = None
self.policy_old = ActorCritic(*ac_params).to(self.device)
self.policy_old.load_state_dict(self.policy.state_dict())
self.MseLoss = nn.MSELoss()
def update(self, memory):
# Time Difference estimate of state rewards:
rewards = []
with torch.no_grad():
logprobs, state_values, dist_entropy = \
self.policy.evaluate(*memory.states[-1], memory.actions[-1].to(self.device))
discounted_reward = state_values
for reward, is_terminal in zip(reversed(memory.rewards), reversed(memory.is_terminals)):
reward = torch.tensor(reward, dtype=torch.float32).to(self.device)
discounted_reward = discounted_reward * (1 - torch.tensor(is_terminal, dtype=torch.float32).to(self.device))
discounted_reward = reward + (self.gamma * discounted_reward).clone()
rewards.insert(0, discounted_reward)
# Normalizing the rewards:
rewards = torch.cat(rewards)
rewards = (rewards - rewards.mean()) / (rewards.std() + 1e-5)
# convert list to tensor
old_graph_1 = []
old_graph_2 = []
old_partial_x = []
for state in memory.states:
old_graph_1 += state[0]
old_graph_2 += state[1]
old_partial_x += state[2]
old_actions = torch.cat(memory.actions, dim=1)
old_logprobs = torch.cat(memory.logprobs, dim=1)
critic_loss_sum = 0
# Optimize policy for K epochs:
for _ in range(self.K_epochs):
# Evaluating old actions and values :
logprobs, state_values, dist_entropy = self.policy.evaluate(old_graph_1, old_graph_2, old_partial_x, old_actions)
# Finding the ratio (pi_theta / pi_theta__old):
ratios = torch.exp(logprobs - old_logprobs)
# Normalizing advantages
advantages = rewards - state_values.detach()
#advantages = (advantages - advantages.mean()) / (advantages.std() + 1e-5)
# Finding Surrogate Loss:
surr1 = ratios * advantages
surr2 = torch.clamp(ratios, 1 - self.eps_clip, 1 + self.eps_clip) * advantages
actor_loss = -torch.min(surr1, surr2)
critic_loss = self.MseLoss(state_values, rewards)
entropy_reg = -0.01 * dist_entropy
critic_loss_sum += critic_loss.detach().mean()
# take gradient step
self.optimizer.zero_grad()
(actor_loss + critic_loss + entropy_reg).mean().backward()
self.optimizer.step()
if self.lr_scheduler:
self.lr_scheduler.step()
# Copy new weights into old policy:
self.policy_old.load_state_dict(self.policy.state_dict())
return critic_loss_sum / self.K_epochs # mean critic loss
def main(args):
# initialize manual seed
if args.random_seed is not None:
random.seed(args.random_seed)
np.random.seed(args.random_seed)
torch.manual_seed(args.random_seed)
# create environment
ged_env = GEDenv(args.solver_type, args.dataset)
args.node_feature_dim = ged_env.feature_dim
# get current device (cuda or cpu)
device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
# load training/testing data
tuples_train = ged_env.generate_tuples(ged_env.training_graphs, args.train_sample, 0, device)
tuples_test = ged_env.generate_tuples(ged_env.val_graphs, args.test_sample, 1, device)
# create tensorboard summary writer
try:
import tensorflow as tf
# local mode: logs stored in ./runs/TIME_STAMP-MACHINE_ID
tfboard_path = 'runs'
import socket
from datetime import datetime
current_time = datetime.now().strftime('%b%d_%H-%M-%S')
tfboard_path = os.path.join(tfboard_path, current_time + '_' + socket.gethostname())
summary_writer = TensorboardUtil(tf.summary.FileWriter(tfboard_path))
except (ModuleNotFoundError, ImportError):
print('Warning: Tensorboard not loading, please install tensorflow to enable...')
summary_writer = None
# init models
memory = Memory()
ppo = PPO(args, device)
num_workers = cpu_count()
mp_pool = Pool(num_workers)
# logging variables
best_test_ratio = 0
running_reward = 0
critic_loss = []
avg_length = 0
timestep = 0
prev_time = time.time()
# training loop
for i_episode in range(1, args.max_episodes + 1):
items_batch = ItemsContainer()
for b in range(args.batch_size):
graph_index = ((i_episode - 1) * args.batch_size + b) % len(tuples_train)
inp_graph_1, inp_graph_2, ori_k, _, baselines, _ = tuples_train[graph_index]
partial_x = torch.zeros(inp_graph_1.num_nodes+1, inp_graph_2.num_nodes+1, device=device)
partial_cost = 0
items_batch.append(0, inp_graph_1, inp_graph_2, ori_k, partial_cost, False, partial_x)
for t in range(args.max_timesteps):
timestep += 1
# Running policy_old:
with torch.no_grad():
action_batch = ppo.policy_old.act(
items_batch.inp_graph_1, items_batch.inp_graph_2, items_batch.partial_x, memory)
def step_func_feeder(batch_size):
partial_x = items_batch.partial_x
batch_ori_k = items_batch.ori_k
batch_partial_cost = items_batch.partial_cost
for b in range(batch_size):
yield partial_x[b], batch_ori_k[b], action_batch[:, b], batch_partial_cost[b]
if args.batch_size > 1:
pool_map = mp_pool.starmap_async(ged_env.step_e2e, step_func_feeder(args.batch_size))
step_list = pool_map.get()
else:
step_list = [ged_env.step_e2e(*x) for x in step_func_feeder(args.batch_size)]
for b, item in enumerate(step_list):
reward, partial_x, partial_cost, done = item # only graph1 is updated
items_batch.update(b, reward=reward, partial_x=partial_x, partial_cost=partial_cost, done=done)
# Saving reward and is_terminal:
memory.rewards.append(items_batch.reward)
memory.is_terminals.append(items_batch.done)
# update if its time
if timestep % args.update_timestep == 0:
closs = ppo.update(memory)
critic_loss.append(closs)
if summary_writer:
summary_writer.add_scalar('critic mse/train', closs, timestep)
memory.clear_memory()
running_reward += sum(items_batch.reward) / args.batch_size
if any(items_batch.done):
break
avg_length += t+1
# logging
if i_episode % args.log_interval == 0:
avg_length = avg_length / args.log_interval
running_reward = running_reward / args.log_interval
if len(critic_loss) > 0:
critic_loss = torch.mean(torch.stack(critic_loss))
else:
critic_loss = -1
now_time = time.time()
avg_time = (now_time - prev_time) / args.log_interval
prev_time = now_time
if summary_writer:
summary_writer.add_scalar('reward/train', running_reward, timestep)
summary_writer.add_scalar('time/train', avg_time, timestep)
for lr_id, x in enumerate(ppo.optimizer.param_groups):
summary_writer.add_scalar(f'lr/{lr_id}', x['lr'], timestep)
print(
f'Episode {i_episode} \t '
f'avg length: {avg_length:.2f} \t '
f'critic mse: {critic_loss:.4f} \t '
f'reward: {running_reward.item():.4f} \t '
f'time per episode: {avg_time:.2f}'
)
running_reward = 0
avg_length = 0
critic_loss = []
# testing
if i_episode % args.test_interval == 0:
with torch.no_grad():
# record time spent on test
prev_test_time = time.time()
#print("########## Evaluate on Train ##########")
#train_dict = evaluate(ppo.policy, dag_graph, tuples_train, args.max_timesteps, args.search_size, mp_pool)
#for key, val in train_dict.items():
# if isinstance(val, dict):
# if summary_writer:
# summary_writer.add_scalars(f'{key}/train-eval', val, timestep)
# else:
# if summary_writer:
# summary_writer.add_scalar(f'{key}/train-eval', val, timestep)
print("########## Evaluate on Test ##########")
# run testing
test_dict = evaluate(ppo.policy, ged_env, tuples_test, args.max_timesteps, args.search_size,
None if torch.cuda.is_available() else mp_pool)
# write to summary writter
for key, val in test_dict.items():
if isinstance(val, dict):
if summary_writer:
summary_writer.add_scalars(f'{key}/test', val, timestep)
else:
if summary_writer:
summary_writer.add_scalar(f'{key}/test', val, timestep)
print("########## Evaluate complete ##########")
# fix running time value
prev_time += time.time() - prev_test_time
if test_dict["ratio"]["mean"] > best_test_ratio:
best_test_ratio = test_dict["ratio"]["mean"]
file_name = f'./E2E_PPO_{args.solver_type}_dataset{args.dataset}' \
f'_beam{args.search_size}_ratio{best_test_ratio:.4f}.pt'
torch.save(ppo.policy.state_dict(), file_name)
def parse_arguments():
parser = argparse.ArgumentParser(description='GED solver. You have two ways of setting the parameters: \n'
'1) set parameters by command line arguments \n'
'2) specify --config path/to/config.yaml')
# environment configs
parser.add_argument('--solver_type', default='hungarian')
parser.add_argument('--resource_limit', default=600, type=float)
parser.add_argument('--add_graph_features', action='store_true')
parser.add_argument('--dataset', default='AIDS700nef')
parser.add_argument('--gamma', default=0.99, type=float, help='discount factor for accumulated reward')
parser.add_argument('--train_sample', default=50, type=int, help='number of training samples')
parser.add_argument('--test_sample', default=10, type=int, help='number of testing samples')
# decode(testing) configs
# parser.add_argument('--beam_search', action='store_true')
parser.add_argument('--search_size', default=5, type=int)
# learning configs
parser.add_argument('--learning_rate', default=0.002, type=float)
parser.add_argument('--lr_steps', default=[], type=list)
parser.add_argument('--batch_size', default=1, type=int, help='Batch size when sampling')
parser.add_argument('--betas', default=(0.9, 0.999), help='Adam optimizer\'s beta')
parser.add_argument('--max_episodes', default=50000, type=int, help='max training episodes')
parser.add_argument('--max_timesteps', default=300, type=int, help='max timesteps in one episode')
parser.add_argument('--update_timestep', default=20, type=int, help='update policy every n timesteps')
parser.add_argument('--k_epochs', default=4, type=int, help='update policy for K epochs')
parser.add_argument('--eps_clip', default=0.2, type=float, help='clip parameter for PPO')
# model parameters
parser.add_argument('--one_hot_degree', default=0, type=int)
parser.add_argument('--batch_norm', action='store_true')
parser.add_argument('--node_output_size', default=16, type=int)
parser.add_argument('--gnn_layers', default=10, type=int, help='number of GNN layers')
# misc configs
parser.add_argument('--config', default=None, type=str, help='path to config file,'
' and command line arguments will be overwritten by the config file')
parser.add_argument('--random_seed', default=None, type=int)
parser.add_argument('--test_interval', default=500, type=int, help='run testing in the interval (episodes)')
parser.add_argument('--log_interval', default=100, type=int, help='print avg reward in the interval (episodes)')
parser.add_argument('--test_model_weight', default='', type=str, help='the path of model weight to be loaded')
args = parser.parse_args()
if args.config:
with open('config/' + args.config) as f:
cfg_dict = yaml.load(f)
for key, val in cfg_dict.items():
assert hasattr(args, key), f'Unknown config key: {key}'
setattr(args, key, val)
f.seek(0)
print(f'Config file: {args.config}', )
for line in f.readlines():
print(line.rstrip())
print_args(args)
return args
if __name__ == '__main__':
main(parse_arguments())