forked from Dan-in-CA/SIP
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsip.py
executable file
·258 lines (225 loc) · 11.2 KB
/
sip.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
#!/usr/bin/env python2
# -*- coding: utf-8 -*-
import i18n
import subprocess
import json
import ast
import time
import thread
from calendar import timegm
import sys
sys.path.append('./plugins')
import web # the Web.py module. See webpy.org (Enables the Python SIP web interface)
import gv
from helpers import (
report_station_completed,
plugin_adjustment,
prog_match,
schedule_stations,
log_run,
stop_onrain,
check_rain,
jsave,
station_names,
get_rpi_revision
)
from urls import urls # Provides access to URLs for UI pages
from gpio_pins import set_output
from ReverseProxied import ReverseProxied
# do not call set output until plugins are loaded because it should NOT be called
# if gv.use_gpio_pins is False (which is set in relay board plugin.
# set_output()
gv.restarted = 1
def timing_loop():
""" ***** Main timing algorithm. Runs in a separate thread.***** """
try:
print _('Starting timing loop') + '\n'
except Exception:
pass
last_min = 0
while True: # infinite loop
gv.nowt = time.localtime() # Current time as time struct. Updated once per second.
gv.now = timegm(gv.nowt) # Current time as timestamp based on local time from the Pi. Updated once per second.
if (gv.sd['en']
and not gv.sd['mm']
and (not gv.sd['bsy'] or not gv.sd['seq'])
):
if gv.now / 60 != last_min: # only check programs once a minute
last_min = gv.now / 60
extra_adjustment = plugin_adjustment()
for i, p in enumerate(gv.pd): # get both index and prog item
# check if program time matches current time, is active, and has a duration
if prog_match(p) and p[0] and p[6]:
# check each station for boards listed in program up to number of boards in Options
for b in range(len(p[7:7 + gv.sd['nbrd']])):
for s in range(8):
sid = b * 8 + s # station index
if gv.sd['mas'] == sid + 1:
continue # skip if this is master station
if gv.srvals[sid] and gv.sd['seq']: # skip if currently on and sequential mode
continue
# station duration conditionally scaled by "water level"
if gv.sd['iw'][b] & 1 << s:
duration_adj = 1.0
if gv.sd['idd'] == 1:
duration = p[-1][sid]
else:
duration = p[6]
else:
duration_adj = (float(gv.sd['wl']) / 100) * extra_adjustment
if gv.sd['idd'] == 1:
duration = p[-1][sid] * duration_adj
else:
duration = p[6] * duration_adj
duration = int(round(duration)) # convert to int
if p[7 + b] & 1 << s: # if this station is scheduled in this program
if gv.sd['seq']: # sequential mode
gv.rs[sid][2] = duration
gv.rs[sid][3] = i + 1 # store program number for scheduling
gv.ps[sid][0] = i + 1 # store program number for display
gv.ps[sid][1] = duration
else: # concurrent mode
if gv.srvals[sid]: # if currently on, log result
gv.lrun[0] = sid
gv.lrun[1] = gv.rs[sid][3]
gv.lrun[2] = int(gv.now - gv.rs[sid][0])
gv.lrun[3] = gv.now # think this is unused
log_run()
report_station_completed(sid + 1)
gv.rs[sid][2] = duration
gv.rs[sid][3] = i + 1 # store program number
gv.ps[sid][0] = i + 1 # store program number for display
gv.ps[sid][1] = duration
schedule_stations(p[7:7 + gv.sd['nbrd']]) # turns on gv.sd['bsy']
if gv.sd['bsy']:
for b in range(gv.sd['nbrd']): # Check each station once a second
for s in range(8):
sid = b * 8 + s # station index
if gv.srvals[sid]: # if this station is on
if gv.now >= gv.rs[sid][1]: # check if time is up
gv.srvals[sid] = 0
set_output()
gv.sbits[b] &= ~(1 << s)
if gv.sd['mas'] != sid +1 : # if not master, fill out log
gv.ps[sid] = [0, 0]
gv.lrun[0] = sid
gv.lrun[1] = gv.rs[sid][3]
gv.lrun[2] = int(gv.now - gv.rs[sid][0])
gv.lrun[3] = gv.now
log_run()
report_station_completed(sid + 1)
gv.pon = None # Program has ended
gv.rs[sid] = [0, 0, 0, 0]
else: # if this station is not yet on
if gv.rs[sid][0] <= gv.now < gv.rs[sid][1]:
if gv.sd['mas'] != sid + 1 : # if not master
gv.srvals[sid] = 1 # station is turned on
set_output()
gv.sbits[b] |= 1 << s # Set display to on
gv.ps[sid][0] = gv.rs[sid][3]
gv.ps[sid][1] = gv.rs[sid][2]
if gv.sd['mas'] and gv.sd['mo'][b] & (1 << s): # Master settings
masid = gv.sd['mas'] - 1 # master index
gv.rs[masid][0] = gv.rs[sid][0] + gv.sd['mton']
gv.rs[masid][1] = gv.rs[sid][1] + gv.sd['mtoff']
gv.rs[masid][3] = gv.rs[sid][3]
elif gv.sd['mas'] == sid + 1:
gv.sbits[b] |= 1 << sid
gv.srvals[masid] = 1
set_output()
for s in range(gv.sd['nst']):
if gv.rs[s][1]: # if any station is scheduled
program_running = True
gv.pon = gv.rs[s][3] # Store number of running program
break
program_running = False
gv.pon = None
if program_running:
if gv.sd['urs'] and gv.sd['rs']: # Stop stations if use rain sensor and rain detected.
stop_onrain() # Clear schedule for stations that do not ignore rain.
for sid in range(len(gv.rs)): # loop through program schedule (gv.ps)
if gv.rs[sid][2] == 0: # skip stations with no duration
continue
if gv.srvals[sid]: # If station is on, decrement time remaining display
if gv.ps[sid][1] > 0: # if time is left
gv.ps[sid][1] -= 1
if not program_running:
gv.srvals = [0] * (gv.sd['nst'])
set_output()
gv.sbits = [0] * (gv.sd['nbrd'] + 1)
gv.ps = []
for i in range(gv.sd['nst']):
gv.ps.append([0, 0])
gv.rs = []
for i in range(gv.sd['nst']):
gv.rs.append([0, 0, 0, 0])
gv.sd['bsy'] = 0
if (gv.sd['mas'] # master is defined
and (gv.sd['mm'] or not gv.sd['seq']) # manual or concurrent mode.
):
for b in range(gv.sd['nbrd']): # set stop time for master
for s in range(8):
sid = b * 8 + s
if (gv.sd['mas'] != sid + 1 # if not master
and gv.srvals[sid] # station is on
and gv.rs[sid][1] >= gv.now # station has a stop time >= now
and gv.sd['mo'][b] & (1 << s) # station activates master
):
gv.rs[gv.sd['mas'] - 1][1] = gv.rs[sid][1] + gv.sd['mtoff'] # set to future...
break # first found will do
if gv.sd['urs']:
check_rain() # in helpers.py
if gv.sd['rd'] and gv.now >= gv.sd['rdst']: # Check if rain delay time is up
gv.sd['rd'] = 0
gv.sd['rdst'] = 0 # Rain delay stop time
jsave(gv.sd, 'sd')
time.sleep(1)
#### End of timing loop ####
class SIPApp(web.application):
"""Allow program to select HTTP port."""
def run(self, port=gv.sd['htp'], ip=gv.sd['htip'], *middleware): # get port number from options settings
func = self.wsgifunc(*middleware)
func = ReverseProxied(func)
return web.httpserver.runsimple(func, (ip, port))
app = SIPApp(urls, globals())
# disableShiftRegisterOutput()
web.config.debug = False # Improves page load speed
if web.config.get('_session') is None:
web.config._session = web.session.Session(app, web.session.DiskStore('sessions'),
initializer={'user': 'anonymous'})
template_globals = {
'gv': gv,
'str': str,
'eval': eval,
'session': web.config._session,
'json': json,
'ast': ast,
'_': _,
'i18n': i18n,
'app_path': lambda p: web.ctx.homepath + p,
'web': web,
}
template_render = web.template.render('templates', globals=template_globals, base='base')
if __name__ == '__main__':
#########################################################
#### Code to import all webpages and plugin webpages ####
import plugins
try:
print _('plugins loaded:')
except Exception:
pass
for name in plugins.__all__:
print ' ', name
gv.plugin_menu.sort(key=lambda entry: entry[0])
# Keep plugin manager at top of menu
try:
for i, item in enumerate(gv.plugin_menu):
if '/plugins' in item:
gv.plugin_menu.pop(i)
except Exception:
pass
thread.start_new_thread(timing_loop, ())
if gv.use_gpio_pins:
set_output()
app.notfound = lambda: web.seeother('/')
app.run()