-
Notifications
You must be signed in to change notification settings - Fork 3
/
pvoutputorg.py
128 lines (104 loc) · 5.3 KB
/
pvoutputorg.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
import requests
import time
import math
from copy import copy
from datetime import datetime
from pvconf import PvConf
class PvOutputOrg:
def __init__(self, conf: PvConf, logger):
self.conf = conf
self.logger = logger
self.logger.debug("PvOutputOrg class instantiated")
def write_pvdata_to_pvoutput(self, fusionsolar_json_data):
if self.conf.pvoutput:
pvoutput_header_obj = {
"X-Pvoutput-Apikey": self.conf.pvoutputapikey,
"X-Pvoutput-SystemId": self.conf.pvoutputsystemid,
}
pvoutput_data_obj = self.make_pvoutput_pvdata_obj(fusionsolar_json_data)
try:
self.logger.info(
"Writing to PVOutput. Header: {} Data: {}".format(
pvoutput_header_obj, pvoutput_data_obj
)
)
api_response = requests.post(
self.conf.pvoutputurl,
data=pvoutput_data_obj,
headers=pvoutput_header_obj,
)
self.logger.debug("PVOutput response {}".format(api_response.text))
except Exception as e:
raise Exception(
"Exception while posting data to PVOutput: '{}'".format(str(e))
)
else:
self.logger.debug("PVOutput writing disabled")
def make_pvoutput_pvdata_obj(self, response_json_data):
localtime = time.localtime()
pvodate = time.strftime("%Y%m%d", localtime)
pvotime = time.strftime("%H:%M", localtime)
pvoutput_data_obj = {
"d": pvodate,
"t": pvotime,
"v1": float(response_json_data["realKpi"]["cumulativeEnergy"]),
"v2": float(response_json_data["powerCurve"]["currentPower"]),
"c1": 2,
}
return pvoutput_data_obj
def write_griddata_to_pvoutput(self, grid_data_obj):
if self.conf.pvoutput:
pvoutput_header_obj = {
"X-Pvoutput-Apikey": self.conf.pvoutputapikey,
"X-Pvoutput-SystemId": self.conf.pvoutputsystemid,
}
# Apply span
grid_data_obj_copy = copy(grid_data_obj)
grid_net_consumption_new = []
new_elem = None
for idx, element in enumerate(grid_data_obj["grid_net_consumption"]):
if(new_elem == None):
new_elem = copy(element)
else:
new_elem['timestamp'] = element['timestamp']
new_elem['interval_power_avg'] += element['interval_power_avg']
new_elem['interval_energy'] += element['interval_energy']
residual = (idx + 1) % self.conf.gridrelaypvoutputspan
if residual == 0:
new_elem['interval_power_avg'] = new_elem['interval_power_avg'] / self.conf.gridrelaypvoutputspan
grid_net_consumption_new.append(new_elem)
new_elem = None
grid_data_obj_copy['grid_net_consumption'] = grid_net_consumption_new
# PVOutput allows max 30 records per batch
pages = math.ceil(len(grid_data_obj_copy["grid_net_consumption"]) / 30)
if (len(grid_data_obj_copy["grid_net_consumption"]) % self.conf.gridrelaypvoutputspan > 0):
self.logger.warn("WARNING! The number of measurements per 24h as supplied by meetdata.nl should be dividable by configured gridrelaypvoutputspan parameter.")
self.logger.warn("Measurements in 24h: {}, configured span: {}.".format(len(grid_data_obj["grid_net_consumption"]),self.conf.gridrelaypvoutputspan,))
for page in range(pages):
pvoutput_batch_data_obj = self.make_pvoutput_griddata_obj_page(grid_data_obj_copy, page)
try:
self.logger.info("Writing GridData batch to PVOutput. Header: {} Data: {}".format(pvoutput_header_obj, pvoutput_batch_data_obj))
api_response = requests.post(
self.conf.pvoutputbatchurl,
data=pvoutput_batch_data_obj,
headers=pvoutput_header_obj,
)
self.logger.debug("PVOutput GridData response {}".format(api_response.text))
except Exception as e:
raise Exception(
"Exception while posting GridData to PVOutput: '{}'".format(
str(e)
)
)
else:
self.logger.debug("PVOutput writing disabled")
def make_pvoutput_griddata_obj_page(self, grid_data_obj, page):
pvoutput_data = []
for idx, measurement in enumerate(grid_data_obj["grid_net_consumption"]):
rec_num = idx + 1
if (rec_num > page * 30) and (rec_num <= page * 30 + 30):
local_date = datetime.fromtimestamp(measurement["timestamp"]).strftime("%Y%m%d")
local_time = datetime.fromtimestamp(measurement["timestamp"]).strftime("%H:%M")
pvoutput_data.append(f"{local_date},{local_time},-1,0,-1,{int(measurement['interval_power_avg'])}")
pvoutput_data_obj = {"c1": 0, "n": 1, "data": ";".join(pvoutput_data)}
return pvoutput_data_obj