-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprocessext_processlist.php
168 lines (146 loc) · 7.96 KB
/
processext_processlist.php
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
<?php
/*
All Emoncms code is released under the GNU Affero General Public License.
See COPYRIGHT.txt and LICENSE.txt.
---------------------------------------------------------------------
Emoncms - open source energy visualisation
Part of the OpenEnergyMonitor project: http://openenergymonitor.org
*/
// no direct access
defined('EMONCMS_EXEC') or die('Restricted access');
// Extended Processlist Module
class processext_ProcessList
{
private $log;
private $schedule;
// Module required constructor, receives parent as reference
public function __construct(&$parent)
{
$this->log = new EmonLogger(__FILE__);
}
public function process_list() {
$list = array(
array(
"name"=>_("pow10 input"),
"short"=>"s inp",
"argtype"=>ProcessArg::INPUTID,
"function"=>"pow10_input",
"datafields"=>0,
"unit"=>"",
"group"=>_("Input"),
"description"=>_("<p>Scales the current value by ten to the power of the other input.</p>")
),
array(
"name"=>_("value and scale"),
"short"=>"v+s",
"argtype"=>ProcessArg::NONE,
"function"=>"value_and_scale",
"datafields"=>0,
"unit"=>"",
"group"=>_("Calibration"),
"description"=>_("<p>Unpack combined value and scale factor, applying the scale factor.</p>")
),
array(
"name"=>_("kWh to Power (smoothed)"),
"short"=>"kwhpwr_smooth",
"argtype"=>ProcessArg::FEEDID,
"function"=>"kwh_to_power_smoothed",
"datafields"=>1,
"unit"=>"W",
"group"=>_("Power & Energy"),
"engines"=>array(Engine::PHPFINA,Engine::PHPTIMESERIES),
"requireredis"=>true,
"description"=>_("<p>Convert accumulating kWh to instantaneous power, averaging over the last 10 input values. This is useful when the kWh measurement is derived from pulse counting because a short (e.g. 5s) sampling interval leads to quantisation which causes the power level to appear to oscillate between coarse-grained levels (720W for 1 Wh pulses counted for 5s). This filter delays power output for 10 samples in order to give more fine grained output. Quantisation becomes less apparent but changes of power level are delayed slightly and spread over the expanded window.</p>")
),
array(
"name"=>_("Wh Accumulator No Limit"),
"short"=>"whaccnl",
"argtype"=>ProcessArg::FEEDID,
"function"=>"wh_accumulator_no_limit",
"datafields"=>1,
"unit"=>"Wh",
"group"=>_("Main"),
"engines"=>array(Engine::PHPFINA,Engine::PHPTIMESERIES),
"requireredis"=>true,
"description"=>_("<b>Wh Accumulator:</b> Use with emontx, emonth or emonpi pulsecount or an emontx running firmware <i>emonTxV3_4_continuous_kwhtotals</i> sending cumulative watt hours.<br><br>This processor ensures that when the emontx is reset the watt hour count in emoncms does not reset, this no-limit version does not check filters for spikes in energy use that are larger than a max power threshold set in the processor, and does not assume these are errors, so it does not implement the max power threshold set to 60 kW on the regular Wh Accumulator. <br><br><b>Visualisation tip:</b> Feeds created with this input processor can be used to generate daily kWh data using the BarGraph visualisation with the delta property set to 1 and scale set to 0.001. See: <a href='https://guide.openenergymonitor.org/setup/daily-kwh/' target='_blank' rel='noopener'>Guide: Daily kWh</a><br><br>")
)
);
return $list;
}
// / Below are functions of this module processlist, same name must exist on process_list()
//---------------------------------------------------------------------------------------
// Combines a value and scale input, multiplies the value by 10^scale
//
// These split value and scale inputs are output from the Modbus interface of solar edge
// inverters.
//---------------------------------------------------------------------------------------
public function value_and_scale($arg, $time, $value)
{
$scale = $value & 0xFFFF; // Right 16 bits are the scale
if($scale >= 0x8000) { // Decode twos complement if negative
$scale = -(($scale ^ 0xFFFF)+1);
}
$val = $value >> 16; // Left 16 bits are the value
// Apply scale factor to the value
return $val * pow(10, $scale);
}
//---------------------------------------------------------------------------------------
// Scales current value by 10 to the power of another input.
//---------------------------------------------------------------------------------------
public function pow10_input($id, $time, $value)
{
$input_value = $this->input->get_last_value($id);
return $value * pow(10, $input_value);
}
//---------------------------------------------------------------------------------------
// Delayed / smoothed kWh to power input processor by Robert Norton
// See: https://github.com/emoncms/emoncms/pull/1686
//---------------------------------------------------------------------------------------
public function kwh_to_power_smoothed($feedid,$time,$value)
{
global $redis;
if (!$redis) return $value; // return if redis is not available
$power = 0;
$queue_len = 10;
// fetch the oldest samples in the queue
$last_time = $redis->lIndex("process:kwhtopower_smoothed:$feedid:times",-1);
$last_value = $redis->lIndex("process:kwhtopower_smoothed:$feedid:values",-1);
// Push the new sample and trim to $queue_len entries (actually queue_len+1 but shrug)
$redis->lPush("process:kwhtopower_smoothed:$feedid:times", $time);
$redis->lTrim("process:kwhtopower_smoothed:$feedid:times", 0, $queue_len);
$redis->lPush("process:kwhtopower_smoothed:$feedid:values", $value);
$redis->lTrim("process:kwhtopower_smoothed:$feedid:values", 0, $queue_len);
if (($last_time !== false) && ($last_value !== false)) { // will fail on first call
// compute the power between now and queue_len samples ago
$kwhinc = $value - $last_value;
$joules = $kwhinc * 3600000.0;
$timeelapsed = ($time - $last_time);
if ($timeelapsed>0) { //This only avoids a crash, it's not ideal to return "power = 0" to the next process.
$power = $joules / $timeelapsed;
$this->feed->insert_data($feedid, $time, $time, $power);
} else {
$this->log->error("Time elapsed was not greater than zero: $timeelapsed.");
}
}
return $power;
}
public function wh_accumulator_no_limit($feedid, $time, $value)
{
$totalwh = $value;
global $redis;
if (!$redis) return $value; // return if redis is not available
if ($redis->exists("process:whaccumulatornl:$feedid")) {
$last_input = $redis->hmget("process:whaccumulatornl:$feedid",array('time','value'));
$last_feed = $this->feed->get_timevalue($feedid);
if ($last_feed===null) return $value; // feed does not exist
$totalwh = $last_feed['value'];
$time_diff = $time - $last_feed['time'];
$val_diff = $value - $last_input['value'];
if ($time_diff>0 && $val_diff>0) $totalwh += $val_diff;
$padding_mode = "join";
$this->feed->insert_data($feedid, $time, $time, $totalwh, $padding_mode);
}
$redis->hMset("process:whaccumulatornl:$feedid", array('time' => $time, 'value' => $value));
return $totalwh;
}
}