##// END OF EJS Templates
Merge branch 'schain_alarm' of http://jro-dev.igp.gob.pe/rhodecode/schain into v2.3
jespinoza -
r1131:e8ab492363ef merge
parent child
Show More
1 NO CONTENT: new file 100644, binary diff hidden
NO CONTENT: new file 100644, binary diff hidden
1 NO CONTENT: new file 100644, binary diff hidden
NO CONTENT: new file 100644, binary diff hidden
@@ -1,395 +1,498
1 """The admin module contains all administrative classes relating to the schain python api.
1 """The admin module contains all administrative classes relating to the schain python api.
2
2
3 The main role of this module is to send some reports. It contains a
3 The main role of this module is to send some reports. It contains a
4 notification class and a standard error handing class.
4 notification class and a standard error handing class.
5
5
6 $Id: admin.py 3966 2015-12-01 14:32:29Z miguel.urco $
6 $Id: admin.py 3966 2015-12-01 14:32:29Z miguel.urco $
7 """
7 """
8 import os, sys
8 import os
9 import sys
10 import time
9 import traceback
11 import traceback
10 import smtplib
12 import smtplib
11 import ConfigParser
13 import ConfigParser
12 import StringIO
14 import StringIO
13
15 from threading import Thread
16 from multiprocessing import Process
14 from email.mime.text import MIMEText
17 from email.mime.text import MIMEText
15 from email.mime.application import MIMEApplication
18 from email.mime.application import MIMEApplication
16 from email.mime.multipart import MIMEMultipart
19 from email.mime.multipart import MIMEMultipart
17
20
18 class SchainConfigure():
21 from schainpy.utils import log
22 from schainpy.model.graphics.jroplot_data import popup
23
24 def get_path():
25 '''
26 Return schainpy path
27 '''
28
29 try:
30 root = __file__
31 if os.path.islink(root):
32 root = os.path.realpath(root)
33
34 return os.path.dirname(os.path.abspath(root))
35 except:
36 log.error('I am sorry, but something is wrong... __file__ not found')
37
38 class Alarm(Process):
39 '''
40 modes:
41 0 - All
42 1 - Sound alarm
43 2 - Send email
44 3 - Popup message
45 4 - Send to alarm system TODO
46 '''
47
48 def __init__(self, modes=[1], **kwargs):
49 Process.__init__(self)
50 self.modes = modes
51 self.kwargs = kwargs
52
53 @staticmethod
54 def play_sound():
55 sound = os.path.join(get_path(), 'alarm1.oga')
56 if os.path.exists(sound):
57 for __ in range(2):
58 os.system('paplay {}'.format(sound))
59 time.sleep(0.5)
60 else:
61 log.warning('Unable to play alarm, sound file not found', 'ADMIN')
62
63 @staticmethod
64 def send_email(**kwargs):
65 notifier = SchainNotify()
66 notifier.notify(**kwargs)
67
68 @staticmethod
69 def show_popup(message='Error'):
70 popup(message)
71
72 @staticmethod
73 def send_alarm():
74 pass
75
76 @staticmethod
77 def get_kwargs(kwargs, keys):
78 ret = {}
79 for key in keys:
80 ret[key] = kwargs[key]
81 return ret
82
83 def run(self):
84 tasks = {
85 1 : self.send_email,
86 2 : self.play_sound,
87 3 : self.show_popup,
88 4 : self.send_alarm,
89 }
90
91 tasks_args = {
92 1: ['email', 'message', 'subject', 'subtitle', 'filename'],
93 2: [],
94 3: ['message'],
95 4: [],
96 }
97 procs = []
98 for mode in self.modes:
99 if 0 in self.modes:
100 for x in tasks:
101 t = Thread(target=tasks[x], kwargs=self.get_kwargs(self.kwargs, tasks_args[x]))
102 t.start()
103 procs.append(t)
104 break
105 else:
106 t = Thread(target=tasks[mode], kwargs=self.get_kwargs(self.kwargs, tasks_args[mode]))
107 t.start()
108 procs.append(t)
109 for t in procs:
110 t.join()
111
112
113 class SchainConfigure():
19
114
20 __DEFAULT_ADMINISTRATOR_EMAIL = ""
115 __DEFAULT_ADMINISTRATOR_EMAIL = "juan.espinoza@jro.igp.gob.pe"
21 __DEFAULT_EMAIL_SERVER = "jro-zimbra.igp.gob.pe"
116 __DEFAULT_EMAIL_SERVER = "jro-zimbra.igp.gob.pe"
22 __DEFAULT_SENDER_EMAIL = "notifier-schain@jro.igp.gob.pe"
117 __DEFAULT_SENDER_EMAIL = "notifier-schain@jro.igp.gob.pe"
23 __DEFAULT_SENDER_PASS = ""
118 __DEFAULT_SENDER_PASS = ""
24
119
25 __SCHAIN_ADMINISTRATOR_EMAIL = "CONTACT"
120 __SCHAIN_ADMINISTRATOR_EMAIL = "CONTACT"
26 __SCHAIN_EMAIL_SERVER = "MAILSERVER"
121 __SCHAIN_EMAIL_SERVER = "MAILSERVER"
27 __SCHAIN_SENDER_EMAIL = "MAILSERVER_ACCOUNT"
122 __SCHAIN_SENDER_EMAIL = "MAILSERVER_ACCOUNT"
28 __SCHAIN_SENDER_PASS = "MAILSERVER_PASSWORD"
123 __SCHAIN_SENDER_PASS = "MAILSERVER_PASSWORD"
29
124
30 def __init__(self, initFile = None):
125 def __init__(self, initFile = None):
31
126
32 # Set configuration file
127 # Set configuration file
33 if (initFile == None):
128 if (initFile == None):
34 self.__confFilePath = "/etc/schain.conf"
129 self.__confFilePath = "/etc/schain.conf"
35 else:
130 else:
36 self.__confFilePath = initFile
131 self.__confFilePath = initFile
37
132
38 # open configuration file
133 # open configuration file
39 try:
134 try:
40 self.__confFile = open(self.__confFilePath, "r")
135 self.__confFile = open(self.__confFilePath, "r")
41 except IOError:
136 except IOError:
42 # can't read from file - use all hard-coded values
137 # can't read from file - use all hard-coded values
43 self.__initFromHardCode()
138 self.__initFromHardCode()
44 return
139 return
45
140
46 # create Parser using standard module ConfigParser
141 # create Parser using standard module ConfigParser
47 self.__parser = ConfigParser.ConfigParser()
142 self.__parser = ConfigParser.ConfigParser()
48
143
49 # read conf file into a StringIO with "[madrigal]\n" section heading prepended
144 # read conf file into a StringIO with "[madrigal]\n" section heading prepended
50 strConfFile = StringIO.StringIO("[schain]\n" + self.__confFile.read())
145 strConfFile = StringIO.StringIO("[schain]\n" + self.__confFile.read())
51
146
52 # parse StringIO configuration file
147 # parse StringIO configuration file
53 self.__parser.readfp(strConfFile)
148 self.__parser.readfp(strConfFile)
54
149
55 # read information from configuration file
150 # read information from configuration file
56 self.__readConfFile()
151 self.__readConfFile()
57
152
58 # close conf file
153 # close conf file
59 self.__confFile.close()
154 self.__confFile.close()
60
155
61
156
62 def __initFromHardCode(self):
157 def __initFromHardCode(self):
63
158
64 self.__sender_email = self.__DEFAULT_SENDER_EMAIL
159 self.__sender_email = self.__DEFAULT_SENDER_EMAIL
65 self.__sender_pass = self.__DEFAULT_SENDER_PASS
160 self.__sender_pass = self.__DEFAULT_SENDER_PASS
66 self.__admin_email = self.__DEFAULT_ADMINISTRATOR_EMAIL
161 self.__admin_email = self.__DEFAULT_ADMINISTRATOR_EMAIL
67 self.__email_server = self.__DEFAULT_EMAIL_SERVER
162 self.__email_server = self.__DEFAULT_EMAIL_SERVER
68
163
69 def __readConfFile(self):
164 def __readConfFile(self):
70 """__readConfFile is a private helper function that reads information from the parsed config file.
165 """__readConfFile is a private helper function that reads information from the parsed config file.
71
166
72 Inputs: None
167 Inputs: None
73
168
74 Returns: Void.
169 Returns: Void.
75
170
76 Affects: Initializes class member variables that are found in the config file.
171 Affects: Initializes class member variables that are found in the config file.
77
172
78 Exceptions: MadrigalError thrown if any key not found.
173 Exceptions: MadrigalError thrown if any key not found.
79 """
174 """
80
175
81 # get the sender email
176 # get the sender email
82 try:
177 try:
83 self.__sender_email = self.__parser.get("schain", self.__SCHAIN_SENDER_EMAIL)
178 self.__sender_email = self.__parser.get("schain", self.__SCHAIN_SENDER_EMAIL)
84 except:
179 except:
85 self.__sender_email = self.__DEFAULT_SENDER_EMAIL
180 self.__sender_email = self.__DEFAULT_SENDER_EMAIL
86
181
87 # get the sender password
182 # get the sender password
88 try:
183 try:
89 self.__sender_pass = self.__parser.get("schain", self.__SCHAIN_SENDER_PASS)
184 self.__sender_pass = self.__parser.get("schain", self.__SCHAIN_SENDER_PASS)
90 except:
185 except:
91 self.__sender_pass = self.__DEFAULT_SENDER_PASS
186 self.__sender_pass = self.__DEFAULT_SENDER_PASS
92
187
93 # get the administrator email
188 # get the administrator email
94 try:
189 try:
95 self.__admin_email = self.__parser.get("schain", self.__SCHAIN_ADMINISTRATOR_EMAIL)
190 self.__admin_email = self.__parser.get("schain", self.__SCHAIN_ADMINISTRATOR_EMAIL)
96 except:
191 except:
97 self.__admin_email = self.__DEFAULT_ADMINISTRATOR_EMAIL
192 self.__admin_email = self.__DEFAULT_ADMINISTRATOR_EMAIL
98
193
99 # get the server email
194 # get the server email
100 try:
195 try:
101 self.__email_server = self.__parser.get("schain", self.__SCHAIN_EMAIL_SERVER)
196 self.__email_server = self.__parser.get("schain", self.__SCHAIN_EMAIL_SERVER)
102 except:
197 except:
103 self.__email_server = self.__DEFAULT_EMAIL_SERVER
198 self.__email_server = self.__DEFAULT_EMAIL_SERVER
104
199
105 def getEmailServer(self):
200 def getEmailServer(self):
106
201
107 return self.__email_server
202 return self.__email_server
108
203
109 def getSenderEmail(self):
204 def getSenderEmail(self):
110
205
111 return self.__sender_email
206 return self.__sender_email
112
207
113 def getSenderPass(self):
208 def getSenderPass(self):
114
209
115 return self.__sender_pass
210 return self.__sender_pass
116
211
117 def getAdminEmail(self):
212 def getAdminEmail(self):
118
213
119 return self.__admin_email
214 return self.__admin_email
120
215
121 class SchainNotify:
216 class SchainNotify:
122 """SchainNotify is an object used to send messages to an administrator about a Schain software.
217 """SchainNotify is an object used to send messages to an administrator about a Schain software.
123
218
124 This object provides functions needed to send messages to an administrator about a Schain , for now
219 This object provides functions needed to send messages to an administrator about a Schain , for now
125 only sendAlert, which sends an email to the site administrator found is ADMIN_EMAIL
220 only sendAlert, which sends an email to the site administrator found is ADMIN_EMAIL
126
221
127 Usage example:
222 Usage example:
128
223
129 import schainpy.admin
224 import schainpy.admin
130
225
131 try:
226 try:
132
227
133 adminObj = schainpy.admin.SchainNotify()
228 adminObj = schainpy.admin.SchainNotify()
134 adminObj.sendAlert('This is important!', 'Important Message')
229 adminObj.sendAlert('This is important!', 'Important Message')
135
230
136 except schainpy.admin.SchainError, e:
231 except schainpy.admin.SchainError, e:
137
232
138 print e.getExceptionStr()
233 print e.getExceptionStr()
139
234
140
235
141 Non-standard Python modules used:
236 Non-standard Python modules used:
142 None
237 None
143
238
144 Exceptions thrown: None - Note that SchainNotify tries every trick it knows to avoid
239 Exceptions thrown: None - Note that SchainNotify tries every trick it knows to avoid
145 throwing exceptions, since this is the class that will generally be called when there is a problem.
240 throwing exceptions, since this is the class that will generally be called when there is a problem.
146
241
147 Change history:
242 Change history:
148
243
149 Written by "Miguel Urco":mailto:miguel.urco@jro.igp.gob.pe Dec. 1, 2015
244 Written by "Miguel Urco":mailto:miguel.urco@jro.igp.gob.pe Dec. 1, 2015
150 """
245 """
151
246
152 #constants
247 #constants
153
248
154 def __init__(self):
249 def __init__(self):
155 """__init__ initializes SchainNotify by getting some basic information from SchainDB and SchainSite.
250 """__init__ initializes SchainNotify by getting some basic information from SchainDB and SchainSite.
156
251
157 Note that SchainNotify tries every trick it knows to avoid throwing exceptions, since
252 Note that SchainNotify tries every trick it knows to avoid throwing exceptions, since
158 this is the class that will generally be called when there is a problem.
253 this is the class that will generally be called when there is a problem.
159
254
160 Inputs: Existing SchainDB object, by default = None.
255 Inputs: Existing SchainDB object, by default = None.
161
256
162 Returns: void
257 Returns: void
163
258
164 Affects: Initializes self.__binDir.
259 Affects: Initializes self.__binDir.
165
260
166 Exceptions: None.
261 Exceptions: None.
167 """
262 """
168
263
169 # note that the main configuration file is unavailable
264 # note that the main configuration file is unavailable
170 # the best that can be done is send an email to root using localhost mailserver
265 # the best that can be done is send an email to root using localhost mailserver
171 confObj = SchainConfigure()
266 confObj = SchainConfigure()
172
267
173 self.__emailFromAddress = confObj.getSenderEmail()
268 self.__emailFromAddress = confObj.getSenderEmail()
174 self.__emailPass = confObj.getSenderPass()
269 self.__emailPass = confObj.getSenderPass()
175 self.__emailToAddress = confObj.getAdminEmail()
270 self.__emailToAddress = confObj.getAdminEmail()
176 self.__emailServer = confObj.getEmailServer()
271 self.__emailServer = confObj.getEmailServer()
177
272
178 def sendEmail(self, email_from, email_to, subject='Error running ...', message="", subtitle="", filename="", html_format=True):
273 def sendEmail(self, email_from, email_to, subject='Error running ...', message="", subtitle="", filename="", html_format=True):
179
274
180 if not email_to:
275 if not email_to:
181 return 0
276 return 0
182
277
183 if not self.__emailServer:
278 if not self.__emailServer:
184 return 0
279 return 0
185
280
281 log.success('Sending email to {}...'.format(email_to), 'System')
282
186 msg = MIMEMultipart()
283 msg = MIMEMultipart()
187 msg['Subject'] = subject
284 msg['Subject'] = subject
188 msg['From'] = "(Python SChain API): " + email_from
285 msg['From'] = "(Python SChain API): " + email_from
189 msg['Reply-to'] = email_from
286 msg['Reply-to'] = email_from
190 msg['To'] = email_to
287 msg['To'] = email_to
191
288
192 # That is what u see if dont have an email reader:
289 # That is what u see if dont have an email reader:
193 msg.preamble = 'SChainPy'
290 msg.preamble = 'SChainPy'
194
291
195 if html_format:
292 if html_format:
196 message = "<h1> %s </h1>" %subject + "<h3>" + subtitle.replace("\n", "</h3><h3>\n") + "</h3>" + message.replace("\n", "<br>\n")
293 message = "<h1> %s </h1>" %subject + "<h3>" + subtitle.replace("\n", "</h3><h3>\n") + "</h3>" + message.replace("\n", "<br>\n")
197 message = "<html>\n" + message + '</html>'
294 message = "<html>\n" + message + '</html>'
198
295
199 # This is the textual part:
296 # This is the textual part:
200 part = MIMEText(message, "html")
297 part = MIMEText(message, "html")
201 else:
298 else:
202 message = subject + "\n" + subtitle + "\n" + message
299 message = subject + "\n" + subtitle + "\n" + message
203 part = MIMEText(message)
300 part = MIMEText(message)
204
301
205 msg.attach(part)
302 msg.attach(part)
206
303
207 if os.path.isfile(filename):
304 if filename and os.path.isfile(filename):
208 # This is the binary part(The Attachment):
305 # This is the binary part(The Attachment):
209 part = MIMEApplication(open(filename,"rb").read())
306 part = MIMEApplication(open(filename,"rb").read())
210 part.add_header('Content-Disposition',
307 part.add_header('Content-Disposition',
211 'attachment',
308 'attachment',
212 filename=os.path.basename(filename))
309 filename=os.path.basename(filename))
213 msg.attach(part)
310 msg.attach(part)
214
311
215 # Create an instance in SMTP server
312 # Create an instance in SMTP server
216 try:
313 try:
217 smtp = smtplib.SMTP(self.__emailServer)
314 smtp = smtplib.SMTP(self.__emailServer)
218 except:
315 except:
219 print "***** Could not connect to server %s *****" %self.__emailServer
316 log.error('Could not connect to server {}'.format(self.__emailServer), 'System')
220 return 0
317 return 0
221
318
222 # Start the server:
319 # Start the server:
223 # smtp.ehlo()
320 # smtp.ehlo()
224 if self.__emailPass:
321 if self.__emailPass:
225 smtp.login(self.__emailFromAddress, self.__emailPass)
322 smtp.login(self.__emailFromAddress, self.__emailPass)
226
323
227 # Send the email
324 # Send the email
228 try:
325 try:
229 smtp.sendmail(msg['From'], msg['To'], msg.as_string())
326 smtp.sendmail(msg['From'], msg['To'], msg.as_string())
230 except:
327 except:
231 print "***** Could not send the email to %s *****" %msg['To']
328 log.error('Could not send the email to {}'.format(msg['To']), 'System')
232 smtp.quit()
329 smtp.quit()
233 return 0
330 return 0
234
331
235 smtp.quit()
332 smtp.quit()
333
334 log.success('Email sent ', 'System')
236
335
237 return 1
336 return 1
238
337
239 def sendAlert(self, message, subject = "", subtitle="", filename=""):
338 def sendAlert(self, message, subject = "", subtitle="", filename=""):
240 """sendAlert sends an email with the given message and optional title.
339 """sendAlert sends an email with the given message and optional title.
241
340
242 Inputs: message (string), and optional title (string)
341 Inputs: message (string), and optional title (string)
243
342
244 Returns: void
343 Returns: void
245
344
246 Affects: none
345 Affects: none
247
346
248 Exceptions: None.
347 Exceptions: None.
249 """
348 """
250
349
251 if not self.__emailToAddress:
350 if not self.__emailToAddress:
252 return 0
351 return 0
253
352
254 print "***** Sending alert to %s *****" %self.__emailToAddress
353 print "***** Sending alert to %s *****" %self.__emailToAddress
255 # set up message
354 # set up message
256
355
257 sent=self.sendEmail(email_from=self.__emailFromAddress,
356 sent=self.sendEmail(email_from=self.__emailFromAddress,
258 email_to=self.__emailToAddress,
357 email_to=self.__emailToAddress,
259 subject=subject,
358 subject=subject,
260 message=message,
359 message=message,
261 subtitle=subtitle,
360 subtitle=subtitle,
262 filename=filename)
361 filename=filename)
263
362
264 if not sent:
363 if not sent:
265 return 0
364 return 0
266
365
267 print "***** Your system administrator has been notified *****"
268
269 return 1
366 return 1
270
367
271 def notify(self, email, message, subject = "", subtitle="", filename=""):
368 def notify(self, email, message, subject = "", subtitle="", filename=""):
272 """notify sends an email with the given message and title to email.
369 """notify sends an email with the given message and title to email.
273
370
274 Inputs: email (string), message (string), and subject (string)
371 Inputs: email (string), message (string), and subject (string)
275
372
276 Returns: void
373 Returns: void
277
374
278 Affects: none
375 Affects: none
279
376
280 Exceptions: None.
377 Exceptions: None.
281 """
378 """
282
379
283 print "Notifying to %s ..." %email
380 if email is None:
284
381 email = self.__emailToAddress
285 self.sendEmail(email_from=self.__emailFromAddress,
286 email_to=email,
287 subject=subject,
288 message=message,
289 subtitle=subtitle,
290 filename=filename)
291
382
292 print "***** Your system administrator has been notified *****"
383 self.sendEmail(
384 email_from=self.__emailFromAddress,
385 email_to=email,
386 subject=subject,
387 message=message,
388 subtitle=subtitle,
389 filename=filename
390 )
391
293
392
294 class SchainError(Exception):
393 class SchainError(Exception):
295 """SchainError is an exception class that is thrown for all known errors using Schain Py lib.
394 """SchainError is an exception class that is thrown for all known errors using Schain Py lib.
296
395
297 Usage example:
396 Usage example:
298
397
299 import sys, traceback
398 import sys, traceback
300 import schainpy.admin
399 import schainpy.admin
301
400
302 try:
401 try:
303
402
304 test = open('ImportantFile.txt', 'r')
403 test = open('ImportantFile.txt', 'r')
305
404
306 except:
405 except:
307
406
308 raise schainpy.admin.SchainError('ImportantFile.txt not opened!',
407 raise schainpy.admin.SchainError('ImportantFile.txt not opened!',
309 traceback.format_exception(sys.exc_info()[0],
408 traceback.format_exception(sys.exc_info()[0],
310 sys.exc_info()[1],
409 sys.exc_info()[1],
311 sys.exc_info()[2]))
410 sys.exc_info()[2]))
312 """
411 """
313
412
314
413
315 def __init__(self, strInterpretation, exceptionList=None):
414 def __init__(self, strInterpretation, exceptionList=None):
316 """ __init__ gathers the interpretation string along with all information from sys.exc_info().
415 """ __init__ gathers the interpretation string along with all information from sys.exc_info().
317
416
318 Inputs:
417 Inputs:
319 strIntepretation - A string representing the programmer's interpretation of
418 strIntepretation - A string representing the programmer's interpretation of
320 why the exception occurred
419 why the exception occurred
321
420
322 exceptionList - a list of strings completely describing the exception.
421 exceptionList - a list of strings completely describing the exception.
323 Generated by traceback.format_exception(sys.exc_info()[0],
422 Generated by traceback.format_exception(sys.exc_info()[0],
324 sys.exc_info()[1],
423 sys.exc_info()[1],
325 sys.exc_info()[2])
424 sys.exc_info()[2])
326
425
327 Returns: Void.
426 Returns: Void.
328
427
329 Affects: Initializes class member variables _strInterp, _strExcList.
428 Affects: Initializes class member variables _strInterp, _strExcList.
330
429
331 Exceptions: None.
430 Exceptions: None.
332 """
431 """
333
432
334 if not exceptionList:
433 if not exceptionList:
335 exceptionList = traceback.format_exception(sys.exc_info()[0],
434 exceptionList = traceback.format_exception(sys.exc_info()[0],
336 sys.exc_info()[1],
435 sys.exc_info()[1],
337 sys.exc_info()[2])
436 sys.exc_info()[2])
338
437
339 self._strInterp = strInterpretation
438 self._strInterp = strInterpretation
340 self._strExcList = exceptionList
439 self._strExcList = exceptionList
341
440
342
441
343 def getExceptionStr(self):
442 def getExceptionStr(self):
344 """ getExceptionStr returns a formatted string ready for printing completely describing the exception.
443 """ getExceptionStr returns a formatted string ready for printing completely describing the exception.
345
444
346 Inputs: None
445 Inputs: None
347
446
348 Returns: A formatted string ready for printing completely describing the exception.
447 Returns: A formatted string ready for printing completely describing the exception.
349
448
350 Affects: None
449 Affects: None
351
450
352 Exceptions: None.
451 Exceptions: None.
353 """
452 """
354 excStr = ''
453 excStr = ''
355 excStr = excStr + self._strInterp + '\n\n'
454 excStr = excStr + self._strInterp + '\n\n'
356
455
357 if self._strExcList != None:
456 if self._strExcList != None:
358 for item in self._strExcList:
457 for item in self._strExcList:
359 excStr = excStr + str(item) + '\n'
458 excStr = excStr + str(item) + '\n'
360
459
361 return excStr
460 return excStr
362
461
363 def __str__(self):
462 def __str__(self):
364
463
365 return(self.getExceptionStr())
464 return(self.getExceptionStr())
366
465
367
466
368 def getExceptionHtml(self):
467 def getExceptionHtml(self):
369 """ getExceptionHtml returns an Html formatted string completely describing the exception.
468 """ getExceptionHtml returns an Html formatted string completely describing the exception.
370
469
371 Inputs: None
470 Inputs: None
372
471
373 Returns: A formatted string ready for printing completely describing the exception.
472 Returns: A formatted string ready for printing completely describing the exception.
374
473
375 Affects: None
474 Affects: None
376
475
377 Exceptions: None.
476 Exceptions: None.
378 """
477 """
379
478
380 excStr = '<BR>The following Schain Python exception has occurred:\n<BR>'
479 excStr = '<BR>The following Schain Python exception has occurred:\n<BR>'
381 excStr = excStr + self._strInterp + '\n<BR>\n'
480 excStr = excStr + self._strInterp + '\n<BR>\n'
382
481
383 if self._strExcList != None:
482 if self._strExcList != None:
384 for item in self._strExcList:
483 for item in self._strExcList:
385 excStr = excStr + str(item) + '\n<BR>'
484 excStr = excStr + str(item) + '\n<BR>'
386
485
387 return excStr
486 return excStr
388
487
488 class SchainWarning(Exception):
489 pass
490
491
389 if __name__ == '__main__':
492 if __name__ == '__main__':
390
493
391 test = SchainNotify()
494 test = SchainNotify()
392
495
393 test.sendAlert('This is a message from the python module SchainNotify', 'Test from SchainNotify')
496 test.sendAlert('This is a message from the python module SchainNotify', 'Test from SchainNotify')
394
497
395 print 'Hopefully message sent - check.'
498 print 'Hopefully message sent - check.'
@@ -1,1323 +1,1330
1 '''
1 '''
2 Created on September , 2012
2 Created on September , 2012
3 @author:
3 @author:
4 '''
4 '''
5
5
6 import sys
6 import sys
7 import ast
7 import ast
8 import datetime
8 import datetime
9 import traceback
9 import traceback
10 import math
10 import math
11 import time
11 import time
12 from multiprocessing import Process, cpu_count
12 from multiprocessing import Process, cpu_count
13
13
14 from xml.etree.ElementTree import ElementTree, Element, SubElement, tostring
14 from xml.etree.ElementTree import ElementTree, Element, SubElement, tostring
15 from xml.dom import minidom
15 from xml.dom import minidom
16
16
17 import schainpy
17 import schainpy
18 import schainpy.admin
18 from schainpy.admin import Alarm, SchainWarning
19 from schainpy.model import *
19 from schainpy.model import *
20 from schainpy.utils import log
20 from schainpy.utils import log
21
21
22 DTYPES = {
22 DTYPES = {
23 'Voltage': '.r',
23 'Voltage': '.r',
24 'Spectra': '.pdata'
24 'Spectra': '.pdata'
25 }
25 }
26
26
27
27
28 def MPProject(project, n=cpu_count()):
28 def MPProject(project, n=cpu_count()):
29 '''
29 '''
30 Project wrapper to run schain in n processes
30 Project wrapper to run schain in n processes
31 '''
31 '''
32
32
33 rconf = project.getReadUnitObj()
33 rconf = project.getReadUnitObj()
34 op = rconf.getOperationObj('run')
34 op = rconf.getOperationObj('run')
35 dt1 = op.getParameterValue('startDate')
35 dt1 = op.getParameterValue('startDate')
36 dt2 = op.getParameterValue('endDate')
36 dt2 = op.getParameterValue('endDate')
37 tm1 = op.getParameterValue('startTime')
37 tm1 = op.getParameterValue('startTime')
38 tm2 = op.getParameterValue('endTime')
38 tm2 = op.getParameterValue('endTime')
39 days = (dt2 - dt1).days
39 days = (dt2 - dt1).days
40
40
41 for day in range(days + 1):
41 for day in range(days + 1):
42 skip = 0
42 skip = 0
43 cursor = 0
43 cursor = 0
44 processes = []
44 processes = []
45 dt = dt1 + datetime.timedelta(day)
45 dt = dt1 + datetime.timedelta(day)
46 dt_str = dt.strftime('%Y/%m/%d')
46 dt_str = dt.strftime('%Y/%m/%d')
47 reader = JRODataReader()
47 reader = JRODataReader()
48 paths, files = reader.searchFilesOffLine(path=rconf.path,
48 paths, files = reader.searchFilesOffLine(path=rconf.path,
49 startDate=dt,
49 startDate=dt,
50 endDate=dt,
50 endDate=dt,
51 startTime=tm1,
51 startTime=tm1,
52 endTime=tm2,
52 endTime=tm2,
53 ext=DTYPES[rconf.datatype])
53 ext=DTYPES[rconf.datatype])
54 nFiles = len(files)
54 nFiles = len(files)
55 if nFiles == 0:
55 if nFiles == 0:
56 continue
56 continue
57 skip = int(math.ceil(nFiles / n))
57 skip = int(math.ceil(nFiles / n))
58 while nFiles > cursor * skip:
58 while nFiles > cursor * skip:
59 rconf.update(startDate=dt_str, endDate=dt_str, cursor=cursor,
59 rconf.update(startDate=dt_str, endDate=dt_str, cursor=cursor,
60 skip=skip)
60 skip=skip)
61 p = project.clone()
61 p = project.clone()
62 p.start()
62 p.start()
63 processes.append(p)
63 processes.append(p)
64 cursor += 1
64 cursor += 1
65
65
66 def beforeExit(exctype, value, trace):
66 def beforeExit(exctype, value, trace):
67 for process in processes:
67 for process in processes:
68 process.terminate()
68 process.terminate()
69 process.join()
69 process.join()
70 print traceback.print_tb(trace)
70 print traceback.print_tb(trace)
71
71
72 sys.excepthook = beforeExit
72 sys.excepthook = beforeExit
73
73
74 for process in processes:
74 for process in processes:
75 process.join()
75 process.join()
76 process.terminate()
76 process.terminate()
77
77
78 time.sleep(3)
78 time.sleep(3)
79
79
80
80
81 class ParameterConf():
81 class ParameterConf():
82
82
83 id = None
83 id = None
84 name = None
84 name = None
85 value = None
85 value = None
86 format = None
86 format = None
87
87
88 __formated_value = None
88 __formated_value = None
89
89
90 ELEMENTNAME = 'Parameter'
90 ELEMENTNAME = 'Parameter'
91
91
92 def __init__(self):
92 def __init__(self):
93
93
94 self.format = 'str'
94 self.format = 'str'
95
95
96 def getElementName(self):
96 def getElementName(self):
97
97
98 return self.ELEMENTNAME
98 return self.ELEMENTNAME
99
99
100 def getValue(self):
100 def getValue(self):
101
101
102 value = self.value
102 value = self.value
103 format = self.format
103 format = self.format
104
104
105 if self.__formated_value != None:
105 if self.__formated_value != None:
106
106
107 return self.__formated_value
107 return self.__formated_value
108
108
109 if format == 'obj':
109 if format == 'obj':
110 return value
110 return value
111
111
112 if format == 'str':
112 if format == 'str':
113 self.__formated_value = str(value)
113 self.__formated_value = str(value)
114 return self.__formated_value
114 return self.__formated_value
115
115
116 if value == '':
116 if value == '':
117 raise ValueError, '%s: This parameter value is empty' % self.name
117 raise ValueError, '%s: This parameter value is empty' % self.name
118
118
119 if format == 'list':
119 if format == 'list':
120 strList = value.split(',')
120 strList = value.split(',')
121
121
122 self.__formated_value = strList
122 self.__formated_value = strList
123
123
124 return self.__formated_value
124 return self.__formated_value
125
125
126 if format == 'intlist':
126 if format == 'intlist':
127 '''
127 '''
128 Example:
128 Example:
129 value = (0,1,2)
129 value = (0,1,2)
130 '''
130 '''
131
131
132 new_value = ast.literal_eval(value)
132 new_value = ast.literal_eval(value)
133
133
134 if type(new_value) not in (tuple, list):
134 if type(new_value) not in (tuple, list):
135 new_value = [int(new_value)]
135 new_value = [int(new_value)]
136
136
137 self.__formated_value = new_value
137 self.__formated_value = new_value
138
138
139 return self.__formated_value
139 return self.__formated_value
140
140
141 if format == 'floatlist':
141 if format == 'floatlist':
142 '''
142 '''
143 Example:
143 Example:
144 value = (0.5, 1.4, 2.7)
144 value = (0.5, 1.4, 2.7)
145 '''
145 '''
146
146
147 new_value = ast.literal_eval(value)
147 new_value = ast.literal_eval(value)
148
148
149 if type(new_value) not in (tuple, list):
149 if type(new_value) not in (tuple, list):
150 new_value = [float(new_value)]
150 new_value = [float(new_value)]
151
151
152 self.__formated_value = new_value
152 self.__formated_value = new_value
153
153
154 return self.__formated_value
154 return self.__formated_value
155
155
156 if format == 'date':
156 if format == 'date':
157 strList = value.split('/')
157 strList = value.split('/')
158 intList = [int(x) for x in strList]
158 intList = [int(x) for x in strList]
159 date = datetime.date(intList[0], intList[1], intList[2])
159 date = datetime.date(intList[0], intList[1], intList[2])
160
160
161 self.__formated_value = date
161 self.__formated_value = date
162
162
163 return self.__formated_value
163 return self.__formated_value
164
164
165 if format == 'time':
165 if format == 'time':
166 strList = value.split(':')
166 strList = value.split(':')
167 intList = [int(x) for x in strList]
167 intList = [int(x) for x in strList]
168 time = datetime.time(intList[0], intList[1], intList[2])
168 time = datetime.time(intList[0], intList[1], intList[2])
169
169
170 self.__formated_value = time
170 self.__formated_value = time
171
171
172 return self.__formated_value
172 return self.__formated_value
173
173
174 if format == 'pairslist':
174 if format == 'pairslist':
175 '''
175 '''
176 Example:
176 Example:
177 value = (0,1),(1,2)
177 value = (0,1),(1,2)
178 '''
178 '''
179
179
180 new_value = ast.literal_eval(value)
180 new_value = ast.literal_eval(value)
181
181
182 if type(new_value) not in (tuple, list):
182 if type(new_value) not in (tuple, list):
183 raise ValueError, '%s has to be a tuple or list of pairs' % value
183 raise ValueError, '%s has to be a tuple or list of pairs' % value
184
184
185 if type(new_value[0]) not in (tuple, list):
185 if type(new_value[0]) not in (tuple, list):
186 if len(new_value) != 2:
186 if len(new_value) != 2:
187 raise ValueError, '%s has to be a tuple or list of pairs' % value
187 raise ValueError, '%s has to be a tuple or list of pairs' % value
188 new_value = [new_value]
188 new_value = [new_value]
189
189
190 for thisPair in new_value:
190 for thisPair in new_value:
191 if len(thisPair) != 2:
191 if len(thisPair) != 2:
192 raise ValueError, '%s has to be a tuple or list of pairs' % value
192 raise ValueError, '%s has to be a tuple or list of pairs' % value
193
193
194 self.__formated_value = new_value
194 self.__formated_value = new_value
195
195
196 return self.__formated_value
196 return self.__formated_value
197
197
198 if format == 'multilist':
198 if format == 'multilist':
199 '''
199 '''
200 Example:
200 Example:
201 value = (0,1,2),(3,4,5)
201 value = (0,1,2),(3,4,5)
202 '''
202 '''
203 multiList = ast.literal_eval(value)
203 multiList = ast.literal_eval(value)
204
204
205 if type(multiList[0]) == int:
205 if type(multiList[0]) == int:
206 multiList = ast.literal_eval('(' + value + ')')
206 multiList = ast.literal_eval('(' + value + ')')
207
207
208 self.__formated_value = multiList
208 self.__formated_value = multiList
209
209
210 return self.__formated_value
210 return self.__formated_value
211
211
212 if format == 'bool':
212 if format == 'bool':
213 value = int(value)
213 value = int(value)
214
214
215 if format == 'int':
215 if format == 'int':
216 value = float(value)
216 value = float(value)
217
217
218 format_func = eval(format)
218 format_func = eval(format)
219
219
220 self.__formated_value = format_func(value)
220 self.__formated_value = format_func(value)
221
221
222 return self.__formated_value
222 return self.__formated_value
223
223
224 def updateId(self, new_id):
224 def updateId(self, new_id):
225
225
226 self.id = str(new_id)
226 self.id = str(new_id)
227
227
228 def setup(self, id, name, value, format='str'):
228 def setup(self, id, name, value, format='str'):
229 self.id = str(id)
229 self.id = str(id)
230 self.name = name
230 self.name = name
231 if format == 'obj':
231 if format == 'obj':
232 self.value = value
232 self.value = value
233 else:
233 else:
234 self.value = str(value)
234 self.value = str(value)
235 self.format = str.lower(format)
235 self.format = str.lower(format)
236
236
237 self.getValue()
237 self.getValue()
238
238
239 return 1
239 return 1
240
240
241 def update(self, name, value, format='str'):
241 def update(self, name, value, format='str'):
242
242
243 self.name = name
243 self.name = name
244 self.value = str(value)
244 self.value = str(value)
245 self.format = format
245 self.format = format
246
246
247 def makeXml(self, opElement):
247 def makeXml(self, opElement):
248 if self.name not in ('queue',):
248 if self.name not in ('queue',):
249 parmElement = SubElement(opElement, self.ELEMENTNAME)
249 parmElement = SubElement(opElement, self.ELEMENTNAME)
250 parmElement.set('id', str(self.id))
250 parmElement.set('id', str(self.id))
251 parmElement.set('name', self.name)
251 parmElement.set('name', self.name)
252 parmElement.set('value', self.value)
252 parmElement.set('value', self.value)
253 parmElement.set('format', self.format)
253 parmElement.set('format', self.format)
254
254
255 def readXml(self, parmElement):
255 def readXml(self, parmElement):
256
256
257 self.id = parmElement.get('id')
257 self.id = parmElement.get('id')
258 self.name = parmElement.get('name')
258 self.name = parmElement.get('name')
259 self.value = parmElement.get('value')
259 self.value = parmElement.get('value')
260 self.format = str.lower(parmElement.get('format'))
260 self.format = str.lower(parmElement.get('format'))
261
261
262 # Compatible with old signal chain version
262 # Compatible with old signal chain version
263 if self.format == 'int' and self.name == 'idfigure':
263 if self.format == 'int' and self.name == 'idfigure':
264 self.name = 'id'
264 self.name = 'id'
265
265
266 def printattr(self):
266 def printattr(self):
267
267
268 print 'Parameter[%s]: name = %s, value = %s, format = %s' % (self.id, self.name, self.value, self.format)
268 print 'Parameter[%s]: name = %s, value = %s, format = %s' % (self.id, self.name, self.value, self.format)
269
269
270
270
271 class OperationConf():
271 class OperationConf():
272
272
273 id = None
273 id = None
274 name = None
274 name = None
275 priority = None
275 priority = None
276 type = None
276 type = None
277
277
278 parmConfObjList = []
278 parmConfObjList = []
279
279
280 ELEMENTNAME = 'Operation'
280 ELEMENTNAME = 'Operation'
281
281
282 def __init__(self):
282 def __init__(self):
283
283
284 self.id = '0'
284 self.id = '0'
285 self.name = None
285 self.name = None
286 self.priority = None
286 self.priority = None
287 self.type = 'self'
287 self.type = 'self'
288
288
289 def __getNewId(self):
289 def __getNewId(self):
290
290
291 return int(self.id) * 10 + len(self.parmConfObjList) + 1
291 return int(self.id) * 10 + len(self.parmConfObjList) + 1
292
292
293 def updateId(self, new_id):
293 def updateId(self, new_id):
294
294
295 self.id = str(new_id)
295 self.id = str(new_id)
296
296
297 n = 1
297 n = 1
298 for parmObj in self.parmConfObjList:
298 for parmObj in self.parmConfObjList:
299
299
300 idParm = str(int(new_id) * 10 + n)
300 idParm = str(int(new_id) * 10 + n)
301 parmObj.updateId(idParm)
301 parmObj.updateId(idParm)
302
302
303 n += 1
303 n += 1
304
304
305 def getElementName(self):
305 def getElementName(self):
306
306
307 return self.ELEMENTNAME
307 return self.ELEMENTNAME
308
308
309 def getParameterObjList(self):
309 def getParameterObjList(self):
310
310
311 return self.parmConfObjList
311 return self.parmConfObjList
312
312
313 def getParameterObj(self, parameterName):
313 def getParameterObj(self, parameterName):
314
314
315 for parmConfObj in self.parmConfObjList:
315 for parmConfObj in self.parmConfObjList:
316
316
317 if parmConfObj.name != parameterName:
317 if parmConfObj.name != parameterName:
318 continue
318 continue
319
319
320 return parmConfObj
320 return parmConfObj
321
321
322 return None
322 return None
323
323
324 def getParameterObjfromValue(self, parameterValue):
324 def getParameterObjfromValue(self, parameterValue):
325
325
326 for parmConfObj in self.parmConfObjList:
326 for parmConfObj in self.parmConfObjList:
327
327
328 if parmConfObj.getValue() != parameterValue:
328 if parmConfObj.getValue() != parameterValue:
329 continue
329 continue
330
330
331 return parmConfObj.getValue()
331 return parmConfObj.getValue()
332
332
333 return None
333 return None
334
334
335 def getParameterValue(self, parameterName):
335 def getParameterValue(self, parameterName):
336
336
337 parameterObj = self.getParameterObj(parameterName)
337 parameterObj = self.getParameterObj(parameterName)
338
338
339 # if not parameterObj:
339 # if not parameterObj:
340 # return None
340 # return None
341
341
342 value = parameterObj.getValue()
342 value = parameterObj.getValue()
343
343
344 return value
344 return value
345
345
346 def getKwargs(self):
346 def getKwargs(self):
347
347
348 kwargs = {}
348 kwargs = {}
349
349
350 for parmConfObj in self.parmConfObjList:
350 for parmConfObj in self.parmConfObjList:
351 if self.name == 'run' and parmConfObj.name == 'datatype':
351 if self.name == 'run' and parmConfObj.name == 'datatype':
352 continue
352 continue
353
353
354 kwargs[parmConfObj.name] = parmConfObj.getValue()
354 kwargs[parmConfObj.name] = parmConfObj.getValue()
355
355
356 return kwargs
356 return kwargs
357
357
358 def setup(self, id, name, priority, type):
358 def setup(self, id, name, priority, type):
359
359
360 self.id = str(id)
360 self.id = str(id)
361 self.name = name
361 self.name = name
362 self.type = type
362 self.type = type
363 self.priority = priority
363 self.priority = priority
364
364
365 self.parmConfObjList = []
365 self.parmConfObjList = []
366
366
367 def removeParameters(self):
367 def removeParameters(self):
368
368
369 for obj in self.parmConfObjList:
369 for obj in self.parmConfObjList:
370 del obj
370 del obj
371
371
372 self.parmConfObjList = []
372 self.parmConfObjList = []
373
373
374 def addParameter(self, name, value, format='str'):
374 def addParameter(self, name, value, format='str'):
375
375
376 if value is None:
376 if value is None:
377 return None
377 return None
378 id = self.__getNewId()
378 id = self.__getNewId()
379
379
380 parmConfObj = ParameterConf()
380 parmConfObj = ParameterConf()
381 if not parmConfObj.setup(id, name, value, format):
381 if not parmConfObj.setup(id, name, value, format):
382 return None
382 return None
383
383
384 self.parmConfObjList.append(parmConfObj)
384 self.parmConfObjList.append(parmConfObj)
385
385
386 return parmConfObj
386 return parmConfObj
387
387
388 def changeParameter(self, name, value, format='str'):
388 def changeParameter(self, name, value, format='str'):
389
389
390 parmConfObj = self.getParameterObj(name)
390 parmConfObj = self.getParameterObj(name)
391 parmConfObj.update(name, value, format)
391 parmConfObj.update(name, value, format)
392
392
393 return parmConfObj
393 return parmConfObj
394
394
395 def makeXml(self, procUnitElement):
395 def makeXml(self, procUnitElement):
396
396
397 opElement = SubElement(procUnitElement, self.ELEMENTNAME)
397 opElement = SubElement(procUnitElement, self.ELEMENTNAME)
398 opElement.set('id', str(self.id))
398 opElement.set('id', str(self.id))
399 opElement.set('name', self.name)
399 opElement.set('name', self.name)
400 opElement.set('type', self.type)
400 opElement.set('type', self.type)
401 opElement.set('priority', str(self.priority))
401 opElement.set('priority', str(self.priority))
402
402
403 for parmConfObj in self.parmConfObjList:
403 for parmConfObj in self.parmConfObjList:
404 parmConfObj.makeXml(opElement)
404 parmConfObj.makeXml(opElement)
405
405
406 def readXml(self, opElement):
406 def readXml(self, opElement):
407
407
408 self.id = opElement.get('id')
408 self.id = opElement.get('id')
409 self.name = opElement.get('name')
409 self.name = opElement.get('name')
410 self.type = opElement.get('type')
410 self.type = opElement.get('type')
411 self.priority = opElement.get('priority')
411 self.priority = opElement.get('priority')
412
412
413 # Compatible with old signal chain version
413 # Compatible with old signal chain version
414 # Use of 'run' method instead 'init'
414 # Use of 'run' method instead 'init'
415 if self.type == 'self' and self.name == 'init':
415 if self.type == 'self' and self.name == 'init':
416 self.name = 'run'
416 self.name = 'run'
417
417
418 self.parmConfObjList = []
418 self.parmConfObjList = []
419
419
420 parmElementList = opElement.iter(ParameterConf().getElementName())
420 parmElementList = opElement.iter(ParameterConf().getElementName())
421
421
422 for parmElement in parmElementList:
422 for parmElement in parmElementList:
423 parmConfObj = ParameterConf()
423 parmConfObj = ParameterConf()
424 parmConfObj.readXml(parmElement)
424 parmConfObj.readXml(parmElement)
425
425
426 # Compatible with old signal chain version
426 # Compatible with old signal chain version
427 # If an 'plot' OPERATION is found, changes name operation by the value of its type PARAMETER
427 # If an 'plot' OPERATION is found, changes name operation by the value of its type PARAMETER
428 if self.type != 'self' and self.name == 'Plot':
428 if self.type != 'self' and self.name == 'Plot':
429 if parmConfObj.format == 'str' and parmConfObj.name == 'type':
429 if parmConfObj.format == 'str' and parmConfObj.name == 'type':
430 self.name = parmConfObj.value
430 self.name = parmConfObj.value
431 continue
431 continue
432
432
433 self.parmConfObjList.append(parmConfObj)
433 self.parmConfObjList.append(parmConfObj)
434
434
435 def printattr(self):
435 def printattr(self):
436
436
437 print '%s[%s]: name = %s, type = %s, priority = %s' % (self.ELEMENTNAME,
437 print '%s[%s]: name = %s, type = %s, priority = %s' % (self.ELEMENTNAME,
438 self.id,
438 self.id,
439 self.name,
439 self.name,
440 self.type,
440 self.type,
441 self.priority)
441 self.priority)
442
442
443 for parmConfObj in self.parmConfObjList:
443 for parmConfObj in self.parmConfObjList:
444 parmConfObj.printattr()
444 parmConfObj.printattr()
445
445
446 def createObject(self, plotter_queue=None):
446 def createObject(self, plotter_queue=None):
447
447
448 if self.type == 'self':
448 if self.type == 'self':
449 raise ValueError, 'This operation type cannot be created'
449 raise ValueError, 'This operation type cannot be created'
450
450
451 if self.type == 'plotter':
451 if self.type == 'plotter':
452 if not plotter_queue:
452 if not plotter_queue:
453 raise ValueError, 'plotter_queue is not defined. Use:\nmyProject = Project()\nmyProject.setPlotterQueue(plotter_queue)'
453 raise ValueError, 'plotter_queue is not defined. Use:\nmyProject = Project()\nmyProject.setPlotterQueue(plotter_queue)'
454
454
455 opObj = Plotter(self.name, plotter_queue)
455 opObj = Plotter(self.name, plotter_queue)
456
456
457 if self.type == 'external' or self.type == 'other':
457 if self.type == 'external' or self.type == 'other':
458
458
459 className = eval(self.name)
459 className = eval(self.name)
460 kwargs = self.getKwargs()
460 kwargs = self.getKwargs()
461
461
462 opObj = className(**kwargs)
462 opObj = className(**kwargs)
463
463
464 return opObj
464 return opObj
465
465
466
466
467 class ProcUnitConf():
467 class ProcUnitConf():
468
468
469 id = None
469 id = None
470 name = None
470 name = None
471 datatype = None
471 datatype = None
472 inputId = None
472 inputId = None
473 parentId = None
473 parentId = None
474
474
475 opConfObjList = []
475 opConfObjList = []
476
476
477 procUnitObj = None
477 procUnitObj = None
478 opObjList = []
478 opObjList = []
479
479
480 ELEMENTNAME = 'ProcUnit'
480 ELEMENTNAME = 'ProcUnit'
481
481
482 def __init__(self):
482 def __init__(self):
483
483
484 self.id = None
484 self.id = None
485 self.datatype = None
485 self.datatype = None
486 self.name = None
486 self.name = None
487 self.inputId = None
487 self.inputId = None
488
488
489 self.opConfObjList = []
489 self.opConfObjList = []
490
490
491 self.procUnitObj = None
491 self.procUnitObj = None
492 self.opObjDict = {}
492 self.opObjDict = {}
493
493
494 def __getPriority(self):
494 def __getPriority(self):
495
495
496 return len(self.opConfObjList) + 1
496 return len(self.opConfObjList) + 1
497
497
498 def __getNewId(self):
498 def __getNewId(self):
499
499
500 return int(self.id) * 10 + len(self.opConfObjList) + 1
500 return int(self.id) * 10 + len(self.opConfObjList) + 1
501
501
502 def getElementName(self):
502 def getElementName(self):
503
503
504 return self.ELEMENTNAME
504 return self.ELEMENTNAME
505
505
506 def getId(self):
506 def getId(self):
507
507
508 return self.id
508 return self.id
509
509
510 def updateId(self, new_id, parentId=parentId):
510 def updateId(self, new_id, parentId=parentId):
511
511
512 new_id = int(parentId) * 10 + (int(self.id) % 10)
512 new_id = int(parentId) * 10 + (int(self.id) % 10)
513 new_inputId = int(parentId) * 10 + (int(self.inputId) % 10)
513 new_inputId = int(parentId) * 10 + (int(self.inputId) % 10)
514
514
515 # If this proc unit has not inputs
515 # If this proc unit has not inputs
516 if self.inputId == '0':
516 if self.inputId == '0':
517 new_inputId = 0
517 new_inputId = 0
518
518
519 n = 1
519 n = 1
520 for opConfObj in self.opConfObjList:
520 for opConfObj in self.opConfObjList:
521
521
522 idOp = str(int(new_id) * 10 + n)
522 idOp = str(int(new_id) * 10 + n)
523 opConfObj.updateId(idOp)
523 opConfObj.updateId(idOp)
524
524
525 n += 1
525 n += 1
526
526
527 self.parentId = str(parentId)
527 self.parentId = str(parentId)
528 self.id = str(new_id)
528 self.id = str(new_id)
529 self.inputId = str(new_inputId)
529 self.inputId = str(new_inputId)
530
530
531 def getInputId(self):
531 def getInputId(self):
532
532
533 return self.inputId
533 return self.inputId
534
534
535 def getOperationObjList(self):
535 def getOperationObjList(self):
536
536
537 return self.opConfObjList
537 return self.opConfObjList
538
538
539 def getOperationObj(self, name=None):
539 def getOperationObj(self, name=None):
540
540
541 for opConfObj in self.opConfObjList:
541 for opConfObj in self.opConfObjList:
542
542
543 if opConfObj.name != name:
543 if opConfObj.name != name:
544 continue
544 continue
545
545
546 return opConfObj
546 return opConfObj
547
547
548 return None
548 return None
549
549
550 def getOpObjfromParamValue(self, value=None):
550 def getOpObjfromParamValue(self, value=None):
551
551
552 for opConfObj in self.opConfObjList:
552 for opConfObj in self.opConfObjList:
553 if opConfObj.getParameterObjfromValue(parameterValue=value) != value:
553 if opConfObj.getParameterObjfromValue(parameterValue=value) != value:
554 continue
554 continue
555 return opConfObj
555 return opConfObj
556 return None
556 return None
557
557
558 def getProcUnitObj(self):
558 def getProcUnitObj(self):
559
559
560 return self.procUnitObj
560 return self.procUnitObj
561
561
562 def setup(self, id, name, datatype, inputId, parentId=None):
562 def setup(self, id, name, datatype, inputId, parentId=None):
563
563
564 # Compatible with old signal chain version
564 # Compatible with old signal chain version
565 if datatype == None and name == None:
565 if datatype == None and name == None:
566 raise ValueError, 'datatype or name should be defined'
566 raise ValueError, 'datatype or name should be defined'
567
567
568 if name == None:
568 if name == None:
569 if 'Proc' in datatype:
569 if 'Proc' in datatype:
570 name = datatype
570 name = datatype
571 else:
571 else:
572 name = '%sProc' % (datatype)
572 name = '%sProc' % (datatype)
573
573
574 if datatype == None:
574 if datatype == None:
575 datatype = name.replace('Proc', '')
575 datatype = name.replace('Proc', '')
576
576
577 self.id = str(id)
577 self.id = str(id)
578 self.name = name
578 self.name = name
579 self.datatype = datatype
579 self.datatype = datatype
580 self.inputId = inputId
580 self.inputId = inputId
581 self.parentId = parentId
581 self.parentId = parentId
582
582
583 self.opConfObjList = []
583 self.opConfObjList = []
584
584
585 self.addOperation(name='run', optype='self')
585 self.addOperation(name='run', optype='self')
586
586
587 def removeOperations(self):
587 def removeOperations(self):
588
588
589 for obj in self.opConfObjList:
589 for obj in self.opConfObjList:
590 del obj
590 del obj
591
591
592 self.opConfObjList = []
592 self.opConfObjList = []
593 self.addOperation(name='run')
593 self.addOperation(name='run')
594
594
595 def addParameter(self, **kwargs):
595 def addParameter(self, **kwargs):
596 '''
596 '''
597 Add parameters to 'run' operation
597 Add parameters to 'run' operation
598 '''
598 '''
599 opObj = self.opConfObjList[0]
599 opObj = self.opConfObjList[0]
600
600
601 opObj.addParameter(**kwargs)
601 opObj.addParameter(**kwargs)
602
602
603 return opObj
603 return opObj
604
604
605 def addOperation(self, name, optype='self'):
605 def addOperation(self, name, optype='self'):
606
606
607 id = self.__getNewId()
607 id = self.__getNewId()
608 priority = self.__getPriority()
608 priority = self.__getPriority()
609
609
610 opConfObj = OperationConf()
610 opConfObj = OperationConf()
611 opConfObj.setup(id, name=name, priority=priority, type=optype)
611 opConfObj.setup(id, name=name, priority=priority, type=optype)
612
612
613 self.opConfObjList.append(opConfObj)
613 self.opConfObjList.append(opConfObj)
614
614
615 return opConfObj
615 return opConfObj
616
616
617 def makeXml(self, projectElement):
617 def makeXml(self, projectElement):
618
618
619 procUnitElement = SubElement(projectElement, self.ELEMENTNAME)
619 procUnitElement = SubElement(projectElement, self.ELEMENTNAME)
620 procUnitElement.set('id', str(self.id))
620 procUnitElement.set('id', str(self.id))
621 procUnitElement.set('name', self.name)
621 procUnitElement.set('name', self.name)
622 procUnitElement.set('datatype', self.datatype)
622 procUnitElement.set('datatype', self.datatype)
623 procUnitElement.set('inputId', str(self.inputId))
623 procUnitElement.set('inputId', str(self.inputId))
624
624
625 for opConfObj in self.opConfObjList:
625 for opConfObj in self.opConfObjList:
626 opConfObj.makeXml(procUnitElement)
626 opConfObj.makeXml(procUnitElement)
627
627
628 def readXml(self, upElement):
628 def readXml(self, upElement):
629
629
630 self.id = upElement.get('id')
630 self.id = upElement.get('id')
631 self.name = upElement.get('name')
631 self.name = upElement.get('name')
632 self.datatype = upElement.get('datatype')
632 self.datatype = upElement.get('datatype')
633 self.inputId = upElement.get('inputId')
633 self.inputId = upElement.get('inputId')
634
634
635 if self.ELEMENTNAME == 'ReadUnit':
635 if self.ELEMENTNAME == 'ReadUnit':
636 self.datatype = self.datatype.replace('Reader', '')
636 self.datatype = self.datatype.replace('Reader', '')
637
637
638 if self.ELEMENTNAME == 'ProcUnit':
638 if self.ELEMENTNAME == 'ProcUnit':
639 self.datatype = self.datatype.replace('Proc', '')
639 self.datatype = self.datatype.replace('Proc', '')
640
640
641 if self.inputId == 'None':
641 if self.inputId == 'None':
642 self.inputId = '0'
642 self.inputId = '0'
643
643
644 self.opConfObjList = []
644 self.opConfObjList = []
645
645
646 opElementList = upElement.iter(OperationConf().getElementName())
646 opElementList = upElement.iter(OperationConf().getElementName())
647
647
648 for opElement in opElementList:
648 for opElement in opElementList:
649 opConfObj = OperationConf()
649 opConfObj = OperationConf()
650 opConfObj.readXml(opElement)
650 opConfObj.readXml(opElement)
651 self.opConfObjList.append(opConfObj)
651 self.opConfObjList.append(opConfObj)
652
652
653 def printattr(self):
653 def printattr(self):
654
654
655 print '%s[%s]: name = %s, datatype = %s, inputId = %s' % (self.ELEMENTNAME,
655 print '%s[%s]: name = %s, datatype = %s, inputId = %s' % (self.ELEMENTNAME,
656 self.id,
656 self.id,
657 self.name,
657 self.name,
658 self.datatype,
658 self.datatype,
659 self.inputId)
659 self.inputId)
660
660
661 for opConfObj in self.opConfObjList:
661 for opConfObj in self.opConfObjList:
662 opConfObj.printattr()
662 opConfObj.printattr()
663
663
664 def getKwargs(self):
664 def getKwargs(self):
665
665
666 opObj = self.opConfObjList[0]
666 opObj = self.opConfObjList[0]
667 kwargs = opObj.getKwargs()
667 kwargs = opObj.getKwargs()
668
668
669 return kwargs
669 return kwargs
670
670
671 def createObjects(self, plotter_queue=None):
671 def createObjects(self, plotter_queue=None):
672
672
673 className = eval(self.name)
673 className = eval(self.name)
674 kwargs = self.getKwargs()
674 kwargs = self.getKwargs()
675 procUnitObj = className(**kwargs)
675 procUnitObj = className(**kwargs)
676
676
677 for opConfObj in self.opConfObjList:
677 for opConfObj in self.opConfObjList:
678
678
679 if opConfObj.type == 'self' and self.name == 'run':
679 if opConfObj.type == 'self' and self.name == 'run':
680 continue
680 continue
681 elif opConfObj.type == 'self':
681 elif opConfObj.type == 'self':
682 procUnitObj.addOperationKwargs(
682 procUnitObj.addOperationKwargs(
683 opConfObj.id, **opConfObj.getKwargs())
683 opConfObj.id, **opConfObj.getKwargs())
684 continue
684 continue
685
685
686 opObj = opConfObj.createObject(plotter_queue)
686 opObj = opConfObj.createObject(plotter_queue)
687
687
688 self.opObjDict[opConfObj.id] = opObj
688 self.opObjDict[opConfObj.id] = opObj
689
689
690 procUnitObj.addOperation(opObj, opConfObj.id)
690 procUnitObj.addOperation(opObj, opConfObj.id)
691
691
692 self.procUnitObj = procUnitObj
692 self.procUnitObj = procUnitObj
693
693
694 return procUnitObj
694 return procUnitObj
695
695
696 def run(self):
696 def run(self):
697
697
698 is_ok = False
698 is_ok = False
699
699
700 for opConfObj in self.opConfObjList:
700 for opConfObj in self.opConfObjList:
701
701
702 kwargs = {}
702 kwargs = {}
703 for parmConfObj in opConfObj.getParameterObjList():
703 for parmConfObj in opConfObj.getParameterObjList():
704 if opConfObj.name == 'run' and parmConfObj.name == 'datatype':
704 if opConfObj.name == 'run' and parmConfObj.name == 'datatype':
705 continue
705 continue
706
706
707 kwargs[parmConfObj.name] = parmConfObj.getValue()
707 kwargs[parmConfObj.name] = parmConfObj.getValue()
708
708
709 sts = self.procUnitObj.call(opType=opConfObj.type,
709 sts = self.procUnitObj.call(opType=opConfObj.type,
710 opName=opConfObj.name,
710 opName=opConfObj.name,
711 opId=opConfObj.id)
711 opId=opConfObj.id)
712
712
713 is_ok = is_ok or sts
713 is_ok = is_ok or sts
714
714
715 return is_ok
715 return is_ok
716
716
717 def close(self):
717 def close(self):
718
718
719 for opConfObj in self.opConfObjList:
719 for opConfObj in self.opConfObjList:
720 if opConfObj.type == 'self':
720 if opConfObj.type == 'self':
721 continue
721 continue
722
722
723 opObj = self.procUnitObj.getOperationObj(opConfObj.id)
723 opObj = self.procUnitObj.getOperationObj(opConfObj.id)
724 opObj.close()
724 opObj.close()
725
725
726 self.procUnitObj.close()
726 self.procUnitObj.close()
727
727
728 return
728 return
729
729
730
730
731 class ReadUnitConf(ProcUnitConf):
731 class ReadUnitConf(ProcUnitConf):
732
732
733 path = None
733 path = None
734 startDate = None
734 startDate = None
735 endDate = None
735 endDate = None
736 startTime = None
736 startTime = None
737 endTime = None
737 endTime = None
738
738
739 ELEMENTNAME = 'ReadUnit'
739 ELEMENTNAME = 'ReadUnit'
740
740
741 def __init__(self):
741 def __init__(self):
742
742
743 self.id = None
743 self.id = None
744 self.datatype = None
744 self.datatype = None
745 self.name = None
745 self.name = None
746 self.inputId = None
746 self.inputId = None
747
747
748 self.parentId = None
748 self.parentId = None
749
749
750 self.opConfObjList = []
750 self.opConfObjList = []
751 self.opObjList = []
751 self.opObjList = []
752
752
753 def getElementName(self):
753 def getElementName(self):
754
754
755 return self.ELEMENTNAME
755 return self.ELEMENTNAME
756
756
757 def setup(self, id, name, datatype, path='', startDate='', endDate='',
757 def setup(self, id, name, datatype, path='', startDate='', endDate='',
758 startTime='', endTime='', parentId=None, server=None, **kwargs):
758 startTime='', endTime='', parentId=None, server=None, **kwargs):
759
759
760 # Compatible with old signal chain version
760 # Compatible with old signal chain version
761 if datatype == None and name == None:
761 if datatype == None and name == None:
762 raise ValueError, 'datatype or name should be defined'
762 raise ValueError, 'datatype or name should be defined'
763 if name == None:
763 if name == None:
764 if 'Reader' in datatype:
764 if 'Reader' in datatype:
765 name = datatype
765 name = datatype
766 datatype = name.replace('Reader','')
766 datatype = name.replace('Reader','')
767 else:
767 else:
768 name = '{}Reader'.format(datatype)
768 name = '{}Reader'.format(datatype)
769 if datatype == None:
769 if datatype == None:
770 if 'Reader' in name:
770 if 'Reader' in name:
771 datatype = name.replace('Reader','')
771 datatype = name.replace('Reader','')
772 else:
772 else:
773 datatype = name
773 datatype = name
774 name = '{}Reader'.format(name)
774 name = '{}Reader'.format(name)
775
775
776 self.id = id
776 self.id = id
777 self.name = name
777 self.name = name
778 self.datatype = datatype
778 self.datatype = datatype
779 if path != '':
779 if path != '':
780 self.path = os.path.abspath(path)
780 self.path = os.path.abspath(path)
781 self.startDate = startDate
781 self.startDate = startDate
782 self.endDate = endDate
782 self.endDate = endDate
783 self.startTime = startTime
783 self.startTime = startTime
784 self.endTime = endTime
784 self.endTime = endTime
785 self.inputId = '0'
785 self.inputId = '0'
786 self.parentId = parentId
786 self.parentId = parentId
787 self.server = server
787 self.server = server
788 self.addRunOperation(**kwargs)
788 self.addRunOperation(**kwargs)
789
789
790 def update(self, **kwargs):
790 def update(self, **kwargs):
791
791
792 if 'datatype' in kwargs:
792 if 'datatype' in kwargs:
793 datatype = kwargs.pop('datatype')
793 datatype = kwargs.pop('datatype')
794 if 'Reader' in datatype:
794 if 'Reader' in datatype:
795 self.name = datatype
795 self.name = datatype
796 else:
796 else:
797 self.name = '%sReader' % (datatype)
797 self.name = '%sReader' % (datatype)
798 self.datatype = self.name.replace('Reader', '')
798 self.datatype = self.name.replace('Reader', '')
799
799
800 attrs = ('path', 'startDate', 'endDate',
800 attrs = ('path', 'startDate', 'endDate',
801 'startTime', 'endTime', 'parentId')
801 'startTime', 'endTime', 'parentId')
802
802
803 for attr in attrs:
803 for attr in attrs:
804 if attr in kwargs:
804 if attr in kwargs:
805 setattr(self, attr, kwargs.pop(attr))
805 setattr(self, attr, kwargs.pop(attr))
806
806
807 self.inputId = '0'
807 self.inputId = '0'
808 self.updateRunOperation(**kwargs)
808 self.updateRunOperation(**kwargs)
809
809
810 def removeOperations(self):
810 def removeOperations(self):
811
811
812 for obj in self.opConfObjList:
812 for obj in self.opConfObjList:
813 del obj
813 del obj
814
814
815 self.opConfObjList = []
815 self.opConfObjList = []
816
816
817 def addRunOperation(self, **kwargs):
817 def addRunOperation(self, **kwargs):
818
818
819 opObj = self.addOperation(name='run', optype='self')
819 opObj = self.addOperation(name='run', optype='self')
820
820
821 if self.server is None:
821 if self.server is None:
822 opObj.addParameter(
822 opObj.addParameter(
823 name='datatype', value=self.datatype, format='str')
823 name='datatype', value=self.datatype, format='str')
824 opObj.addParameter(name='path', value=self.path, format='str')
824 opObj.addParameter(name='path', value=self.path, format='str')
825 opObj.addParameter(
825 opObj.addParameter(
826 name='startDate', value=self.startDate, format='date')
826 name='startDate', value=self.startDate, format='date')
827 opObj.addParameter(
827 opObj.addParameter(
828 name='endDate', value=self.endDate, format='date')
828 name='endDate', value=self.endDate, format='date')
829 opObj.addParameter(
829 opObj.addParameter(
830 name='startTime', value=self.startTime, format='time')
830 name='startTime', value=self.startTime, format='time')
831 opObj.addParameter(
831 opObj.addParameter(
832 name='endTime', value=self.endTime, format='time')
832 name='endTime', value=self.endTime, format='time')
833
833
834 for key, value in kwargs.items():
834 for key, value in kwargs.items():
835 opObj.addParameter(name=key, value=value,
835 opObj.addParameter(name=key, value=value,
836 format=type(value).__name__)
836 format=type(value).__name__)
837 else:
837 else:
838 opObj.addParameter(name='server', value=self.server, format='str')
838 opObj.addParameter(name='server', value=self.server, format='str')
839
839
840 return opObj
840 return opObj
841
841
842 def updateRunOperation(self, **kwargs):
842 def updateRunOperation(self, **kwargs):
843
843
844 opObj = self.getOperationObj(name='run')
844 opObj = self.getOperationObj(name='run')
845 opObj.removeParameters()
845 opObj.removeParameters()
846
846
847 opObj.addParameter(name='datatype', value=self.datatype, format='str')
847 opObj.addParameter(name='datatype', value=self.datatype, format='str')
848 opObj.addParameter(name='path', value=self.path, format='str')
848 opObj.addParameter(name='path', value=self.path, format='str')
849 opObj.addParameter(
849 opObj.addParameter(
850 name='startDate', value=self.startDate, format='date')
850 name='startDate', value=self.startDate, format='date')
851 opObj.addParameter(name='endDate', value=self.endDate, format='date')
851 opObj.addParameter(name='endDate', value=self.endDate, format='date')
852 opObj.addParameter(
852 opObj.addParameter(
853 name='startTime', value=self.startTime, format='time')
853 name='startTime', value=self.startTime, format='time')
854 opObj.addParameter(name='endTime', value=self.endTime, format='time')
854 opObj.addParameter(name='endTime', value=self.endTime, format='time')
855
855
856 for key, value in kwargs.items():
856 for key, value in kwargs.items():
857 opObj.addParameter(name=key, value=value,
857 opObj.addParameter(name=key, value=value,
858 format=type(value).__name__)
858 format=type(value).__name__)
859
859
860 return opObj
860 return opObj
861
861
862 def readXml(self, upElement):
862 def readXml(self, upElement):
863
863
864 self.id = upElement.get('id')
864 self.id = upElement.get('id')
865 self.name = upElement.get('name')
865 self.name = upElement.get('name')
866 self.datatype = upElement.get('datatype')
866 self.datatype = upElement.get('datatype')
867 self.inputId = upElement.get('inputId')
867 self.inputId = upElement.get('inputId')
868
868
869 if self.ELEMENTNAME == 'ReadUnit':
869 if self.ELEMENTNAME == 'ReadUnit':
870 self.datatype = self.datatype.replace('Reader', '')
870 self.datatype = self.datatype.replace('Reader', '')
871
871
872 if self.inputId == 'None':
872 if self.inputId == 'None':
873 self.inputId = '0'
873 self.inputId = '0'
874
874
875 self.opConfObjList = []
875 self.opConfObjList = []
876
876
877 opElementList = upElement.iter(OperationConf().getElementName())
877 opElementList = upElement.iter(OperationConf().getElementName())
878
878
879 for opElement in opElementList:
879 for opElement in opElementList:
880 opConfObj = OperationConf()
880 opConfObj = OperationConf()
881 opConfObj.readXml(opElement)
881 opConfObj.readXml(opElement)
882 self.opConfObjList.append(opConfObj)
882 self.opConfObjList.append(opConfObj)
883
883
884 if opConfObj.name == 'run':
884 if opConfObj.name == 'run':
885 self.path = opConfObj.getParameterValue('path')
885 self.path = opConfObj.getParameterValue('path')
886 self.startDate = opConfObj.getParameterValue('startDate')
886 self.startDate = opConfObj.getParameterValue('startDate')
887 self.endDate = opConfObj.getParameterValue('endDate')
887 self.endDate = opConfObj.getParameterValue('endDate')
888 self.startTime = opConfObj.getParameterValue('startTime')
888 self.startTime = opConfObj.getParameterValue('startTime')
889 self.endTime = opConfObj.getParameterValue('endTime')
889 self.endTime = opConfObj.getParameterValue('endTime')
890
890
891
891
892 class Project(Process):
892 class Project(Process):
893
893
894 id = None
894 id = None
895 # name = None
895 # name = None
896 description = None
896 description = None
897 filename = None
897 filename = None
898
898
899 procUnitConfObjDict = None
899 procUnitConfObjDict = None
900
900
901 ELEMENTNAME = 'Project'
901 ELEMENTNAME = 'Project'
902
902
903 plotterQueue = None
903 plotterQueue = None
904
904
905 def __init__(self, plotter_queue=None):
905 def __init__(self, plotter_queue=None):
906
906
907 Process.__init__(self)
907 Process.__init__(self)
908 self.id = None
908 self.id = None
909 # self.name = None
910 self.description = None
909 self.description = None
911
910 self.email = None
911 self.alarm = [0]
912 self.plotterQueue = plotter_queue
912 self.plotterQueue = plotter_queue
913
914 self.procUnitConfObjDict = {}
913 self.procUnitConfObjDict = {}
915
914
916 def __getNewId(self):
915 def __getNewId(self):
917
916
918 idList = self.procUnitConfObjDict.keys()
917 idList = self.procUnitConfObjDict.keys()
919
918
920 id = int(self.id) * 10
919 id = int(self.id) * 10
921
920
922 while True:
921 while True:
923 id += 1
922 id += 1
924
923
925 if str(id) in idList:
924 if str(id) in idList:
926 continue
925 continue
927
926
928 break
927 break
929
928
930 return str(id)
929 return str(id)
931
930
932 def getElementName(self):
931 def getElementName(self):
933
932
934 return self.ELEMENTNAME
933 return self.ELEMENTNAME
935
934
936 def getId(self):
935 def getId(self):
937
936
938 return self.id
937 return self.id
939
938
940 def updateId(self, new_id):
939 def updateId(self, new_id):
941
940
942 self.id = str(new_id)
941 self.id = str(new_id)
943
942
944 keyList = self.procUnitConfObjDict.keys()
943 keyList = self.procUnitConfObjDict.keys()
945 keyList.sort()
944 keyList.sort()
946
945
947 n = 1
946 n = 1
948 newProcUnitConfObjDict = {}
947 newProcUnitConfObjDict = {}
949
948
950 for procKey in keyList:
949 for procKey in keyList:
951
950
952 procUnitConfObj = self.procUnitConfObjDict[procKey]
951 procUnitConfObj = self.procUnitConfObjDict[procKey]
953 idProcUnit = str(int(self.id) * 10 + n)
952 idProcUnit = str(int(self.id) * 10 + n)
954 procUnitConfObj.updateId(idProcUnit, parentId=self.id)
953 procUnitConfObj.updateId(idProcUnit, parentId=self.id)
955
956 newProcUnitConfObjDict[idProcUnit] = procUnitConfObj
954 newProcUnitConfObjDict[idProcUnit] = procUnitConfObj
957 n += 1
955 n += 1
958
956
959 self.procUnitConfObjDict = newProcUnitConfObjDict
957 self.procUnitConfObjDict = newProcUnitConfObjDict
960
958
961 def setup(self, id, name='', description=''):
959 def setup(self, id, name='', description='', email=None, alarm=[0]):
962
960
963 print
961 print
964 print '*' * 60
962 print '*' * 60
965 print ' Starting SIGNAL CHAIN PROCESSING v%s ' % schainpy.__version__
963 print ' Starting SIGNAL CHAIN PROCESSING v%s ' % schainpy.__version__
966 print '*' * 60
964 print '*' * 60
967 print
965 print
968 self.id = str(id)
966 self.id = str(id)
969 self.description = description
967 self.description = description
968 self.email = email
969 self.alarm = alarm
970
970
971 def update(self, name, description):
971 def update(self, **kwargs):
972
972
973 self.description = description
973 for key, value in kwargs:
974 setattr(self, key, value)
974
975
975 def clone(self):
976 def clone(self):
976
977
977 p = Project()
978 p = Project()
978 p.procUnitConfObjDict = self.procUnitConfObjDict
979 p.procUnitConfObjDict = self.procUnitConfObjDict
979 return p
980 return p
980
981
981 def addReadUnit(self, id=None, datatype=None, name=None, **kwargs):
982 def addReadUnit(self, id=None, datatype=None, name=None, **kwargs):
982
983
983 if id is None:
984 if id is None:
984 idReadUnit = self.__getNewId()
985 idReadUnit = self.__getNewId()
985 else:
986 else:
986 idReadUnit = str(id)
987 idReadUnit = str(id)
987
988
988 readUnitConfObj = ReadUnitConf()
989 readUnitConfObj = ReadUnitConf()
989 readUnitConfObj.setup(idReadUnit, name, datatype,
990 readUnitConfObj.setup(idReadUnit, name, datatype,
990 parentId=self.id, **kwargs)
991 parentId=self.id, **kwargs)
991
992
992 self.procUnitConfObjDict[readUnitConfObj.getId()] = readUnitConfObj
993 self.procUnitConfObjDict[readUnitConfObj.getId()] = readUnitConfObj
993
994
994 return readUnitConfObj
995 return readUnitConfObj
995
996
996 def addProcUnit(self, inputId='0', datatype=None, name=None):
997 def addProcUnit(self, inputId='0', datatype=None, name=None):
997
998
998 idProcUnit = self.__getNewId()
999 idProcUnit = self.__getNewId()
999
1000
1000 procUnitConfObj = ProcUnitConf()
1001 procUnitConfObj = ProcUnitConf()
1001 procUnitConfObj.setup(idProcUnit, name, datatype,
1002 procUnitConfObj.setup(idProcUnit, name, datatype,
1002 inputId, parentId=self.id)
1003 inputId, parentId=self.id)
1003
1004
1004 self.procUnitConfObjDict[procUnitConfObj.getId()] = procUnitConfObj
1005 self.procUnitConfObjDict[procUnitConfObj.getId()] = procUnitConfObj
1005
1006
1006 return procUnitConfObj
1007 return procUnitConfObj
1007
1008
1008 def removeProcUnit(self, id):
1009 def removeProcUnit(self, id):
1009
1010
1010 if id in self.procUnitConfObjDict.keys():
1011 if id in self.procUnitConfObjDict.keys():
1011 self.procUnitConfObjDict.pop(id)
1012 self.procUnitConfObjDict.pop(id)
1012
1013
1013 def getReadUnitId(self):
1014 def getReadUnitId(self):
1014
1015
1015 readUnitConfObj = self.getReadUnitObj()
1016 readUnitConfObj = self.getReadUnitObj()
1016
1017
1017 return readUnitConfObj.id
1018 return readUnitConfObj.id
1018
1019
1019 def getReadUnitObj(self):
1020 def getReadUnitObj(self):
1020
1021
1021 for obj in self.procUnitConfObjDict.values():
1022 for obj in self.procUnitConfObjDict.values():
1022 if obj.getElementName() == 'ReadUnit':
1023 if obj.getElementName() == 'ReadUnit':
1023 return obj
1024 return obj
1024
1025
1025 return None
1026 return None
1026
1027
1027 def getProcUnitObj(self, id=None, name=None):
1028 def getProcUnitObj(self, id=None, name=None):
1028
1029
1029 if id != None:
1030 if id != None:
1030 return self.procUnitConfObjDict[id]
1031 return self.procUnitConfObjDict[id]
1031
1032
1032 if name != None:
1033 if name != None:
1033 return self.getProcUnitObjByName(name)
1034 return self.getProcUnitObjByName(name)
1034
1035
1035 return None
1036 return None
1036
1037
1037 def getProcUnitObjByName(self, name):
1038 def getProcUnitObjByName(self, name):
1038
1039
1039 for obj in self.procUnitConfObjDict.values():
1040 for obj in self.procUnitConfObjDict.values():
1040 if obj.name == name:
1041 if obj.name == name:
1041 return obj
1042 return obj
1042
1043
1043 return None
1044 return None
1044
1045
1045 def procUnitItems(self):
1046 def procUnitItems(self):
1046
1047
1047 return self.procUnitConfObjDict.items()
1048 return self.procUnitConfObjDict.items()
1048
1049
1049 def makeXml(self):
1050 def makeXml(self):
1050
1051
1051 projectElement = Element('Project')
1052 projectElement = Element('Project')
1052 projectElement.set('id', str(self.id))
1053 projectElement.set('id', str(self.id))
1053 projectElement.set('name', self.name)
1054 projectElement.set('name', self.name)
1054 projectElement.set('description', self.description)
1055 projectElement.set('description', self.description)
1055
1056
1056 for procUnitConfObj in self.procUnitConfObjDict.values():
1057 for procUnitConfObj in self.procUnitConfObjDict.values():
1057 procUnitConfObj.makeXml(projectElement)
1058 procUnitConfObj.makeXml(projectElement)
1058
1059
1059 self.projectElement = projectElement
1060 self.projectElement = projectElement
1060
1061
1061 def writeXml(self, filename=None):
1062 def writeXml(self, filename=None):
1062
1063
1063 if filename == None:
1064 if filename == None:
1064 if self.filename:
1065 if self.filename:
1065 filename = self.filename
1066 filename = self.filename
1066 else:
1067 else:
1067 filename = 'schain.xml'
1068 filename = 'schain.xml'
1068
1069
1069 if not filename:
1070 if not filename:
1070 print 'filename has not been defined. Use setFilename(filename) for do it.'
1071 print 'filename has not been defined. Use setFilename(filename) for do it.'
1071 return 0
1072 return 0
1072
1073
1073 abs_file = os.path.abspath(filename)
1074 abs_file = os.path.abspath(filename)
1074
1075
1075 if not os.access(os.path.dirname(abs_file), os.W_OK):
1076 if not os.access(os.path.dirname(abs_file), os.W_OK):
1076 print 'No write permission on %s' % os.path.dirname(abs_file)
1077 print 'No write permission on %s' % os.path.dirname(abs_file)
1077 return 0
1078 return 0
1078
1079
1079 if os.path.isfile(abs_file) and not(os.access(abs_file, os.W_OK)):
1080 if os.path.isfile(abs_file) and not(os.access(abs_file, os.W_OK)):
1080 print 'File %s already exists and it could not be overwriten' % abs_file
1081 print 'File %s already exists and it could not be overwriten' % abs_file
1081 return 0
1082 return 0
1082
1083
1083 self.makeXml()
1084 self.makeXml()
1084
1085
1085 ElementTree(self.projectElement).write(abs_file, method='xml')
1086 ElementTree(self.projectElement).write(abs_file, method='xml')
1086
1087
1087 self.filename = abs_file
1088 self.filename = abs_file
1088
1089
1089 return 1
1090 return 1
1090
1091
1091 def readXml(self, filename=None):
1092 def readXml(self, filename=None):
1092
1093
1093 if not filename:
1094 if not filename:
1094 print 'filename is not defined'
1095 print 'filename is not defined'
1095 return 0
1096 return 0
1096
1097
1097 abs_file = os.path.abspath(filename)
1098 abs_file = os.path.abspath(filename)
1098
1099
1099 if not os.path.isfile(abs_file):
1100 if not os.path.isfile(abs_file):
1100 print '%s file does not exist' % abs_file
1101 print '%s file does not exist' % abs_file
1101 return 0
1102 return 0
1102
1103
1103 self.projectElement = None
1104 self.projectElement = None
1104 self.procUnitConfObjDict = {}
1105 self.procUnitConfObjDict = {}
1105
1106
1106 try:
1107 try:
1107 self.projectElement = ElementTree().parse(abs_file)
1108 self.projectElement = ElementTree().parse(abs_file)
1108 except:
1109 except:
1109 print 'Error reading %s, verify file format' % filename
1110 print 'Error reading %s, verify file format' % filename
1110 return 0
1111 return 0
1111
1112
1112 self.project = self.projectElement.tag
1113 self.project = self.projectElement.tag
1113
1114
1114 self.id = self.projectElement.get('id')
1115 self.id = self.projectElement.get('id')
1115 self.name = self.projectElement.get('name')
1116 self.name = self.projectElement.get('name')
1116 self.description = self.projectElement.get('description')
1117 self.description = self.projectElement.get('description')
1117
1118
1118 readUnitElementList = self.projectElement.iter(
1119 readUnitElementList = self.projectElement.iter(
1119 ReadUnitConf().getElementName())
1120 ReadUnitConf().getElementName())
1120
1121
1121 for readUnitElement in readUnitElementList:
1122 for readUnitElement in readUnitElementList:
1122 readUnitConfObj = ReadUnitConf()
1123 readUnitConfObj = ReadUnitConf()
1123 readUnitConfObj.readXml(readUnitElement)
1124 readUnitConfObj.readXml(readUnitElement)
1124
1125
1125 if readUnitConfObj.parentId == None:
1126 if readUnitConfObj.parentId == None:
1126 readUnitConfObj.parentId = self.id
1127 readUnitConfObj.parentId = self.id
1127
1128
1128 self.procUnitConfObjDict[readUnitConfObj.getId()] = readUnitConfObj
1129 self.procUnitConfObjDict[readUnitConfObj.getId()] = readUnitConfObj
1129
1130
1130 procUnitElementList = self.projectElement.iter(
1131 procUnitElementList = self.projectElement.iter(
1131 ProcUnitConf().getElementName())
1132 ProcUnitConf().getElementName())
1132
1133
1133 for procUnitElement in procUnitElementList:
1134 for procUnitElement in procUnitElementList:
1134 procUnitConfObj = ProcUnitConf()
1135 procUnitConfObj = ProcUnitConf()
1135 procUnitConfObj.readXml(procUnitElement)
1136 procUnitConfObj.readXml(procUnitElement)
1136
1137
1137 if procUnitConfObj.parentId == None:
1138 if procUnitConfObj.parentId == None:
1138 procUnitConfObj.parentId = self.id
1139 procUnitConfObj.parentId = self.id
1139
1140
1140 self.procUnitConfObjDict[procUnitConfObj.getId()] = procUnitConfObj
1141 self.procUnitConfObjDict[procUnitConfObj.getId()] = procUnitConfObj
1141
1142
1142 self.filename = abs_file
1143 self.filename = abs_file
1143
1144
1144 return 1
1145 return 1
1145
1146
1146 def printattr(self):
1147 def printattr(self):
1147
1148
1148 print 'Project[%s]: name = %s, description = %s' % (self.id,
1149 print 'Project[%s]: name = %s, description = %s' % (self.id,
1149 self.name,
1150 self.name,
1150 self.description)
1151 self.description)
1151
1152
1152 for procUnitConfObj in self.procUnitConfObjDict.values():
1153 for procUnitConfObj in self.procUnitConfObjDict.values():
1153 procUnitConfObj.printattr()
1154 procUnitConfObj.printattr()
1154
1155
1155 def createObjects(self):
1156 def createObjects(self):
1156
1157
1157 for procUnitConfObj in self.procUnitConfObjDict.values():
1158 for procUnitConfObj in self.procUnitConfObjDict.values():
1158 procUnitConfObj.createObjects(self.plotterQueue)
1159 procUnitConfObj.createObjects(self.plotterQueue)
1159
1160
1160 def __connect(self, objIN, thisObj):
1161 def __connect(self, objIN, thisObj):
1161
1162
1162 thisObj.setInput(objIN.getOutputObj())
1163 thisObj.setInput(objIN.getOutputObj())
1163
1164
1164 def connectObjects(self):
1165 def connectObjects(self):
1165
1166
1166 for thisPUConfObj in self.procUnitConfObjDict.values():
1167 for thisPUConfObj in self.procUnitConfObjDict.values():
1167
1168
1168 inputId = thisPUConfObj.getInputId()
1169 inputId = thisPUConfObj.getInputId()
1169
1170
1170 if int(inputId) == 0:
1171 if int(inputId) == 0:
1171 continue
1172 continue
1172
1173
1173 # Get input object
1174 # Get input object
1174 puConfINObj = self.procUnitConfObjDict[inputId]
1175 puConfINObj = self.procUnitConfObjDict[inputId]
1175 puObjIN = puConfINObj.getProcUnitObj()
1176 puObjIN = puConfINObj.getProcUnitObj()
1176
1177
1177 # Get current object
1178 # Get current object
1178 thisPUObj = thisPUConfObj.getProcUnitObj()
1179 thisPUObj = thisPUConfObj.getProcUnitObj()
1179
1180
1180 self.__connect(puObjIN, thisPUObj)
1181 self.__connect(puObjIN, thisPUObj)
1181
1182
1182 def __handleError(self, procUnitConfObj, send_email=False):
1183 def __handleError(self, procUnitConfObj, modes=None):
1183
1184
1184 import socket
1185 import socket
1185
1186
1187 if modes is None:
1188 modes = self.alarm
1189
1186 err = traceback.format_exception(sys.exc_info()[0],
1190 err = traceback.format_exception(sys.exc_info()[0],
1187 sys.exc_info()[1],
1191 sys.exc_info()[1],
1188 sys.exc_info()[2])
1192 sys.exc_info()[2])
1189
1193
1190 print '***** Error occurred in %s *****' % (procUnitConfObj.name)
1194 log.error('{}'.format(err[-1]), procUnitConfObj.name)
1191 print '***** %s' % err[-1]
1192
1195
1193 message = ''.join(err)
1196 message = ''.join(err)
1194
1197
1195 sys.stderr.write(message)
1198 sys.stderr.write(message)
1196
1199
1197 if not send_email:
1198 return
1199
1200 subject = 'SChain v%s: Error running %s\n' % (
1200 subject = 'SChain v%s: Error running %s\n' % (
1201 schainpy.__version__, procUnitConfObj.name)
1201 schainpy.__version__, procUnitConfObj.name)
1202
1202
1203 subtitle = '%s: %s\n' % (
1203 subtitle = '%s: %s\n' % (
1204 procUnitConfObj.getElementName(), procUnitConfObj.name)
1204 procUnitConfObj.getElementName(), procUnitConfObj.name)
1205 subtitle += 'Hostname: %s\n' % socket.gethostbyname(
1205 subtitle += 'Hostname: %s\n' % socket.gethostbyname(
1206 socket.gethostname())
1206 socket.gethostname())
1207 subtitle += 'Working directory: %s\n' % os.path.abspath('./')
1207 subtitle += 'Working directory: %s\n' % os.path.abspath('./')
1208 subtitle += 'Configuration file: %s\n' % self.filename
1208 subtitle += 'Configuration file: %s\n' % self.filename
1209 subtitle += 'Time: %s\n' % str(datetime.datetime.now())
1209 subtitle += 'Time: %s\n' % str(datetime.datetime.now())
1210
1210
1211 readUnitConfObj = self.getReadUnitObj()
1211 readUnitConfObj = self.getReadUnitObj()
1212 if readUnitConfObj:
1212 if readUnitConfObj:
1213 subtitle += '\nInput parameters:\n'
1213 subtitle += '\nInput parameters:\n'
1214 subtitle += '[Data path = %s]\n' % readUnitConfObj.path
1214 subtitle += '[Data path = %s]\n' % readUnitConfObj.path
1215 subtitle += '[Data type = %s]\n' % readUnitConfObj.datatype
1215 subtitle += '[Data type = %s]\n' % readUnitConfObj.datatype
1216 subtitle += '[Start date = %s]\n' % readUnitConfObj.startDate
1216 subtitle += '[Start date = %s]\n' % readUnitConfObj.startDate
1217 subtitle += '[End date = %s]\n' % readUnitConfObj.endDate
1217 subtitle += '[End date = %s]\n' % readUnitConfObj.endDate
1218 subtitle += '[Start time = %s]\n' % readUnitConfObj.startTime
1218 subtitle += '[Start time = %s]\n' % readUnitConfObj.startTime
1219 subtitle += '[End time = %s]\n' % readUnitConfObj.endTime
1219 subtitle += '[End time = %s]\n' % readUnitConfObj.endTime
1220
1220
1221 adminObj = schainpy.admin.SchainNotify()
1221 a = Alarm(
1222 adminObj.sendAlert(message=message,
1222 modes=modes,
1223 subject=subject,
1223 email=self.email,
1224 subtitle=subtitle,
1224 message=message,
1225 filename=self.filename)
1225 subject=subject,
1226 subtitle=subtitle,
1227 filename=self.filename
1228 )
1229
1230 a.start()
1226
1231
1227 def isPaused(self):
1232 def isPaused(self):
1228 return 0
1233 return 0
1229
1234
1230 def isStopped(self):
1235 def isStopped(self):
1231 return 0
1236 return 0
1232
1237
1233 def runController(self):
1238 def runController(self):
1234 '''
1239 '''
1235 returns 0 when this process has been stopped, 1 otherwise
1240 returns 0 when this process has been stopped, 1 otherwise
1236 '''
1241 '''
1237
1242
1238 if self.isPaused():
1243 if self.isPaused():
1239 print 'Process suspended'
1244 print 'Process suspended'
1240
1245
1241 while True:
1246 while True:
1242 time.sleep(0.1)
1247 time.sleep(0.1)
1243
1248
1244 if not self.isPaused():
1249 if not self.isPaused():
1245 break
1250 break
1246
1251
1247 if self.isStopped():
1252 if self.isStopped():
1248 break
1253 break
1249
1254
1250 print 'Process reinitialized'
1255 print 'Process reinitialized'
1251
1256
1252 if self.isStopped():
1257 if self.isStopped():
1253 print 'Process stopped'
1258 print 'Process stopped'
1254 return 0
1259 return 0
1255
1260
1256 return 1
1261 return 1
1257
1262
1258 def setFilename(self, filename):
1263 def setFilename(self, filename):
1259
1264
1260 self.filename = filename
1265 self.filename = filename
1261
1266
1262 def setPlotterQueue(self, plotter_queue):
1267 def setPlotterQueue(self, plotter_queue):
1263
1268
1264 raise NotImplementedError, 'Use schainpy.controller_api.ControllerThread instead Project class'
1269 raise NotImplementedError, 'Use schainpy.controller_api.ControllerThread instead Project class'
1265
1270
1266 def getPlotterQueue(self):
1271 def getPlotterQueue(self):
1267
1272
1268 raise NotImplementedError, 'Use schainpy.controller_api.ControllerThread instead Project class'
1273 raise NotImplementedError, 'Use schainpy.controller_api.ControllerThread instead Project class'
1269
1274
1270 def useExternalPlotter(self):
1275 def useExternalPlotter(self):
1271
1276
1272 raise NotImplementedError, 'Use schainpy.controller_api.ControllerThread instead Project class'
1277 raise NotImplementedError, 'Use schainpy.controller_api.ControllerThread instead Project class'
1273
1278
1274 def run(self):
1279 def run(self):
1275
1280
1276 log.success('Starting {}'.format(self.name))
1281 log.success('Starting {}'.format(self.name))
1277 self.start_time = time.time()
1282 self.start_time = time.time()
1278 self.createObjects()
1283 self.createObjects()
1279 self.connectObjects()
1284 self.connectObjects()
1280
1285
1281 keyList = self.procUnitConfObjDict.keys()
1286 keyList = self.procUnitConfObjDict.keys()
1282 keyList.sort()
1287 keyList.sort()
1283
1288
1284 while(True):
1289 while(True):
1285
1290
1286 is_ok = False
1291 is_ok = False
1287
1292
1288 for procKey in keyList:
1293 for procKey in keyList:
1289
1294
1290 procUnitConfObj = self.procUnitConfObjDict[procKey]
1295 procUnitConfObj = self.procUnitConfObjDict[procKey]
1291
1296
1292 try:
1297 try:
1293 sts = procUnitConfObj.run()
1298 sts = procUnitConfObj.run()
1294 is_ok = is_ok or sts
1299 is_ok = is_ok or sts
1300 except SchainWarning:
1301 self.__handleError(procUnitConfObj, modes=[2, 3])
1295 except KeyboardInterrupt:
1302 except KeyboardInterrupt:
1296 is_ok = False
1303 is_ok = False
1297 break
1304 break
1298 except ValueError, e:
1305 except ValueError, e:
1299 time.sleep(0.5)
1306 time.sleep(0.5)
1300 self.__handleError(procUnitConfObj, send_email=True)
1307 self.__handleError(procUnitConfObj)
1301 is_ok = False
1308 is_ok = False
1302 break
1309 break
1303 except:
1310 except:
1304 time.sleep(0.5)
1311 time.sleep(0.5)
1305 self.__handleError(procUnitConfObj)
1312 self.__handleError(procUnitConfObj)
1306 is_ok = False
1313 is_ok = False
1307 break
1314 break
1308
1315
1309 # If every process unit finished so end process
1316 # If every process unit finished so end process
1310 if not(is_ok):
1317 if not(is_ok):
1311 break
1318 break
1312
1319
1313 if not self.runController():
1320 if not self.runController():
1314 break
1321 break
1315
1322
1316 # Closing every process
1323 # Closing every process
1317 for procKey in keyList:
1324 for procKey in keyList:
1318 procUnitConfObj = self.procUnitConfObjDict[procKey]
1325 procUnitConfObj = self.procUnitConfObjDict[procKey]
1319 procUnitConfObj.close()
1326 procUnitConfObj.close()
1320
1327
1321 log.success('{} finished (time: {}s)'.format(
1328 log.success('{} finished (time: {}s)'.format(
1322 self.name,
1329 self.name,
1323 time.time()-self.start_time))
1330 time.time()-self.start_time))
@@ -1,965 +1,971
1
1
2 import os
2 import os
3 import time
3 import time
4 import glob
4 import glob
5 import datetime
5 import datetime
6 from multiprocessing import Process
6 from multiprocessing import Process
7
7
8 import zmq
8 import zmq
9 import numpy
9 import numpy
10 import matplotlib
10 import matplotlib
11 import matplotlib.pyplot as plt
11 import matplotlib.pyplot as plt
12 from mpl_toolkits.axes_grid1 import make_axes_locatable
12 from mpl_toolkits.axes_grid1 import make_axes_locatable
13 from matplotlib.ticker import FuncFormatter, LinearLocator, MultipleLocator
13 from matplotlib.ticker import FuncFormatter, LinearLocator, MultipleLocator
14
14
15 from schainpy.model.proc.jroproc_base import Operation
15 from schainpy.model.proc.jroproc_base import Operation
16 from schainpy.utils import log
16 from schainpy.utils import log
17
17
18 jet_values = matplotlib.pyplot.get_cmap('jet', 100)(numpy.arange(100))[10:90]
18 jet_values = matplotlib.pyplot.get_cmap('jet', 100)(numpy.arange(100))[10:90]
19 blu_values = matplotlib.pyplot.get_cmap(
19 blu_values = matplotlib.pyplot.get_cmap(
20 'seismic_r', 20)(numpy.arange(20))[10:15]
20 'seismic_r', 20)(numpy.arange(20))[10:15]
21 ncmap = matplotlib.colors.LinearSegmentedColormap.from_list(
21 ncmap = matplotlib.colors.LinearSegmentedColormap.from_list(
22 'jro', numpy.vstack((blu_values, jet_values)))
22 'jro', numpy.vstack((blu_values, jet_values)))
23 matplotlib.pyplot.register_cmap(cmap=ncmap)
23 matplotlib.pyplot.register_cmap(cmap=ncmap)
24
24
25 CMAPS = [plt.get_cmap(s) for s in ('jro', 'jet', 'viridis', 'plasma', 'inferno', 'Greys', 'seismic', 'bwr', 'coolwarm')]
25 CMAPS = [plt.get_cmap(s) for s in ('jro', 'jet', 'viridis', 'plasma', 'inferno', 'Greys', 'seismic', 'bwr', 'coolwarm')]
26
26
27
27
28 def figpause(interval):
28 def figpause(interval):
29 backend = plt.rcParams['backend']
29 backend = plt.rcParams['backend']
30 if backend in matplotlib.rcsetup.interactive_bk:
30 if backend in matplotlib.rcsetup.interactive_bk:
31 figManager = matplotlib._pylab_helpers.Gcf.get_active()
31 figManager = matplotlib._pylab_helpers.Gcf.get_active()
32 if figManager is not None:
32 if figManager is not None:
33 canvas = figManager.canvas
33 canvas = figManager.canvas
34 if canvas.figure.stale:
34 if canvas.figure.stale:
35 canvas.draw()
35 canvas.draw()
36 canvas.start_event_loop(interval)
36 canvas.start_event_loop(interval)
37 return
37 return
38
38
39 def popup(message):
40 fig = plt.figure(figsize=(12, 8), facecolor='r')
41 fig.text(0.5, 0.5, message, ha='center', va='center', size='20', weight='heavy', color='w')
42 fig.show()
43 figpause(1000)
44
39
45
40 class PlotData(Operation, Process):
46 class PlotData(Operation, Process):
41 '''
47 '''
42 Base class for Schain plotting operations
48 Base class for Schain plotting operations
43 '''
49 '''
44
50
45 CODE = 'Figure'
51 CODE = 'Figure'
46 colormap = 'jro'
52 colormap = 'jro'
47 bgcolor = 'white'
53 bgcolor = 'white'
48 CONFLATE = False
54 CONFLATE = False
49 __missing = 1E30
55 __missing = 1E30
50
56
51 __attrs__ = ['show', 'save', 'xmin', 'xmax', 'ymin', 'ymax', 'zmin', 'zmax',
57 __attrs__ = ['show', 'save', 'xmin', 'xmax', 'ymin', 'ymax', 'zmin', 'zmax',
52 'zlimits', 'xlabel', 'ylabel', 'xaxis','cb_label', 'title',
58 'zlimits', 'xlabel', 'ylabel', 'xaxis','cb_label', 'title',
53 'colorbar', 'bgcolor', 'width', 'height', 'localtime', 'oneFigure',
59 'colorbar', 'bgcolor', 'width', 'height', 'localtime', 'oneFigure',
54 'showprofile', 'decimation']
60 'showprofile', 'decimation']
55
61
56 def __init__(self, **kwargs):
62 def __init__(self, **kwargs):
57
63
58 Operation.__init__(self, plot=True, **kwargs)
64 Operation.__init__(self, plot=True, **kwargs)
59 Process.__init__(self)
65 Process.__init__(self)
60
66
61 self.kwargs['code'] = self.CODE
67 self.kwargs['code'] = self.CODE
62 self.mp = False
68 self.mp = False
63 self.data = None
69 self.data = None
64 self.isConfig = False
70 self.isConfig = False
65 self.figures = []
71 self.figures = []
66 self.axes = []
72 self.axes = []
67 self.cb_axes = []
73 self.cb_axes = []
68 self.localtime = kwargs.pop('localtime', True)
74 self.localtime = kwargs.pop('localtime', True)
69 self.show = kwargs.get('show', True)
75 self.show = kwargs.get('show', True)
70 self.save = kwargs.get('save', False)
76 self.save = kwargs.get('save', False)
71 self.colormap = kwargs.get('colormap', self.colormap)
77 self.colormap = kwargs.get('colormap', self.colormap)
72 self.colormap_coh = kwargs.get('colormap_coh', 'jet')
78 self.colormap_coh = kwargs.get('colormap_coh', 'jet')
73 self.colormap_phase = kwargs.get('colormap_phase', 'RdBu_r')
79 self.colormap_phase = kwargs.get('colormap_phase', 'RdBu_r')
74 self.colormaps = kwargs.get('colormaps', None)
80 self.colormaps = kwargs.get('colormaps', None)
75 self.bgcolor = kwargs.get('bgcolor', self.bgcolor)
81 self.bgcolor = kwargs.get('bgcolor', self.bgcolor)
76 self.showprofile = kwargs.get('showprofile', False)
82 self.showprofile = kwargs.get('showprofile', False)
77 self.title = kwargs.get('wintitle', self.CODE.upper())
83 self.title = kwargs.get('wintitle', self.CODE.upper())
78 self.cb_label = kwargs.get('cb_label', None)
84 self.cb_label = kwargs.get('cb_label', None)
79 self.cb_labels = kwargs.get('cb_labels', None)
85 self.cb_labels = kwargs.get('cb_labels', None)
80 self.xaxis = kwargs.get('xaxis', 'frequency')
86 self.xaxis = kwargs.get('xaxis', 'frequency')
81 self.zmin = kwargs.get('zmin', None)
87 self.zmin = kwargs.get('zmin', None)
82 self.zmax = kwargs.get('zmax', None)
88 self.zmax = kwargs.get('zmax', None)
83 self.zlimits = kwargs.get('zlimits', None)
89 self.zlimits = kwargs.get('zlimits', None)
84 self.xmin = kwargs.get('xmin', None)
90 self.xmin = kwargs.get('xmin', None)
85 self.xmax = kwargs.get('xmax', None)
91 self.xmax = kwargs.get('xmax', None)
86 self.xrange = kwargs.get('xrange', 24)
92 self.xrange = kwargs.get('xrange', 24)
87 self.ymin = kwargs.get('ymin', None)
93 self.ymin = kwargs.get('ymin', None)
88 self.ymax = kwargs.get('ymax', None)
94 self.ymax = kwargs.get('ymax', None)
89 self.xlabel = kwargs.get('xlabel', None)
95 self.xlabel = kwargs.get('xlabel', None)
90 self.decimation = kwargs.get('decimation', None)
96 self.decimation = kwargs.get('decimation', None)
91 self.showSNR = kwargs.get('showSNR', False)
97 self.showSNR = kwargs.get('showSNR', False)
92 self.oneFigure = kwargs.get('oneFigure', True)
98 self.oneFigure = kwargs.get('oneFigure', True)
93 self.width = kwargs.get('width', None)
99 self.width = kwargs.get('width', None)
94 self.height = kwargs.get('height', None)
100 self.height = kwargs.get('height', None)
95 self.colorbar = kwargs.get('colorbar', True)
101 self.colorbar = kwargs.get('colorbar', True)
96 self.factors = kwargs.get('factors', [1, 1, 1, 1, 1, 1, 1, 1])
102 self.factors = kwargs.get('factors', [1, 1, 1, 1, 1, 1, 1, 1])
97 self.titles = kwargs.get('titles', [])
103 self.titles = kwargs.get('titles', [])
98 self.polar = False
104 self.polar = False
99
105
100 def __fmtTime(self, x, pos):
106 def __fmtTime(self, x, pos):
101 '''
107 '''
102 '''
108 '''
103
109
104 return '{}'.format(self.getDateTime(x).strftime('%H:%M'))
110 return '{}'.format(self.getDateTime(x).strftime('%H:%M'))
105
111
106 def __setup(self):
112 def __setup(self):
107 '''
113 '''
108 Common setup for all figures, here figures and axes are created
114 Common setup for all figures, here figures and axes are created
109 '''
115 '''
110
116
111 if self.CODE not in self.data:
117 if self.CODE not in self.data:
112 raise ValueError(log.error('Missing data for {}'.format(self.CODE),
118 raise ValueError(log.error('Missing data for {}'.format(self.CODE),
113 self.name))
119 self.name))
114
120
115 self.setup()
121 self.setup()
116
122
117 self.time_label = 'LT' if self.localtime else 'UTC'
123 self.time_label = 'LT' if self.localtime else 'UTC'
118 if self.data.localtime:
124 if self.data.localtime:
119 self.getDateTime = datetime.datetime.fromtimestamp
125 self.getDateTime = datetime.datetime.fromtimestamp
120 else:
126 else:
121 self.getDateTime = datetime.datetime.utcfromtimestamp
127 self.getDateTime = datetime.datetime.utcfromtimestamp
122
128
123 if self.width is None:
129 if self.width is None:
124 self.width = 8
130 self.width = 8
125
131
126 self.figures = []
132 self.figures = []
127 self.axes = []
133 self.axes = []
128 self.cb_axes = []
134 self.cb_axes = []
129 self.pf_axes = []
135 self.pf_axes = []
130 self.cmaps = []
136 self.cmaps = []
131
137
132 size = '15%' if self.ncols == 1 else '30%'
138 size = '15%' if self.ncols == 1 else '30%'
133 pad = '4%' if self.ncols == 1 else '8%'
139 pad = '4%' if self.ncols == 1 else '8%'
134
140
135 if self.oneFigure:
141 if self.oneFigure:
136 if self.height is None:
142 if self.height is None:
137 self.height = 1.4 * self.nrows + 1
143 self.height = 1.4 * self.nrows + 1
138 fig = plt.figure(figsize=(self.width, self.height),
144 fig = plt.figure(figsize=(self.width, self.height),
139 edgecolor='k',
145 edgecolor='k',
140 facecolor='w')
146 facecolor='w')
141 self.figures.append(fig)
147 self.figures.append(fig)
142 for n in range(self.nplots):
148 for n in range(self.nplots):
143 ax = fig.add_subplot(self.nrows, self.ncols,
149 ax = fig.add_subplot(self.nrows, self.ncols,
144 n + 1, polar=self.polar)
150 n + 1, polar=self.polar)
145 ax.tick_params(labelsize=8)
151 ax.tick_params(labelsize=8)
146 ax.firsttime = True
152 ax.firsttime = True
147 ax.index = 0
153 ax.index = 0
148 ax.press = None
154 ax.press = None
149 self.axes.append(ax)
155 self.axes.append(ax)
150 if self.showprofile:
156 if self.showprofile:
151 cax = self.__add_axes(ax, size=size, pad=pad)
157 cax = self.__add_axes(ax, size=size, pad=pad)
152 cax.tick_params(labelsize=8)
158 cax.tick_params(labelsize=8)
153 self.pf_axes.append(cax)
159 self.pf_axes.append(cax)
154 else:
160 else:
155 if self.height is None:
161 if self.height is None:
156 self.height = 3
162 self.height = 3
157 for n in range(self.nplots):
163 for n in range(self.nplots):
158 fig = plt.figure(figsize=(self.width, self.height),
164 fig = plt.figure(figsize=(self.width, self.height),
159 edgecolor='k',
165 edgecolor='k',
160 facecolor='w')
166 facecolor='w')
161 ax = fig.add_subplot(1, 1, 1, polar=self.polar)
167 ax = fig.add_subplot(1, 1, 1, polar=self.polar)
162 ax.tick_params(labelsize=8)
168 ax.tick_params(labelsize=8)
163 ax.firsttime = True
169 ax.firsttime = True
164 ax.index = 0
170 ax.index = 0
165 ax.press = None
171 ax.press = None
166 self.figures.append(fig)
172 self.figures.append(fig)
167 self.axes.append(ax)
173 self.axes.append(ax)
168 if self.showprofile:
174 if self.showprofile:
169 cax = self.__add_axes(ax, size=size, pad=pad)
175 cax = self.__add_axes(ax, size=size, pad=pad)
170 cax.tick_params(labelsize=8)
176 cax.tick_params(labelsize=8)
171 self.pf_axes.append(cax)
177 self.pf_axes.append(cax)
172
178
173 for n in range(self.nrows):
179 for n in range(self.nrows):
174 if self.colormaps is not None:
180 if self.colormaps is not None:
175 cmap = plt.get_cmap(self.colormaps[n])
181 cmap = plt.get_cmap(self.colormaps[n])
176 else:
182 else:
177 cmap = plt.get_cmap(self.colormap)
183 cmap = plt.get_cmap(self.colormap)
178 cmap.set_bad(self.bgcolor, 1.)
184 cmap.set_bad(self.bgcolor, 1.)
179 self.cmaps.append(cmap)
185 self.cmaps.append(cmap)
180
186
181 for fig in self.figures:
187 for fig in self.figures:
182 fig.canvas.mpl_connect('key_press_event', self.OnKeyPress)
188 fig.canvas.mpl_connect('key_press_event', self.OnKeyPress)
183 fig.canvas.mpl_connect('scroll_event', self.OnBtnScroll)
189 fig.canvas.mpl_connect('scroll_event', self.OnBtnScroll)
184 fig.canvas.mpl_connect('button_press_event', self.onBtnPress)
190 fig.canvas.mpl_connect('button_press_event', self.onBtnPress)
185 fig.canvas.mpl_connect('motion_notify_event', self.onMotion)
191 fig.canvas.mpl_connect('motion_notify_event', self.onMotion)
186 fig.canvas.mpl_connect('button_release_event', self.onBtnRelease)
192 fig.canvas.mpl_connect('button_release_event', self.onBtnRelease)
187 if self.show:
193 if self.show:
188 fig.show()
194 fig.show()
189
195
190 def OnKeyPress(self, event):
196 def OnKeyPress(self, event):
191 '''
197 '''
192 Event for pressing keys (up, down) change colormap
198 Event for pressing keys (up, down) change colormap
193 '''
199 '''
194 ax = event.inaxes
200 ax = event.inaxes
195 if ax in self.axes:
201 if ax in self.axes:
196 if event.key == 'down':
202 if event.key == 'down':
197 ax.index += 1
203 ax.index += 1
198 elif event.key == 'up':
204 elif event.key == 'up':
199 ax.index -= 1
205 ax.index -= 1
200 if ax.index < 0:
206 if ax.index < 0:
201 ax.index = len(CMAPS) - 1
207 ax.index = len(CMAPS) - 1
202 elif ax.index == len(CMAPS):
208 elif ax.index == len(CMAPS):
203 ax.index = 0
209 ax.index = 0
204 cmap = CMAPS[ax.index]
210 cmap = CMAPS[ax.index]
205 ax.cbar.set_cmap(cmap)
211 ax.cbar.set_cmap(cmap)
206 ax.cbar.draw_all()
212 ax.cbar.draw_all()
207 ax.plt.set_cmap(cmap)
213 ax.plt.set_cmap(cmap)
208 ax.cbar.patch.figure.canvas.draw()
214 ax.cbar.patch.figure.canvas.draw()
209 self.colormap = cmap.name
215 self.colormap = cmap.name
210
216
211 def OnBtnScroll(self, event):
217 def OnBtnScroll(self, event):
212 '''
218 '''
213 Event for scrolling, scale figure
219 Event for scrolling, scale figure
214 '''
220 '''
215 cb_ax = event.inaxes
221 cb_ax = event.inaxes
216 if cb_ax in [ax.cbar.ax for ax in self.axes if ax.cbar]:
222 if cb_ax in [ax.cbar.ax for ax in self.axes if ax.cbar]:
217 ax = [ax for ax in self.axes if cb_ax == ax.cbar.ax][0]
223 ax = [ax for ax in self.axes if cb_ax == ax.cbar.ax][0]
218 pt = ax.cbar.ax.bbox.get_points()[:, 1]
224 pt = ax.cbar.ax.bbox.get_points()[:, 1]
219 nrm = ax.cbar.norm
225 nrm = ax.cbar.norm
220 vmin, vmax, p0, p1, pS = (
226 vmin, vmax, p0, p1, pS = (
221 nrm.vmin, nrm.vmax, pt[0], pt[1], event.y)
227 nrm.vmin, nrm.vmax, pt[0], pt[1], event.y)
222 scale = 2 if event.step == 1 else 0.5
228 scale = 2 if event.step == 1 else 0.5
223 point = vmin + (vmax - vmin) / (p1 - p0) * (pS - p0)
229 point = vmin + (vmax - vmin) / (p1 - p0) * (pS - p0)
224 ax.cbar.norm.vmin = point - scale * (point - vmin)
230 ax.cbar.norm.vmin = point - scale * (point - vmin)
225 ax.cbar.norm.vmax = point - scale * (point - vmax)
231 ax.cbar.norm.vmax = point - scale * (point - vmax)
226 ax.plt.set_norm(ax.cbar.norm)
232 ax.plt.set_norm(ax.cbar.norm)
227 ax.cbar.draw_all()
233 ax.cbar.draw_all()
228 ax.cbar.patch.figure.canvas.draw()
234 ax.cbar.patch.figure.canvas.draw()
229
235
230 def onBtnPress(self, event):
236 def onBtnPress(self, event):
231 '''
237 '''
232 Event for mouse button press
238 Event for mouse button press
233 '''
239 '''
234 cb_ax = event.inaxes
240 cb_ax = event.inaxes
235 if cb_ax is None:
241 if cb_ax is None:
236 return
242 return
237
243
238 if cb_ax in [ax.cbar.ax for ax in self.axes if ax.cbar]:
244 if cb_ax in [ax.cbar.ax for ax in self.axes if ax.cbar]:
239 cb_ax.press = event.x, event.y
245 cb_ax.press = event.x, event.y
240 else:
246 else:
241 cb_ax.press = None
247 cb_ax.press = None
242
248
243 def onMotion(self, event):
249 def onMotion(self, event):
244 '''
250 '''
245 Event for move inside colorbar
251 Event for move inside colorbar
246 '''
252 '''
247 cb_ax = event.inaxes
253 cb_ax = event.inaxes
248 if cb_ax is None:
254 if cb_ax is None:
249 return
255 return
250 if cb_ax not in [ax.cbar.ax for ax in self.axes if ax.cbar]:
256 if cb_ax not in [ax.cbar.ax for ax in self.axes if ax.cbar]:
251 return
257 return
252 if cb_ax.press is None:
258 if cb_ax.press is None:
253 return
259 return
254
260
255 ax = [ax for ax in self.axes if cb_ax == ax.cbar.ax][0]
261 ax = [ax for ax in self.axes if cb_ax == ax.cbar.ax][0]
256 xprev, yprev = cb_ax.press
262 xprev, yprev = cb_ax.press
257 dx = event.x - xprev
263 dx = event.x - xprev
258 dy = event.y - yprev
264 dy = event.y - yprev
259 cb_ax.press = event.x, event.y
265 cb_ax.press = event.x, event.y
260 scale = ax.cbar.norm.vmax - ax.cbar.norm.vmin
266 scale = ax.cbar.norm.vmax - ax.cbar.norm.vmin
261 perc = 0.03
267 perc = 0.03
262
268
263 if event.button == 1:
269 if event.button == 1:
264 ax.cbar.norm.vmin -= (perc * scale) * numpy.sign(dy)
270 ax.cbar.norm.vmin -= (perc * scale) * numpy.sign(dy)
265 ax.cbar.norm.vmax -= (perc * scale) * numpy.sign(dy)
271 ax.cbar.norm.vmax -= (perc * scale) * numpy.sign(dy)
266 elif event.button == 3:
272 elif event.button == 3:
267 ax.cbar.norm.vmin -= (perc * scale) * numpy.sign(dy)
273 ax.cbar.norm.vmin -= (perc * scale) * numpy.sign(dy)
268 ax.cbar.norm.vmax += (perc * scale) * numpy.sign(dy)
274 ax.cbar.norm.vmax += (perc * scale) * numpy.sign(dy)
269
275
270 ax.cbar.draw_all()
276 ax.cbar.draw_all()
271 ax.plt.set_norm(ax.cbar.norm)
277 ax.plt.set_norm(ax.cbar.norm)
272 ax.cbar.patch.figure.canvas.draw()
278 ax.cbar.patch.figure.canvas.draw()
273
279
274 def onBtnRelease(self, event):
280 def onBtnRelease(self, event):
275 '''
281 '''
276 Event for mouse button release
282 Event for mouse button release
277 '''
283 '''
278 cb_ax = event.inaxes
284 cb_ax = event.inaxes
279 if cb_ax is not None:
285 if cb_ax is not None:
280 cb_ax.press = None
286 cb_ax.press = None
281
287
282 def __add_axes(self, ax, size='30%', pad='8%'):
288 def __add_axes(self, ax, size='30%', pad='8%'):
283 '''
289 '''
284 Add new axes to the given figure
290 Add new axes to the given figure
285 '''
291 '''
286 divider = make_axes_locatable(ax)
292 divider = make_axes_locatable(ax)
287 nax = divider.new_horizontal(size=size, pad=pad)
293 nax = divider.new_horizontal(size=size, pad=pad)
288 ax.figure.add_axes(nax)
294 ax.figure.add_axes(nax)
289 return nax
295 return nax
290
296
291 self.setup()
297 self.setup()
292
298
293 def setup(self):
299 def setup(self):
294 '''
300 '''
295 This method should be implemented in the child class, the following
301 This method should be implemented in the child class, the following
296 attributes should be set:
302 attributes should be set:
297
303
298 self.nrows: number of rows
304 self.nrows: number of rows
299 self.ncols: number of cols
305 self.ncols: number of cols
300 self.nplots: number of plots (channels or pairs)
306 self.nplots: number of plots (channels or pairs)
301 self.ylabel: label for Y axes
307 self.ylabel: label for Y axes
302 self.titles: list of axes title
308 self.titles: list of axes title
303
309
304 '''
310 '''
305 raise(NotImplementedError, 'Implement this method in child class')
311 raise(NotImplementedError, 'Implement this method in child class')
306
312
307 def fill_gaps(self, x_buffer, y_buffer, z_buffer):
313 def fill_gaps(self, x_buffer, y_buffer, z_buffer):
308 '''
314 '''
309 Create a masked array for missing data
315 Create a masked array for missing data
310 '''
316 '''
311 if x_buffer.shape[0] < 2:
317 if x_buffer.shape[0] < 2:
312 return x_buffer, y_buffer, z_buffer
318 return x_buffer, y_buffer, z_buffer
313
319
314 deltas = x_buffer[1:] - x_buffer[0:-1]
320 deltas = x_buffer[1:] - x_buffer[0:-1]
315 x_median = numpy.median(deltas)
321 x_median = numpy.median(deltas)
316
322
317 index = numpy.where(deltas > 5 * x_median)
323 index = numpy.where(deltas > 5 * x_median)
318
324
319 if len(index[0]) != 0:
325 if len(index[0]) != 0:
320 z_buffer[::, index[0], ::] = self.__missing
326 z_buffer[::, index[0], ::] = self.__missing
321 z_buffer = numpy.ma.masked_inside(z_buffer,
327 z_buffer = numpy.ma.masked_inside(z_buffer,
322 0.99 * self.__missing,
328 0.99 * self.__missing,
323 1.01 * self.__missing)
329 1.01 * self.__missing)
324
330
325 return x_buffer, y_buffer, z_buffer
331 return x_buffer, y_buffer, z_buffer
326
332
327 def decimate(self):
333 def decimate(self):
328
334
329 # dx = int(len(self.x)/self.__MAXNUMX) + 1
335 # dx = int(len(self.x)/self.__MAXNUMX) + 1
330 dy = int(len(self.y) / self.decimation) + 1
336 dy = int(len(self.y) / self.decimation) + 1
331
337
332 # x = self.x[::dx]
338 # x = self.x[::dx]
333 x = self.x
339 x = self.x
334 y = self.y[::dy]
340 y = self.y[::dy]
335 z = self.z[::, ::, ::dy]
341 z = self.z[::, ::, ::dy]
336
342
337 return x, y, z
343 return x, y, z
338
344
339 def format(self):
345 def format(self):
340 '''
346 '''
341 Set min and max values, labels, ticks and titles
347 Set min and max values, labels, ticks and titles
342 '''
348 '''
343
349
344 if self.xmin is None:
350 if self.xmin is None:
345 xmin = self.min_time
351 xmin = self.min_time
346 else:
352 else:
347 if self.xaxis is 'time':
353 if self.xaxis is 'time':
348 dt = self.getDateTime(self.min_time)
354 dt = self.getDateTime(self.min_time)
349 xmin = (dt.replace(hour=int(self.xmin), minute=0, second=0) -
355 xmin = (dt.replace(hour=int(self.xmin), minute=0, second=0) -
350 datetime.datetime(1970, 1, 1)).total_seconds()
356 datetime.datetime(1970, 1, 1)).total_seconds()
351 if self.data.localtime:
357 if self.data.localtime:
352 xmin += time.timezone
358 xmin += time.timezone
353 else:
359 else:
354 xmin = self.xmin
360 xmin = self.xmin
355
361
356 if self.xmax is None:
362 if self.xmax is None:
357 xmax = xmin + self.xrange * 60 * 60
363 xmax = xmin + self.xrange * 60 * 60
358 else:
364 else:
359 if self.xaxis is 'time':
365 if self.xaxis is 'time':
360 dt = self.getDateTime(self.max_time)
366 dt = self.getDateTime(self.max_time)
361 xmax = (dt.replace(hour=int(self.xmax), minute=59, second=59) -
367 xmax = (dt.replace(hour=int(self.xmax), minute=59, second=59) -
362 datetime.datetime(1970, 1, 1) + datetime.timedelta(seconds=1)).total_seconds()
368 datetime.datetime(1970, 1, 1) + datetime.timedelta(seconds=1)).total_seconds()
363 if self.data.localtime:
369 if self.data.localtime:
364 xmax += time.timezone
370 xmax += time.timezone
365 else:
371 else:
366 xmax = self.xmax
372 xmax = self.xmax
367
373
368 ymin = self.ymin if self.ymin else numpy.nanmin(self.y)
374 ymin = self.ymin if self.ymin else numpy.nanmin(self.y)
369 ymax = self.ymax if self.ymax else numpy.nanmax(self.y)
375 ymax = self.ymax if self.ymax else numpy.nanmax(self.y)
370
376
371 Y = numpy.array([5, 10, 20, 50, 100, 200, 500, 1000, 2000])
377 Y = numpy.array([5, 10, 20, 50, 100, 200, 500, 1000, 2000])
372 i = 1 if numpy.where(ymax-ymin < Y)[0][0] < 0 else numpy.where(ymax-ymin < Y)[0][0]
378 i = 1 if numpy.where(ymax-ymin < Y)[0][0] < 0 else numpy.where(ymax-ymin < Y)[0][0]
373 ystep = Y[i] / 5
379 ystep = Y[i] / 5
374
380
375 for n, ax in enumerate(self.axes):
381 for n, ax in enumerate(self.axes):
376 if ax.firsttime:
382 if ax.firsttime:
377 ax.set_facecolor(self.bgcolor)
383 ax.set_facecolor(self.bgcolor)
378 ax.yaxis.set_major_locator(MultipleLocator(ystep))
384 ax.yaxis.set_major_locator(MultipleLocator(ystep))
379 if self.xaxis is 'time':
385 if self.xaxis is 'time':
380 ax.xaxis.set_major_formatter(FuncFormatter(self.__fmtTime))
386 ax.xaxis.set_major_formatter(FuncFormatter(self.__fmtTime))
381 ax.xaxis.set_major_locator(LinearLocator(9))
387 ax.xaxis.set_major_locator(LinearLocator(9))
382 if self.xlabel is not None:
388 if self.xlabel is not None:
383 ax.set_xlabel(self.xlabel)
389 ax.set_xlabel(self.xlabel)
384 ax.set_ylabel(self.ylabel)
390 ax.set_ylabel(self.ylabel)
385 ax.firsttime = False
391 ax.firsttime = False
386 if self.showprofile:
392 if self.showprofile:
387 self.pf_axes[n].set_ylim(ymin, ymax)
393 self.pf_axes[n].set_ylim(ymin, ymax)
388 self.pf_axes[n].set_xlim(self.zmin, self.zmax)
394 self.pf_axes[n].set_xlim(self.zmin, self.zmax)
389 self.pf_axes[n].set_xlabel('dB')
395 self.pf_axes[n].set_xlabel('dB')
390 self.pf_axes[n].grid(b=True, axis='x')
396 self.pf_axes[n].grid(b=True, axis='x')
391 [tick.set_visible(False)
397 [tick.set_visible(False)
392 for tick in self.pf_axes[n].get_yticklabels()]
398 for tick in self.pf_axes[n].get_yticklabels()]
393 if self.colorbar:
399 if self.colorbar:
394 ax.cbar = plt.colorbar(
400 ax.cbar = plt.colorbar(
395 ax.plt, ax=ax, fraction=0.05, pad=0.02, aspect=10)
401 ax.plt, ax=ax, fraction=0.05, pad=0.02, aspect=10)
396 ax.cbar.ax.tick_params(labelsize=8)
402 ax.cbar.ax.tick_params(labelsize=8)
397 ax.cbar.ax.press = None
403 ax.cbar.ax.press = None
398 if self.cb_label:
404 if self.cb_label:
399 ax.cbar.set_label(self.cb_label, size=8)
405 ax.cbar.set_label(self.cb_label, size=8)
400 elif self.cb_labels:
406 elif self.cb_labels:
401 ax.cbar.set_label(self.cb_labels[n], size=8)
407 ax.cbar.set_label(self.cb_labels[n], size=8)
402 else:
408 else:
403 ax.cbar = None
409 ax.cbar = None
404
410
405 if not self.polar:
411 if not self.polar:
406 ax.set_xlim(xmin, xmax)
412 ax.set_xlim(xmin, xmax)
407 ax.set_ylim(ymin, ymax)
413 ax.set_ylim(ymin, ymax)
408 ax.set_title('{} - {} {}'.format(
414 ax.set_title('{} - {} {}'.format(
409 self.titles[n],
415 self.titles[n],
410 self.getDateTime(self.max_time).strftime('%H:%M:%S'),
416 self.getDateTime(self.max_time).strftime('%H:%M:%S'),
411 self.time_label),
417 self.time_label),
412 size=8)
418 size=8)
413 else:
419 else:
414 ax.set_title('{}'.format(self.titles[n]), size=8)
420 ax.set_title('{}'.format(self.titles[n]), size=8)
415 ax.set_ylim(0, 90)
421 ax.set_ylim(0, 90)
416 ax.set_yticks(numpy.arange(0, 90, 20))
422 ax.set_yticks(numpy.arange(0, 90, 20))
417 ax.yaxis.labelpad = 40
423 ax.yaxis.labelpad = 40
418
424
419 def __plot(self):
425 def __plot(self):
420 '''
426 '''
421 '''
427 '''
422 log.success('Plotting', self.name)
428 log.success('Plotting', self.name)
423
429
424 try:
430 try:
425 self.plot()
431 self.plot()
426 self.format()
432 self.format()
427 except:
433 except:
428 log.warning('{} Plot could not be updated... check data'.format(self.CODE), self.name)
434 log.warning('{} Plot could not be updated... check data'.format(self.CODE), self.name)
429
435
430 for n, fig in enumerate(self.figures):
436 for n, fig in enumerate(self.figures):
431 if self.nrows == 0 or self.nplots == 0:
437 if self.nrows == 0 or self.nplots == 0:
432 log.warning('No data', self.name)
438 log.warning('No data', self.name)
433 fig.text(0.5, 0.5, 'No Data', fontsize='large', ha='center')
439 fig.text(0.5, 0.5, 'No Data', fontsize='large', ha='center')
434 fig.canvas.manager.set_window_title(self.CODE)
440 fig.canvas.manager.set_window_title(self.CODE)
435 continue
441 continue
436
442
437 fig.tight_layout()
443 fig.tight_layout()
438 fig.canvas.manager.set_window_title('{} - {}'.format(self.title,
444 fig.canvas.manager.set_window_title('{} - {}'.format(self.title,
439 self.getDateTime(self.max_time).strftime('%Y/%m/%d')))
445 self.getDateTime(self.max_time).strftime('%Y/%m/%d')))
440 fig.canvas.draw()
446 fig.canvas.draw()
441
447
442 if self.save and self.data.ended:
448 if self.save and self.data.ended:
443 channels = range(self.nrows)
449 channels = range(self.nrows)
444 if self.oneFigure:
450 if self.oneFigure:
445 label = ''
451 label = ''
446 else:
452 else:
447 label = '_{}'.format(channels[n])
453 label = '_{}'.format(channels[n])
448 figname = os.path.join(
454 figname = os.path.join(
449 self.save,
455 self.save,
450 '{}{}_{}.png'.format(
456 '{}{}_{}.png'.format(
451 self.CODE,
457 self.CODE,
452 label,
458 label,
453 self.getDateTime(self.saveTime).strftime(
459 self.getDateTime(self.saveTime).strftime(
454 '%Y%m%d_%H%M%S'),
460 '%Y%m%d_%H%M%S'),
455 )
461 )
456 )
462 )
457 log.log('Saving figure: {}'.format(figname), self.name)
463 log.log('Saving figure: {}'.format(figname), self.name)
458 fig.savefig(figname)
464 fig.savefig(figname)
459
465
460 def plot(self):
466 def plot(self):
461 '''
467 '''
462 '''
468 '''
463 raise(NotImplementedError, 'Implement this method in child class')
469 raise(NotImplementedError, 'Implement this method in child class')
464
470
465 def run(self):
471 def run(self):
466
472
467 log.success('Starting', self.name)
473 log.success('Starting', self.name)
468
474
469 context = zmq.Context()
475 context = zmq.Context()
470 receiver = context.socket(zmq.SUB)
476 receiver = context.socket(zmq.SUB)
471 receiver.setsockopt(zmq.SUBSCRIBE, '')
477 receiver.setsockopt(zmq.SUBSCRIBE, '')
472 receiver.setsockopt(zmq.CONFLATE, self.CONFLATE)
478 receiver.setsockopt(zmq.CONFLATE, self.CONFLATE)
473
479
474 if 'server' in self.kwargs['parent']:
480 if 'server' in self.kwargs['parent']:
475 receiver.connect(
481 receiver.connect(
476 'ipc:///tmp/{}.plots'.format(self.kwargs['parent']['server']))
482 'ipc:///tmp/{}.plots'.format(self.kwargs['parent']['server']))
477 else:
483 else:
478 receiver.connect("ipc:///tmp/zmq.plots")
484 receiver.connect("ipc:///tmp/zmq.plots")
479
485
480 while True:
486 while True:
481 try:
487 try:
482 self.data = receiver.recv_pyobj(flags=zmq.NOBLOCK)
488 self.data = receiver.recv_pyobj(flags=zmq.NOBLOCK)
483 if self.data.localtime and self.localtime:
489 if self.data.localtime and self.localtime:
484 self.times = self.data.times
490 self.times = self.data.times
485 elif self.data.localtime and not self.localtime:
491 elif self.data.localtime and not self.localtime:
486 self.times = self.data.times + time.timezone
492 self.times = self.data.times + time.timezone
487 elif not self.data.localtime and self.localtime:
493 elif not self.data.localtime and self.localtime:
488 self.times = self.data.times - time.timezone
494 self.times = self.data.times - time.timezone
489 else:
495 else:
490 self.times = self.data.times
496 self.times = self.data.times
491
497
492 self.min_time = self.times[0]
498 self.min_time = self.times[0]
493 self.max_time = self.times[-1]
499 self.max_time = self.times[-1]
494
500
495 if self.isConfig is False:
501 if self.isConfig is False:
496 self.__setup()
502 self.__setup()
497 self.isConfig = True
503 self.isConfig = True
498
504
499 self.__plot()
505 self.__plot()
500
506
501 except zmq.Again as e:
507 except zmq.Again as e:
502 log.log('Waiting for data...')
508 log.log('Waiting for data...')
503 if self.data:
509 if self.data:
504 figpause(self.data.throttle)
510 figpause(self.data.throttle)
505 else:
511 else:
506 time.sleep(2)
512 time.sleep(2)
507
513
508 def close(self):
514 def close(self):
509 if self.data:
515 if self.data:
510 self.__plot()
516 self.__plot()
511
517
512
518
513 class PlotSpectraData(PlotData):
519 class PlotSpectraData(PlotData):
514 '''
520 '''
515 Plot for Spectra data
521 Plot for Spectra data
516 '''
522 '''
517
523
518 CODE = 'spc'
524 CODE = 'spc'
519 colormap = 'jro'
525 colormap = 'jro'
520
526
521 def setup(self):
527 def setup(self):
522 self.nplots = len(self.data.channels)
528 self.nplots = len(self.data.channels)
523 self.ncols = int(numpy.sqrt(self.nplots) + 0.9)
529 self.ncols = int(numpy.sqrt(self.nplots) + 0.9)
524 self.nrows = int((1.0 * self.nplots / self.ncols) + 0.9)
530 self.nrows = int((1.0 * self.nplots / self.ncols) + 0.9)
525 self.width = 3.4 * self.ncols
531 self.width = 3.4 * self.ncols
526 self.height = 3 * self.nrows
532 self.height = 3 * self.nrows
527 self.cb_label = 'dB'
533 self.cb_label = 'dB'
528 if self.showprofile:
534 if self.showprofile:
529 self.width += 0.8 * self.ncols
535 self.width += 0.8 * self.ncols
530
536
531 self.ylabel = 'Range [km]'
537 self.ylabel = 'Range [km]'
532
538
533 def plot(self):
539 def plot(self):
534 if self.xaxis == "frequency":
540 if self.xaxis == "frequency":
535 x = self.data.xrange[0]
541 x = self.data.xrange[0]
536 self.xlabel = "Frequency (kHz)"
542 self.xlabel = "Frequency (kHz)"
537 elif self.xaxis == "time":
543 elif self.xaxis == "time":
538 x = self.data.xrange[1]
544 x = self.data.xrange[1]
539 self.xlabel = "Time (ms)"
545 self.xlabel = "Time (ms)"
540 else:
546 else:
541 x = self.data.xrange[2]
547 x = self.data.xrange[2]
542 self.xlabel = "Velocity (m/s)"
548 self.xlabel = "Velocity (m/s)"
543
549
544 if self.CODE == 'spc_mean':
550 if self.CODE == 'spc_mean':
545 x = self.data.xrange[2]
551 x = self.data.xrange[2]
546 self.xlabel = "Velocity (m/s)"
552 self.xlabel = "Velocity (m/s)"
547
553
548 self.titles = []
554 self.titles = []
549
555
550 y = self.data.heights
556 y = self.data.heights
551 self.y = y
557 self.y = y
552 z = self.data['spc']
558 z = self.data['spc']
553
559
554 for n, ax in enumerate(self.axes):
560 for n, ax in enumerate(self.axes):
555 noise = self.data['noise'][n][-1]
561 noise = self.data['noise'][n][-1]
556 if self.CODE == 'spc_mean':
562 if self.CODE == 'spc_mean':
557 mean = self.data['mean'][n][-1]
563 mean = self.data['mean'][n][-1]
558 if ax.firsttime:
564 if ax.firsttime:
559 self.xmax = self.xmax if self.xmax else numpy.nanmax(x)
565 self.xmax = self.xmax if self.xmax else numpy.nanmax(x)
560 self.xmin = self.xmin if self.xmin else -self.xmax
566 self.xmin = self.xmin if self.xmin else -self.xmax
561 self.zmin = self.zmin if self.zmin else numpy.nanmin(z)
567 self.zmin = self.zmin if self.zmin else numpy.nanmin(z)
562 self.zmax = self.zmax if self.zmax else numpy.nanmax(z)
568 self.zmax = self.zmax if self.zmax else numpy.nanmax(z)
563 ax.plt = ax.pcolormesh(x, y, z[n].T,
569 ax.plt = ax.pcolormesh(x, y, z[n].T,
564 vmin=self.zmin,
570 vmin=self.zmin,
565 vmax=self.zmax,
571 vmax=self.zmax,
566 cmap=plt.get_cmap(self.colormap)
572 cmap=plt.get_cmap(self.colormap)
567 )
573 )
568
574
569 if self.showprofile:
575 if self.showprofile:
570 ax.plt_profile = self.pf_axes[n].plot(
576 ax.plt_profile = self.pf_axes[n].plot(
571 self.data['rti'][n][-1], y)[0]
577 self.data['rti'][n][-1], y)[0]
572 ax.plt_noise = self.pf_axes[n].plot(numpy.repeat(noise, len(y)), y,
578 ax.plt_noise = self.pf_axes[n].plot(numpy.repeat(noise, len(y)), y,
573 color="k", linestyle="dashed", lw=1)[0]
579 color="k", linestyle="dashed", lw=1)[0]
574 if self.CODE == 'spc_mean':
580 if self.CODE == 'spc_mean':
575 ax.plt_mean = ax.plot(mean, y, color='k')[0]
581 ax.plt_mean = ax.plot(mean, y, color='k')[0]
576 else:
582 else:
577 ax.plt.set_array(z[n].T.ravel())
583 ax.plt.set_array(z[n].T.ravel())
578 if self.showprofile:
584 if self.showprofile:
579 ax.plt_profile.set_data(self.data['rti'][n][-1], y)
585 ax.plt_profile.set_data(self.data['rti'][n][-1], y)
580 ax.plt_noise.set_data(numpy.repeat(noise, len(y)), y)
586 ax.plt_noise.set_data(numpy.repeat(noise, len(y)), y)
581 if self.CODE == 'spc_mean':
587 if self.CODE == 'spc_mean':
582 ax.plt_mean.set_data(mean, y)
588 ax.plt_mean.set_data(mean, y)
583
589
584 self.titles.append('CH {}: {:3.2f}dB'.format(n, noise))
590 self.titles.append('CH {}: {:3.2f}dB'.format(n, noise))
585 self.saveTime = self.max_time
591 self.saveTime = self.max_time
586
592
587
593
588 class PlotCrossSpectraData(PlotData):
594 class PlotCrossSpectraData(PlotData):
589
595
590 CODE = 'cspc'
596 CODE = 'cspc'
591 zmin_coh = None
597 zmin_coh = None
592 zmax_coh = None
598 zmax_coh = None
593 zmin_phase = None
599 zmin_phase = None
594 zmax_phase = None
600 zmax_phase = None
595
601
596 def setup(self):
602 def setup(self):
597
603
598 self.ncols = 4
604 self.ncols = 4
599 self.nrows = len(self.data.pairs)
605 self.nrows = len(self.data.pairs)
600 self.nplots = self.nrows * 4
606 self.nplots = self.nrows * 4
601 self.width = 3.4 * self.ncols
607 self.width = 3.4 * self.ncols
602 self.height = 3 * self.nrows
608 self.height = 3 * self.nrows
603 self.ylabel = 'Range [km]'
609 self.ylabel = 'Range [km]'
604 self.showprofile = False
610 self.showprofile = False
605
611
606 def plot(self):
612 def plot(self):
607
613
608 if self.xaxis == "frequency":
614 if self.xaxis == "frequency":
609 x = self.data.xrange[0]
615 x = self.data.xrange[0]
610 self.xlabel = "Frequency (kHz)"
616 self.xlabel = "Frequency (kHz)"
611 elif self.xaxis == "time":
617 elif self.xaxis == "time":
612 x = self.data.xrange[1]
618 x = self.data.xrange[1]
613 self.xlabel = "Time (ms)"
619 self.xlabel = "Time (ms)"
614 else:
620 else:
615 x = self.data.xrange[2]
621 x = self.data.xrange[2]
616 self.xlabel = "Velocity (m/s)"
622 self.xlabel = "Velocity (m/s)"
617
623
618 self.titles = []
624 self.titles = []
619
625
620 y = self.data.heights
626 y = self.data.heights
621 self.y = y
627 self.y = y
622 spc = self.data['spc']
628 spc = self.data['spc']
623 cspc = self.data['cspc']
629 cspc = self.data['cspc']
624
630
625 for n in range(self.nrows):
631 for n in range(self.nrows):
626 noise = self.data['noise'][n][-1]
632 noise = self.data['noise'][n][-1]
627 pair = self.data.pairs[n]
633 pair = self.data.pairs[n]
628 ax = self.axes[4 * n]
634 ax = self.axes[4 * n]
629 ax3 = self.axes[4 * n + 3]
635 ax3 = self.axes[4 * n + 3]
630 if ax.firsttime:
636 if ax.firsttime:
631 self.xmax = self.xmax if self.xmax else numpy.nanmax(x)
637 self.xmax = self.xmax if self.xmax else numpy.nanmax(x)
632 self.xmin = self.xmin if self.xmin else -self.xmax
638 self.xmin = self.xmin if self.xmin else -self.xmax
633 self.zmin = self.zmin if self.zmin else numpy.nanmin(spc)
639 self.zmin = self.zmin if self.zmin else numpy.nanmin(spc)
634 self.zmax = self.zmax if self.zmax else numpy.nanmax(spc)
640 self.zmax = self.zmax if self.zmax else numpy.nanmax(spc)
635 ax.plt = ax.pcolormesh(x, y, spc[pair[0]].T,
641 ax.plt = ax.pcolormesh(x, y, spc[pair[0]].T,
636 vmin=self.zmin,
642 vmin=self.zmin,
637 vmax=self.zmax,
643 vmax=self.zmax,
638 cmap=plt.get_cmap(self.colormap)
644 cmap=plt.get_cmap(self.colormap)
639 )
645 )
640 else:
646 else:
641 ax.plt.set_array(spc[pair[0]].T.ravel())
647 ax.plt.set_array(spc[pair[0]].T.ravel())
642 self.titles.append('CH {}: {:3.2f}dB'.format(n, noise))
648 self.titles.append('CH {}: {:3.2f}dB'.format(n, noise))
643
649
644 ax = self.axes[4 * n + 1]
650 ax = self.axes[4 * n + 1]
645 if ax.firsttime:
651 if ax.firsttime:
646 ax.plt = ax.pcolormesh(x, y, spc[pair[1]].T,
652 ax.plt = ax.pcolormesh(x, y, spc[pair[1]].T,
647 vmin=self.zmin,
653 vmin=self.zmin,
648 vmax=self.zmax,
654 vmax=self.zmax,
649 cmap=plt.get_cmap(self.colormap)
655 cmap=plt.get_cmap(self.colormap)
650 )
656 )
651 else:
657 else:
652 ax.plt.set_array(spc[pair[1]].T.ravel())
658 ax.plt.set_array(spc[pair[1]].T.ravel())
653 self.titles.append('CH {}: {:3.2f}dB'.format(n, noise))
659 self.titles.append('CH {}: {:3.2f}dB'.format(n, noise))
654
660
655 out = cspc[n] / numpy.sqrt(spc[pair[0]] * spc[pair[1]])
661 out = cspc[n] / numpy.sqrt(spc[pair[0]] * spc[pair[1]])
656 coh = numpy.abs(out)
662 coh = numpy.abs(out)
657 phase = numpy.arctan2(out.imag, out.real) * 180 / numpy.pi
663 phase = numpy.arctan2(out.imag, out.real) * 180 / numpy.pi
658
664
659 ax = self.axes[4 * n + 2]
665 ax = self.axes[4 * n + 2]
660 if ax.firsttime:
666 if ax.firsttime:
661 ax.plt = ax.pcolormesh(x, y, coh.T,
667 ax.plt = ax.pcolormesh(x, y, coh.T,
662 vmin=0,
668 vmin=0,
663 vmax=1,
669 vmax=1,
664 cmap=plt.get_cmap(self.colormap_coh)
670 cmap=plt.get_cmap(self.colormap_coh)
665 )
671 )
666 else:
672 else:
667 ax.plt.set_array(coh.T.ravel())
673 ax.plt.set_array(coh.T.ravel())
668 self.titles.append(
674 self.titles.append(
669 'Coherence Ch{} * Ch{}'.format(pair[0], pair[1]))
675 'Coherence Ch{} * Ch{}'.format(pair[0], pair[1]))
670
676
671 ax = self.axes[4 * n + 3]
677 ax = self.axes[4 * n + 3]
672 if ax.firsttime:
678 if ax.firsttime:
673 ax.plt = ax.pcolormesh(x, y, phase.T,
679 ax.plt = ax.pcolormesh(x, y, phase.T,
674 vmin=-180,
680 vmin=-180,
675 vmax=180,
681 vmax=180,
676 cmap=plt.get_cmap(self.colormap_phase)
682 cmap=plt.get_cmap(self.colormap_phase)
677 )
683 )
678 else:
684 else:
679 ax.plt.set_array(phase.T.ravel())
685 ax.plt.set_array(phase.T.ravel())
680 self.titles.append('Phase CH{} * CH{}'.format(pair[0], pair[1]))
686 self.titles.append('Phase CH{} * CH{}'.format(pair[0], pair[1]))
681
687
682 self.saveTime = self.max_time
688 self.saveTime = self.max_time
683
689
684
690
685 class PlotSpectraMeanData(PlotSpectraData):
691 class PlotSpectraMeanData(PlotSpectraData):
686 '''
692 '''
687 Plot for Spectra and Mean
693 Plot for Spectra and Mean
688 '''
694 '''
689 CODE = 'spc_mean'
695 CODE = 'spc_mean'
690 colormap = 'jro'
696 colormap = 'jro'
691
697
692
698
693 class PlotRTIData(PlotData):
699 class PlotRTIData(PlotData):
694 '''
700 '''
695 Plot for RTI data
701 Plot for RTI data
696 '''
702 '''
697
703
698 CODE = 'rti'
704 CODE = 'rti'
699 colormap = 'jro'
705 colormap = 'jro'
700
706
701 def setup(self):
707 def setup(self):
702 self.xaxis = 'time'
708 self.xaxis = 'time'
703 self.ncols = 1
709 self.ncols = 1
704 self.nrows = len(self.data.channels)
710 self.nrows = len(self.data.channels)
705 self.nplots = len(self.data.channels)
711 self.nplots = len(self.data.channels)
706 self.ylabel = 'Range [km]'
712 self.ylabel = 'Range [km]'
707 self.cb_label = 'dB'
713 self.cb_label = 'dB'
708 self.titles = ['{} Channel {}'.format(
714 self.titles = ['{} Channel {}'.format(
709 self.CODE.upper(), x) for x in range(self.nrows)]
715 self.CODE.upper(), x) for x in range(self.nrows)]
710
716
711 def plot(self):
717 def plot(self):
712 self.x = self.times
718 self.x = self.times
713 self.y = self.data.heights
719 self.y = self.data.heights
714 self.z = self.data[self.CODE]
720 self.z = self.data[self.CODE]
715 self.z = numpy.ma.masked_invalid(self.z)
721 self.z = numpy.ma.masked_invalid(self.z)
716
722
717 if self.decimation is None:
723 if self.decimation is None:
718 x, y, z = self.fill_gaps(self.x, self.y, self.z)
724 x, y, z = self.fill_gaps(self.x, self.y, self.z)
719 else:
725 else:
720 x, y, z = self.fill_gaps(*self.decimate())
726 x, y, z = self.fill_gaps(*self.decimate())
721
727
722 for n, ax in enumerate(self.axes):
728 for n, ax in enumerate(self.axes):
723 self.zmin = self.zmin if self.zmin else numpy.min(self.z)
729 self.zmin = self.zmin if self.zmin else numpy.min(self.z)
724 self.zmax = self.zmax if self.zmax else numpy.max(self.z)
730 self.zmax = self.zmax if self.zmax else numpy.max(self.z)
725 if ax.firsttime:
731 if ax.firsttime:
726 ax.plt = ax.pcolormesh(x, y, z[n].T,
732 ax.plt = ax.pcolormesh(x, y, z[n].T,
727 vmin=self.zmin,
733 vmin=self.zmin,
728 vmax=self.zmax,
734 vmax=self.zmax,
729 cmap=plt.get_cmap(self.colormap)
735 cmap=plt.get_cmap(self.colormap)
730 )
736 )
731 if self.showprofile:
737 if self.showprofile:
732 ax.plot_profile = self.pf_axes[n].plot(
738 ax.plot_profile = self.pf_axes[n].plot(
733 self.data['rti'][n][-1], self.y)[0]
739 self.data['rti'][n][-1], self.y)[0]
734 ax.plot_noise = self.pf_axes[n].plot(numpy.repeat(self.data['noise'][n][-1], len(self.y)), self.y,
740 ax.plot_noise = self.pf_axes[n].plot(numpy.repeat(self.data['noise'][n][-1], len(self.y)), self.y,
735 color="k", linestyle="dashed", lw=1)[0]
741 color="k", linestyle="dashed", lw=1)[0]
736 else:
742 else:
737 ax.collections.remove(ax.collections[0])
743 ax.collections.remove(ax.collections[0])
738 ax.plt = ax.pcolormesh(x, y, z[n].T,
744 ax.plt = ax.pcolormesh(x, y, z[n].T,
739 vmin=self.zmin,
745 vmin=self.zmin,
740 vmax=self.zmax,
746 vmax=self.zmax,
741 cmap=plt.get_cmap(self.colormap)
747 cmap=plt.get_cmap(self.colormap)
742 )
748 )
743 if self.showprofile:
749 if self.showprofile:
744 ax.plot_profile.set_data(self.data['rti'][n][-1], self.y)
750 ax.plot_profile.set_data(self.data['rti'][n][-1], self.y)
745 ax.plot_noise.set_data(numpy.repeat(
751 ax.plot_noise.set_data(numpy.repeat(
746 self.data['noise'][n][-1], len(self.y)), self.y)
752 self.data['noise'][n][-1], len(self.y)), self.y)
747
753
748 self.saveTime = self.min_time
754 self.saveTime = self.min_time
749
755
750
756
751 class PlotCOHData(PlotRTIData):
757 class PlotCOHData(PlotRTIData):
752 '''
758 '''
753 Plot for Coherence data
759 Plot for Coherence data
754 '''
760 '''
755
761
756 CODE = 'coh'
762 CODE = 'coh'
757
763
758 def setup(self):
764 def setup(self):
759 self.xaxis = 'time'
765 self.xaxis = 'time'
760 self.ncols = 1
766 self.ncols = 1
761 self.nrows = len(self.data.pairs)
767 self.nrows = len(self.data.pairs)
762 self.nplots = len(self.data.pairs)
768 self.nplots = len(self.data.pairs)
763 self.ylabel = 'Range [km]'
769 self.ylabel = 'Range [km]'
764 if self.CODE == 'coh':
770 if self.CODE == 'coh':
765 self.cb_label = ''
771 self.cb_label = ''
766 self.titles = [
772 self.titles = [
767 'Coherence Map Ch{} * Ch{}'.format(x[0], x[1]) for x in self.data.pairs]
773 'Coherence Map Ch{} * Ch{}'.format(x[0], x[1]) for x in self.data.pairs]
768 else:
774 else:
769 self.cb_label = 'Degrees'
775 self.cb_label = 'Degrees'
770 self.titles = [
776 self.titles = [
771 'Phase Map Ch{} * Ch{}'.format(x[0], x[1]) for x in self.data.pairs]
777 'Phase Map Ch{} * Ch{}'.format(x[0], x[1]) for x in self.data.pairs]
772
778
773
779
774 class PlotPHASEData(PlotCOHData):
780 class PlotPHASEData(PlotCOHData):
775 '''
781 '''
776 Plot for Phase map data
782 Plot for Phase map data
777 '''
783 '''
778
784
779 CODE = 'phase'
785 CODE = 'phase'
780 colormap = 'seismic'
786 colormap = 'seismic'
781
787
782
788
783 class PlotNoiseData(PlotData):
789 class PlotNoiseData(PlotData):
784 '''
790 '''
785 Plot for noise
791 Plot for noise
786 '''
792 '''
787
793
788 CODE = 'noise'
794 CODE = 'noise'
789
795
790 def setup(self):
796 def setup(self):
791 self.xaxis = 'time'
797 self.xaxis = 'time'
792 self.ncols = 1
798 self.ncols = 1
793 self.nrows = 1
799 self.nrows = 1
794 self.nplots = 1
800 self.nplots = 1
795 self.ylabel = 'Intensity [dB]'
801 self.ylabel = 'Intensity [dB]'
796 self.titles = ['Noise']
802 self.titles = ['Noise']
797 self.colorbar = False
803 self.colorbar = False
798
804
799 def plot(self):
805 def plot(self):
800
806
801 x = self.times
807 x = self.times
802 xmin = self.min_time
808 xmin = self.min_time
803 xmax = xmin + self.xrange * 60 * 60
809 xmax = xmin + self.xrange * 60 * 60
804 Y = self.data[self.CODE]
810 Y = self.data[self.CODE]
805
811
806 if self.axes[0].firsttime:
812 if self.axes[0].firsttime:
807 for ch in self.data.channels:
813 for ch in self.data.channels:
808 y = Y[ch]
814 y = Y[ch]
809 self.axes[0].plot(x, y, lw=1, label='Ch{}'.format(ch))
815 self.axes[0].plot(x, y, lw=1, label='Ch{}'.format(ch))
810 plt.legend()
816 plt.legend()
811 else:
817 else:
812 for ch in self.data.channels:
818 for ch in self.data.channels:
813 y = Y[ch]
819 y = Y[ch]
814 self.axes[0].lines[ch].set_data(x, y)
820 self.axes[0].lines[ch].set_data(x, y)
815
821
816 self.ymin = numpy.nanmin(Y) - 5
822 self.ymin = numpy.nanmin(Y) - 5
817 self.ymax = numpy.nanmax(Y) + 5
823 self.ymax = numpy.nanmax(Y) + 5
818 self.saveTime = self.min_time
824 self.saveTime = self.min_time
819
825
820
826
821 class PlotSNRData(PlotRTIData):
827 class PlotSNRData(PlotRTIData):
822 '''
828 '''
823 Plot for SNR Data
829 Plot for SNR Data
824 '''
830 '''
825
831
826 CODE = 'snr'
832 CODE = 'snr'
827 colormap = 'jet'
833 colormap = 'jet'
828
834
829
835
830 class PlotDOPData(PlotRTIData):
836 class PlotDOPData(PlotRTIData):
831 '''
837 '''
832 Plot for DOPPLER Data
838 Plot for DOPPLER Data
833 '''
839 '''
834
840
835 CODE = 'dop'
841 CODE = 'dop'
836 colormap = 'jet'
842 colormap = 'jet'
837
843
838
844
839 class PlotSkyMapData(PlotData):
845 class PlotSkyMapData(PlotData):
840 '''
846 '''
841 Plot for meteors detection data
847 Plot for meteors detection data
842 '''
848 '''
843
849
844 CODE = 'param'
850 CODE = 'param'
845
851
846 def setup(self):
852 def setup(self):
847
853
848 self.ncols = 1
854 self.ncols = 1
849 self.nrows = 1
855 self.nrows = 1
850 self.width = 7.2
856 self.width = 7.2
851 self.height = 7.2
857 self.height = 7.2
852 self.nplots = 1
858 self.nplots = 1
853 self.xlabel = 'Zonal Zenith Angle (deg)'
859 self.xlabel = 'Zonal Zenith Angle (deg)'
854 self.ylabel = 'Meridional Zenith Angle (deg)'
860 self.ylabel = 'Meridional Zenith Angle (deg)'
855 self.polar = True
861 self.polar = True
856 self.ymin = -180
862 self.ymin = -180
857 self.ymax = 180
863 self.ymax = 180
858 self.colorbar = False
864 self.colorbar = False
859
865
860 def plot(self):
866 def plot(self):
861
867
862 arrayParameters = numpy.concatenate(self.data['param'])
868 arrayParameters = numpy.concatenate(self.data['param'])
863 error = arrayParameters[:, -1]
869 error = arrayParameters[:, -1]
864 indValid = numpy.where(error == 0)[0]
870 indValid = numpy.where(error == 0)[0]
865 finalMeteor = arrayParameters[indValid, :]
871 finalMeteor = arrayParameters[indValid, :]
866 finalAzimuth = finalMeteor[:, 3]
872 finalAzimuth = finalMeteor[:, 3]
867 finalZenith = finalMeteor[:, 4]
873 finalZenith = finalMeteor[:, 4]
868
874
869 x = finalAzimuth * numpy.pi / 180
875 x = finalAzimuth * numpy.pi / 180
870 y = finalZenith
876 y = finalZenith
871
877
872 ax = self.axes[0]
878 ax = self.axes[0]
873
879
874 if ax.firsttime:
880 if ax.firsttime:
875 ax.plot = ax.plot(x, y, 'bo', markersize=5)[0]
881 ax.plot = ax.plot(x, y, 'bo', markersize=5)[0]
876 else:
882 else:
877 ax.plot.set_data(x, y)
883 ax.plot.set_data(x, y)
878
884
879 dt1 = self.getDateTime(self.min_time).strftime('%y/%m/%d %H:%M:%S')
885 dt1 = self.getDateTime(self.min_time).strftime('%y/%m/%d %H:%M:%S')
880 dt2 = self.getDateTime(self.max_time).strftime('%y/%m/%d %H:%M:%S')
886 dt2 = self.getDateTime(self.max_time).strftime('%y/%m/%d %H:%M:%S')
881 title = 'Meteor Detection Sky Map\n %s - %s \n Number of events: %5.0f\n' % (dt1,
887 title = 'Meteor Detection Sky Map\n %s - %s \n Number of events: %5.0f\n' % (dt1,
882 dt2,
888 dt2,
883 len(x))
889 len(x))
884 self.titles[0] = title
890 self.titles[0] = title
885 self.saveTime = self.max_time
891 self.saveTime = self.max_time
886
892
887
893
888 class PlotParamData(PlotRTIData):
894 class PlotParamData(PlotRTIData):
889 '''
895 '''
890 Plot for data_param object
896 Plot for data_param object
891 '''
897 '''
892
898
893 CODE = 'param'
899 CODE = 'param'
894 colormap = 'seismic'
900 colormap = 'seismic'
895
901
896 def setup(self):
902 def setup(self):
897 self.xaxis = 'time'
903 self.xaxis = 'time'
898 self.ncols = 1
904 self.ncols = 1
899 self.nrows = self.data.shape(self.CODE)[0]
905 self.nrows = self.data.shape(self.CODE)[0]
900 self.nplots = self.nrows
906 self.nplots = self.nrows
901 if self.showSNR:
907 if self.showSNR:
902 self.nrows += 1
908 self.nrows += 1
903 self.nplots += 1
909 self.nplots += 1
904
910
905 self.ylabel = 'Height [km]'
911 self.ylabel = 'Height [km]'
906 if not self.titles:
912 if not self.titles:
907 self.titles = self.data.parameters \
913 self.titles = self.data.parameters \
908 if self.data.parameters else ['Param {}'.format(x) for x in xrange(self.nrows)]
914 if self.data.parameters else ['Param {}'.format(x) for x in xrange(self.nrows)]
909 if self.showSNR:
915 if self.showSNR:
910 self.titles.append('SNR')
916 self.titles.append('SNR')
911
917
912 def plot(self):
918 def plot(self):
913 self.data.normalize_heights()
919 self.data.normalize_heights()
914 self.x = self.times
920 self.x = self.times
915 self.y = self.data.heights
921 self.y = self.data.heights
916 if self.showSNR:
922 if self.showSNR:
917 self.z = numpy.concatenate(
923 self.z = numpy.concatenate(
918 (self.data[self.CODE], self.data['snr'])
924 (self.data[self.CODE], self.data['snr'])
919 )
925 )
920 else:
926 else:
921 self.z = self.data[self.CODE]
927 self.z = self.data[self.CODE]
922
928
923 self.z = numpy.ma.masked_invalid(self.z)
929 self.z = numpy.ma.masked_invalid(self.z)
924
930
925 if self.decimation is None:
931 if self.decimation is None:
926 x, y, z = self.fill_gaps(self.x, self.y, self.z)
932 x, y, z = self.fill_gaps(self.x, self.y, self.z)
927 else:
933 else:
928 x, y, z = self.fill_gaps(*self.decimate())
934 x, y, z = self.fill_gaps(*self.decimate())
929
935
930 for n, ax in enumerate(self.axes):
936 for n, ax in enumerate(self.axes):
931
937
932 self.zmax = self.zmax if self.zmax is not None else numpy.max(
938 self.zmax = self.zmax if self.zmax is not None else numpy.max(
933 self.z[n])
939 self.z[n])
934 self.zmin = self.zmin if self.zmin is not None else numpy.min(
940 self.zmin = self.zmin if self.zmin is not None else numpy.min(
935 self.z[n])
941 self.z[n])
936
942
937 if ax.firsttime:
943 if ax.firsttime:
938 if self.zlimits is not None:
944 if self.zlimits is not None:
939 self.zmin, self.zmax = self.zlimits[n]
945 self.zmin, self.zmax = self.zlimits[n]
940
946
941 ax.plt = ax.pcolormesh(x, y, z[n].T * self.factors[n],
947 ax.plt = ax.pcolormesh(x, y, z[n].T * self.factors[n],
942 vmin=self.zmin,
948 vmin=self.zmin,
943 vmax=self.zmax,
949 vmax=self.zmax,
944 cmap=self.cmaps[n]
950 cmap=self.cmaps[n]
945 )
951 )
946 else:
952 else:
947 if self.zlimits is not None:
953 if self.zlimits is not None:
948 self.zmin, self.zmax = self.zlimits[n]
954 self.zmin, self.zmax = self.zlimits[n]
949 ax.collections.remove(ax.collections[0])
955 ax.collections.remove(ax.collections[0])
950 ax.plt = ax.pcolormesh(x, y, z[n].T * self.factors[n],
956 ax.plt = ax.pcolormesh(x, y, z[n].T * self.factors[n],
951 vmin=self.zmin,
957 vmin=self.zmin,
952 vmax=self.zmax,
958 vmax=self.zmax,
953 cmap=self.cmaps[n]
959 cmap=self.cmaps[n]
954 )
960 )
955
961
956 self.saveTime = self.min_time
962 self.saveTime = self.min_time
957
963
958
964
959 class PlotOutputData(PlotParamData):
965 class PlotOutputData(PlotParamData):
960 '''
966 '''
961 Plot data_output object
967 Plot data_output object
962 '''
968 '''
963
969
964 CODE = 'output'
970 CODE = 'output'
965 colormap = 'seismic'
971 colormap = 'seismic'
@@ -1,69 +1,70
1 '''
1 '''
2 Created on Jul 16, 2014
2 Created on Jul 16, 2014
3
3
4 @author: Miguel Urco
4 @author: Miguel Urco
5 '''
5 '''
6
6
7 import os
7 import os
8 from setuptools import setup, Extension
8 from setuptools import setup, Extension
9 from setuptools.command.build_ext import build_ext as _build_ext
9 from setuptools.command.build_ext import build_ext as _build_ext
10 from schainpy import __version__
10 from schainpy import __version__
11
11
12 class build_ext(_build_ext):
12 class build_ext(_build_ext):
13 def finalize_options(self):
13 def finalize_options(self):
14 _build_ext.finalize_options(self)
14 _build_ext.finalize_options(self)
15 # Prevent numpy from thinking it is still in its setup process:
15 # Prevent numpy from thinking it is still in its setup process:
16 __builtins__.__NUMPY_SETUP__ = False
16 __builtins__.__NUMPY_SETUP__ = False
17 import numpy
17 import numpy
18 self.include_dirs.append(numpy.get_include())
18 self.include_dirs.append(numpy.get_include())
19
19
20 setup(name = "schainpy",
20 setup(name = "schainpy",
21 version = __version__,
21 version = __version__,
22 description = "Python tools to read, write and process Jicamarca data",
22 description = "Python tools to read, write and process Jicamarca data",
23 author = "Miguel Urco",
23 author = "Miguel Urco",
24 author_email = "miguel.urco@jro.igp.gob.pe",
24 author_email = "miguel.urco@jro.igp.gob.pe",
25 url = "http://jro.igp.gob.pe",
25 url = "http://jro.igp.gob.pe",
26 packages = {'schainpy',
26 packages = {'schainpy',
27 'schainpy.model',
27 'schainpy.model',
28 'schainpy.model.data',
28 'schainpy.model.data',
29 'schainpy.model.graphics',
29 'schainpy.model.graphics',
30 'schainpy.model.io',
30 'schainpy.model.io',
31 'schainpy.model.proc',
31 'schainpy.model.proc',
32 'schainpy.model.serializer',
32 'schainpy.model.serializer',
33 'schainpy.model.utils',
33 'schainpy.model.utils',
34 'schainpy.utils',
34 'schainpy.utils',
35 'schainpy.gui',
35 'schainpy.gui',
36 'schainpy.gui.figures',
36 'schainpy.gui.figures',
37 'schainpy.gui.viewcontroller',
37 'schainpy.gui.viewcontroller',
38 'schainpy.gui.viewer',
38 'schainpy.gui.viewer',
39 'schainpy.gui.viewer.windows',
39 'schainpy.gui.viewer.windows',
40 'schainpy.cli'},
40 'schainpy.cli'},
41 ext_package = 'schainpy',
41 ext_package = 'schainpy',
42 package_data = {'': ['schain.conf.template'],
42 package_data = {'': ['schain.conf.template'],
43 'schainpy.gui.figures': ['*.png', '*.jpg'],
43 'schainpy.gui.figures': ['*.png', '*.jpg'],
44 'schainpy.files': ['*.oga']
44 },
45 },
45 include_package_data = False,
46 include_package_data = False,
46 scripts = ['schainpy/gui/schainGUI'],
47 scripts = ['schainpy/gui/schainGUI'],
47 ext_modules = [
48 ext_modules = [
48 Extension("cSchain", ["schainpy/model/proc/extensions.c"])
49 Extension("cSchain", ["schainpy/model/proc/extensions.c"])
49 ],
50 ],
50 entry_points = {
51 entry_points = {
51 'console_scripts': [
52 'console_scripts': [
52 'schain = schainpy.cli.cli:main',
53 'schain = schainpy.cli.cli:main',
53 ],
54 ],
54 },
55 },
55 cmdclass = {'build_ext': build_ext},
56 cmdclass = {'build_ext': build_ext},
56 setup_requires = ["numpy >= 1.11.2"],
57 setup_requires = ["numpy >= 1.11.2"],
57 install_requires = [
58 install_requires = [
58 "scipy >= 0.14.0",
59 "scipy >= 0.14.0",
59 "h5py >= 2.2.1",
60 "h5py >= 2.2.1",
60 "matplotlib >= 2.0.0",
61 "matplotlib >= 2.0.0",
61 "pyfits >= 3.4",
62 "pyfits >= 3.4",
62 "paramiko >= 2.1.2",
63 "paramiko >= 2.1.2",
63 "paho-mqtt >= 1.2",
64 "paho-mqtt >= 1.2",
64 "zmq",
65 "zmq",
65 "fuzzywuzzy",
66 "fuzzywuzzy",
66 "click",
67 "click",
67 "python-Levenshtein"
68 "python-Levenshtein"
68 ],
69 ],
69 )
70 )
General Comments 0
You need to be logged in to leave comments. Login now