-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathsys_control.py
480 lines (360 loc) · 15.8 KB
/
sys_control.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
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# File : sys_control.py
# Author : tzhang
# Date : 24.11.2019
# Last Modified Date: 12.10.2020
# Last Modified By : tzhang
import numpy as np
"""
a module account for system overall configuration
num_chars:
00 nuclear power plant
01 wind turbine
02 solar panels (not yet implemented)
10 PEM stacks
20 hydrogen stoage (very simple mode currently)
"""
class sys_config:
def __init__(self,components,num_chars,unit_power,n_units,lifetimes,con_time,sys_lifetime):
config = {}
# dictionary for the system configuration
for i in range(len(components)):
config[components[i]] = []
config[components[i]].append(num_chars[i]) # characteristic number of each component
config[components[i]].append(unit_power[i]) # nominal power of unit of each component, in MW
config[components[i]].append(n_units[i]) # number of units of each component
config[components[i]].append(lifetimes[i]) # lifetime of each component
config[components[i]].append(con_time[i]) # lifetime of each component
# different lifetime of components causese re-construction of the components during the system life time. As an example, the lifetime of the system is 60 years, and the lifetime of wind turbines are 30 years, 60/30 = 2 batches are needed in the system; if the system liffetime is 50 year, wind turbines are 30 years, 50/30 = 2 batches are still needed in this case. However, this should be avoid in making the energy mix
n_int = int(sys_lifetime/lifetimes[i])
n_float = sys_lifetime/lifetimes[i]
div = n_float - n_int
if div > 0 or div < 0:
n_batch = n_int + 1
else:
n_batch = n_int
config[components[i]].append(n_batch) # number of batches in the life cycle
# system config data class
self.config = config
# lifetime of the system
self.lifetime = sys_lifetime
"""
a test module
components = ['SMR','wind','PEM','storage']
num_chars = ['00','01','10','20']
unit_power = [60.0,2.0,0.6,0.0]
n_units = [6,80,500,1]
lifetimes = [60,30,20,60]
con_time = [2,1,1,1]
sys_lifetime = 60
system = sys_config(components,num_chars,unit_power,n_units,lifetimes,con_time,sys_lifetime)
"""
"""
a module account for operating units of each components in lifetime according to construction plan
auto_con:
1 automatic construction plan with pre-defined method
0 manually read construction plan from file
"""
class con_plan(sys_config):
def __init__(self,components,num_chars,unit_power,n_units,lifetimes,con_time,sys_lifetime):
super().__init__(components,num_chars,unit_power,n_units,lifetimes,con_time,sys_lifetime)
self.sys_scale = [] # array for system scaling data
self.lifetime_scale = [] # array of construciton and operation schedule
def _con_plan_auto_scale_(self):
npp_in = False # default no npp in the system
sys_scale = []
char = []
# check whether there is nuclear power plant in the mix
for key in self.config.keys():
if '00' in self.config[key][0]:
npp_in = True # there is npp
npp_key = key
npp_unit = self.config[key][2]
char.append(self.config[key][0])
# construction plan when there is npp
if npp_in:
unit_renew = []
unit_couple = []
match_renew = []
match_couple = []
# matching scale of units, for example, 1 npp to 20 wind and 40 PEM
for key in self.config.keys():
if self.config[key][0].startswith('0') and self.config[key][0] != '00':
unit_renew.append(self.config[key][2])
match = int(self.config[key][2]/npp_unit)
match_renew.append(match)
char.append(self.config[key][0])
elif self.config[key][0].startswith('1'):
unit_couple.append(self.config[key][2])
match = int(self.config[key][2]/npp_unit)
match_couple.append(match)
char.append(self.config[key][0])
sys_scale.append(char)
for i in range(self.config[npp_key][2]+1):
scale = []
comp_renew = []
comp_couple = []
if i != self.config[npp_key][2]:
for j in range(len(unit_renew)):
unit_curr = i * match_renew[j]
comp_renew.append(unit_curr)
for j in range(len(unit_couple)):
unit_curr = i * match_couple[j]
comp_couple.append(unit_curr)
else:
for j in range(len(unit_renew)):
unit_last = unit_renew[j]
comp_renew.append(unit_last)
for j in range(len(unit_couple)):
unit_last = unit_couple[j]
comp_couple.append(unit_last)
scale.append(i)
for n_uni in comp_renew:
scale.append(n_uni)
for n_uni in comp_couple:
scale.append(n_uni)
sys_scale.append(scale)
# construction plan when there is no npp
else:
unit_renew = []
unit_couple = []
# matching scale of units, for example, 1 npp to 20 wind and 40 PEM
for key in self.config.keys():
if self.config[key][0].startswith('0') and self.config[key][0] != '00':
unit_renew.append(self.config[key][2])
char.append(self.config[key][0])
elif self.config[key][0].startswith('1'):
unit_couple.append(self.config[key][2])
char.append(self.config[key][0])
sys_scale.append(char)
scale = []
for i in range(len(unit_renew)+len(unit_couple)):
scale.append(0.0)
sys_scale.append(scale)
scale = []
for n_uni in unit_renew:
scale.append(n_uni)
for n_uni in unit_couple:
scale.append(n_uni)
sys_scale.append(scale)
return sys_scale
# calculate installed capacity and capacity ratio, only npp and renewables are accounted, pem or other compnents with char_num started other than 0 are not accounted
def _cal_capacity_(self,sys_scale):
unit_power = []
n_units = []
for key in self.config.keys():
if self.config[key][0].startswith('0'):
unit_power.append(self.config[key][1])
n_units.append(self.config[key][2])
# calculate nominal capacity
nomi_power = 0.0
for i in range(len(n_units)):
power = n_units[i] * unit_power[i]
nomi_power = nomi_power + power
for i in range(1,len(sys_scale)):
capacity = 0
for j in range(len(unit_power)):
capacity_comp = sys_scale[i][j] * unit_power[j]
capacity = capacity + capacity_comp
r_capa = capacity/nomi_power
sys_scale[i].append(capacity)
sys_scale[i].append(r_capa)
return sys_scale
# calculate scale data
def cal_scale(self,auto_con):
if auto_con == 1:
sys_scale = self._con_plan_auto_scale_()
else:
#########################
#### to be developed ####
#########################
print ('module under development')
pass
# calculate installed capacity and capacity ratio
sys_scale = self._cal_capacity_(sys_scale)
self.sys_scale = sys_scale
# calculste construction schedule:
def con_schedule(self,auto_con):
if auto_con == 1:
self._con_schedule_auto_()
else:
#########################
#### to be developed ####
#########################
print ('module under development')
pass
# automatic form construction schedule
def _con_schedule_auto_(self):
con_time = []
for key in self.config.keys():
if self.config[key][0].startswith('0') or self.config[key][0].startswith('1'):
data = []
data.append(self.config[key][0])
data.append(self.config[key][2])
data.append(self.config[key][4])
con_time.append(data)
con_time_tmp = np.asarray(con_time,dtype=int)
# char of core component
char_comp = con_time[np.argmax(con_time_tmp[:,2])][0]
# number of units of npp component or if no nuclear, a batch do the work
unit_comp = con_time[np.argmax(con_time_tmp[:,2])][1]
# years needed for construction
y_unit_construct = np.max(con_time_tmp[:,2])
# calculate duration for construction
#y_construction = unit_comp * y_unit_construct
y_construction = (len(self.sys_scale)-1) * y_unit_construct
# find the column of core comp in sys_scale array
col = self.sys_scale[0].index(char_comp)
col_other_comp = []
for data in self.sys_scale[0]:
if data != char_comp:
idx = self.sys_scale[0].index(data)
col_other_comp.append(idx)
lifetime_scale = []
name_idx = []
# add index to column
name_idx.append('year')
for comp in self.sys_scale[0]:
name_idx.append(comp)
name_idx.append('capacity')
name_idx.append('capacity ratio')
lifetime_scale.append(name_idx)
for i in range (self.lifetime + int(y_construction/2)):
# for i in range (y_construction+2):
data = []
if char_comp == '00':
unit_done = min(max(0,int((i-1)/y_unit_construct)),unit_comp)
else:
unit_done = unit_comp
data.append(i)
data.append(unit_done)
# scale matching
idx = [row[col] for row in self.sys_scale].index(unit_done)
for j in col_other_comp:
data.append(self.sys_scale[idx][j])
# add insalled capacity and capacity ratio
data.append(self.sys_scale[idx][-2])
data.append(self.sys_scale[idx][-1])
lifetime_scale.append(data)
self.lifetime_scale = lifetime_scale
"""
a test module
auto_con = 1
sys_con = con_plan(system,con_time)
sys_con.cal_scale(auto_con)
sys_con.con_schedule(auto_con)
print (sys_con.lifetime_scale)
"""
"""
a control module to balance energy between coupled system and hydrogen system
"""
class balancing:
def __init__(self):
self.dP = []
# calculate the power from coupled nu-re system to h2 system (can be positive or negative)
def cal_to_h2sys_virtual(self,P_coupled,P_demand,Pmin_cluster):
P_to_h2sys_array = []
balancing._dP_cal_(self,P_coupled,P_demand)
for i in range(len(self.dP)):
if self.dP[i] >= Pmin_cluster:
P_to_h2sys = balancing._full_electrolysis_(self,i)
elif self.dP[i] >= 0 and self.dP[i] < Pmin_cluster:
P_to_h2sys = balancing._partial_electrolysis_(self,i)
else:
P_to_h2sys = balancing._full_fuelcell_(self,i)
P_to_h2sys_array.append(P_to_h2sys)
return P_to_h2sys_array
# calculated the delta between coupled nuclear-wind system and the grid demand
def _dP_cal_(self,P_coupled,P_demand):
dP = P_coupled - P_demand
dP = list(dP)
# print (dP)
self.dP = dP
# calculate the power to hydrogen system if residual power is larger than the minimum opearion demand
def _full_electrolysis_(self,i):
P_nure_to_h2sys = self.dP[i]
return P_nure_to_h2sys
# calulate the power to the hydrogen system when the residual power can only run some electrolysis
def _partial_electrolysis_(self,i):
P_nure_to_h2sys = self.dP[i]
return P_nure_to_h2sys
# calulate the power demand from the hydrogen system when coupled system at low production rate
def _full_fuelcell_(self,i):
P_nure_to_h2sys = self.dP[i]
return P_nure_to_h2sys
# calculate the power to the grid
def cal_to_grid(self,P_demand,P_coupled,P_h2_produced,P_h2_consumed):
P_to_grid = []
for i in range(len(P_demand)):
P = P_coupled[i] + P_h2_produced[i] - P_h2_consumed[i]
if P >= P_demand[i]:
P_to_grid.append(P_demand[i])
else:
P_to_grid.append(P)
return P_to_grid
# calculate the power to pem system
def cal_to_h2sys(self,P_h2_produced,P_h2_consumed,P_abandon):
P_to_h2sys_array = []
for i in range(len(P_h2_consumed)):
#P_to_h2sys = P_h2_consumed[i] + P_abandon[i] - P_h2_produced[i]
P_to_h2sys = P_h2_consumed[i] - P_h2_produced[i]
P_to_h2sys_array.append(P_to_h2sys)
return P_to_h2sys_array
# calculate ratio fit the demand
def ratio_fit(self,P_demand,P_to_grid):
ratio_list = []
for i in range(len(P_demand)):
#ratio_demand = P_to_grid[i]/P_demand[i]
ratio = P_to_grid[i]/P_demand[i]
# calculate fitting ratio
#if ratio_demand == 1.0:
# ratio = 1.0
#else:
# ratio = 0.0
ratio_list.append(ratio)
# calculate average ratio
ratio_ave = sum(ratio_list)/len(ratio_list)
return ratio_ave
# calculate total energy send to grid
def cal_e_acc_grid(self,P_to_grid,time):
e_acc_to_grid = [0.0]
for i in range(1,len(time)):
e_grid = P_to_grid[i-1] * (time[i]-time[i-1]) * 60 # power produced in the period, MWs
e_grid = e_grid/3600.0 # convert to MWh
e_acc_grid = e_acc_to_grid[i-1] + e_grid
e_acc_to_grid.append(e_acc_grid)
return e_acc_to_grid
# calculate total energy send to pem system
def cal_e_acc_h2sys(self,P_to_h2sys,time):
e_acc_to_h2sys = [0.0]
e_acc_from_h2sys = [0.0]
e_acc_net_h2sys = [0.0]
for i in range(1,len(time)):
e_h2 = P_to_h2sys[i-1] * (time[i]-time[i-1]) * 60 # please note the unit of time is min here
e_h2 = e_h2/3600.0 # convert MWmin to MWh
if e_h2 > 0:
e_acc_h2 = e_acc_to_h2sys[i-1] + e_h2
e_acc_to_h2sys.append(e_acc_h2)
e_acc_from_h2sys.append(e_acc_from_h2sys[i-1])
elif e_h2 < 0:
e_acc_h2 = e_acc_from_h2sys[i-1] + e_h2
e_acc_to_h2sys.append(e_acc_to_h2sys[i-1])
e_acc_from_h2sys.append(e_acc_h2)
else:
e_acc_to_h2sys.append(e_acc_to_h2sys[i-1])
e_acc_from_h2sys.append(e_acc_from_h2sys[i-1])
e_acc_net = e_acc_net_h2sys[i-1] + e_h2
e_acc_net_h2sys.append(e_acc_net)
return e_acc_to_h2sys,e_acc_from_h2sys,e_acc_net_h2sys
# calculate total energy abondoned, in MWh
def cal_e_abandon(self,P_res,time):
e_abandon = [0.0]
e_acc_abandon = [0.0]
for i in range(1,len(time)):
e_ab = P_res[i-1] * (time[i]-time[i-1]) # please note the unit of time is min here
e_ab = e_ab/60.0 # convert MWmin to MWh
e_ab_acc = e_acc_abandon[i-1] + e_ab
e_abandon.append(e_ab)
e_acc_abandon.append(e_ab_acc)
return e_abandon, e_acc_abandon