-
Notifications
You must be signed in to change notification settings - Fork 4
/
generate_state_space.py
110 lines (88 loc) · 4.04 KB
/
generate_state_space.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
# Kevin Moy, 7/24/2020
# Uses IEEE 13-bus system to scale loads and determine state space for RL agent learning
# Return vector of load values given load inputs
import win32com.client
import pandas as pd
import os
import numpy as np
MAX_NUM_CONFIG = 20
MIN_BUS_VOLT = 0.9
MAX_BUS_VOLT = 1.1
def load_states(loadNames, DSSCircuit, DSSSolution, min_load=0.1, max_load=3):
# Currently, only takes in IEEE 13 bus OpenDSS as input
# loadNames: vector of bus names for each load
# DSSCircuit: object of type DSSObj.ActiveCircuit (COM interface for OpenDSS Circuit)
# DSSSolution: object of type DSSObj.ActiveCircuit.Solution
# Randomly scales loads with uniform distribution between min_load and max_load (percentages)
# Then solves circuit with all capacitors off, then all capacitors on
# If the voltages fall within the MIN_BUS_VOLT and MAX_BUS_VOLT range,
# then the load configuration is returned as an array
while True:
# Array of random numbers to scale loads by:
randScale = np.random.uniform(min_load, max_load, loadNames.__len__())
scale_up(DSSCircuit, randScale)
# Initially disable both capacitor banks and set both to 1500 KVAR rating
capNames = DSSCircuit.Capacitors.AllNames
for cap in capNames:
DSSCircuit.SetActiveElement("Capacitor." + cap)
DSSCircuit.ActiveDSSElement.Properties("kVAR").Val = 1500
DSSCircuit.Capacitors.Name = "Cap1"
DSSCircuit.Capacitors.States = (0,)
DSSCircuit.Capacitors.Name = "Cap2"
DSSCircuit.Capacitors.States = (0,)
# Solve the Circuit
DSSSolution.Solve()
# Retrieve all voltage magnitudes from all phases of all buses
VoltageMagCapOff = DSSCircuit.AllBusVmagPu
maxVCapOff = max(VoltageMagCapOff)
minVCapOff = min(VoltageMagCapOff)
# ----- DETERMINING STATE SPACE -----
# print("Enabling Capacitor Banks")
# Enable both capacitor banks
DSSCircuit.Capacitors.Name = "Cap1"
DSSCircuit.Capacitors.States = (1,)
DSSCircuit.Capacitors.Name = "Cap2"
DSSCircuit.Capacitors.States = (1,)
# Solve the Circuit
DSSSolution.Solve()
# Retrieve all voltage magnitudes from all phases of all buses
VoltageMagCapOn = DSSCircuit.AllBusVmagPu
maxVCapOn = max(VoltageMagCapOn)
minVCapOn = min(VoltageMagCapOn)
if (min(maxVCapOff, maxVCapOn) > MAX_BUS_VOLT) & \
(max(minVCapOff, minVCapOn) < MIN_BUS_VOLT):
print("Voltages not within acceptable range [" + str(MIN_BUS_VOLT) + ", " + str(MAX_BUS_VOLT) +
"] p.u., not saving")
print("Min-Max voltage: ", min(maxVCapOff, maxVCapOn, minVCapOff, minVCapOn))
print("Max-min voltage: ", max(maxVCapOff, maxVCapOn, minVCapOff, minVCapOn))
scale_down(DSSCircuit, randScale)
else:
# print("Voltages within acceptable range")
loadKws = []
for loadnum in range(np.size(loadNames)):
DSSCircuit.SetActiveElement("Load." + loadNames[loadnum])
# Get bus of load
kwLoad = DSSCircuit.ActiveDSSElement.Properties("kW").Val
loadKws.append(kwLoad)
scale_down(DSSCircuit, randScale)
return np.array(loadKws)
def scale_up(DSSCircuit, randScale):
# Step through every load and scale it up by a random percentage
iLoads = DSSCircuit.Loads.First
rdx = 0
while iLoads:
# Scale load up
DSSCircuit.Loads.kW = DSSCircuit.Loads.kW * randScale[rdx]
# Move to next load and random number
iLoads = DSSCircuit.Loads.Next
rdx += 1
def scale_down(DSSCircuit, randScale):
# Step through every load and scale back down by the same random percentage
iLoads = DSSCircuit.Loads.First
rdx = 0
while iLoads:
# Scale load down
DSSCircuit.Loads.kW = DSSCircuit.Loads.kW / randScale[rdx]
# Move to next load and random number
iLoads = DSSCircuit.Loads.Next
rdx += 1