models.py
389 lines
| 14.7 KiB
| text/x-python
|
PythonLexer
|
r263 | import json | ||
import requests | ||||
|
r6 | from django.db import models | ||
|
r118 | from django.core.validators import MinValueValidator, MaxValueValidator | ||
|
r157 | from django.core.urlresolvers import reverse | ||
|
r123 | |||
|
r263 | from apps.main.models import Configuration | ||
from apps.main.utils import Params | ||||
|
r276 | from .utils import create_jarsfiles | ||
|
r274 | |||
|
r6 | # Create your models here. | ||
|
r123 | EXPERIMENT_TYPE = ( | ||
|
r316 | (0, 'RAW_DATA'), | ||
(1, 'PDATA'), | ||||
) | ||||
|
r123 | |||
DATA_TYPE = ( | ||||
|
r316 | (0, 'SHORT'), | ||
(1, 'FLOAT'), | ||||
) | ||||
|
r123 | |||
|
r263 | DECODE_TYPE = ( | ||
|
r316 | (0, 'None'), | ||
(1, 'TimeDomain'), | ||||
(2, 'FreqDomain'), | ||||
(3, 'InvFreqDomain'), | ||||
) | ||||
|
r263 | |||
|
r316 | FILTER = '{"id":1, "clock": 60, "multiplier": 5, "frequency": 49.92, "f_decimal": 721554506, "fir": 2, "cic_2": 12, "cic_5": 25}' | ||
class JARSFilter(models.Model): | ||||
|
r172 | |||
|
r139 | JARS_NBITS = 32 | ||
|
r172 | |||
|
r316 | name = models.CharField(verbose_name='Name', max_length=60, unique=True, default='') | ||
clock = models.FloatField(verbose_name='Clock In (MHz)', validators=[ | ||||
MinValueValidator(5), MaxValueValidator(75)], null=True, default=60) | ||||
multiplier = models.PositiveIntegerField(verbose_name='Multiplier', validators=[ | ||||
MinValueValidator(1), MaxValueValidator(20)], default=5) | ||||
frequency = models.FloatField(verbose_name='Frequency (MHz)', validators=[ | ||||
MaxValueValidator(150)], null=True, default=49.9200) | ||||
f_decimal = models.BigIntegerField(verbose_name='Frequency (Decimal)', validators=[ | ||||
MinValueValidator(-9223372036854775808), MaxValueValidator(2**JARS_NBITS-1)], null=True, default=721554505) | ||||
cic_2 = models.PositiveIntegerField(verbose_name='CIC2', validators=[ | ||||
MinValueValidator(2), MaxValueValidator(100)], default=10) | ||||
scale_cic_2 = models.PositiveIntegerField(verbose_name='Scale CIC2', validators=[ | ||||
MinValueValidator(0), MaxValueValidator(6)], default=1) | ||||
cic_5 = models.PositiveIntegerField(verbose_name='CIC5', validators=[ | ||||
MinValueValidator(1), MaxValueValidator(100)], default=1) | ||||
scale_cic_5 = models.PositiveIntegerField(verbose_name='Scale CIC5', validators=[ | ||||
MinValueValidator(0), MaxValueValidator(20)], default=5) | ||||
fir = models.PositiveIntegerField(verbose_name='FIR', validators=[ | ||||
MinValueValidator(1), MaxValueValidator(100)], default=6) | ||||
scale_fir = models.PositiveIntegerField(verbose_name='Scale FIR', validators=[ | ||||
MinValueValidator(0), MaxValueValidator(7)], default=3) | ||||
number_taps = models.PositiveIntegerField(verbose_name='Number of taps', validators=[ | ||||
MinValueValidator(1), MaxValueValidator(256)], default=4) | ||||
taps = models.CharField(verbose_name='Taps', max_length=256, default='') | ||||
|
r172 | |||
|
r123 | class Meta: | ||
db_table = 'jars_filters' | ||||
|
r172 | |||
|
r136 | def __unicode__(self): | ||
return u'%s' % (self.name) | ||||
|
r172 | |||
|
r316 | def jsonify(self): | ||
data = {} | ||||
ignored = () | ||||
for field in self._meta.fields: | ||||
if field.name in ignored: | ||||
continue | ||||
data[field.name] = field.value_from_object(self) | ||||
return data | ||||
|
r136 | def parms_to_dict(self): | ||
|
r172 | |||
|
r136 | parameters = {} | ||
|
r172 | |||
|
r316 | parameters['name'] = self.name | ||
parameters['clock'] = float(self.clock) | ||||
parameters['multiplier'] = int(self.multiplier) | ||||
parameters['frequency'] = float(self.frequency) | ||||
parameters['f_decimal'] = int(self.frequency) | ||||
parameters['fir'] = int(self.fir) | ||||
parameters['cic_2'] = int(self.cic_2) | ||||
parameters['cic_5'] = int(self.cic_5) | ||||
|
r172 | |||
|
r136 | return parameters | ||
|
r172 | |||
|
r136 | def dict_to_parms(self, parameters): | ||
|
r172 | |||
|
r316 | self.name = parameters['name'] | ||
self.clock = parameters['clock'] | ||||
self.multiplier = parameters['multiplier'] | ||||
self.frequency = parameters['frequency'] | ||||
self.f_decimal = parameters['f_decimal'] | ||||
self.fir = parameters['fir'] | ||||
self.cic_2 = parameters['cic_2'] | ||||
self.cic_5 = parameters['cic_5'] | ||||
|
r172 | |||
|
r118 | |||
|
r6 | class JARSConfiguration(Configuration): | ||
|
r172 | |||
|
r316 | ADC_RESOLUTION = 8 | ||
|
r118 | PCI_DIO_BUSWIDTH = 32 | ||
|
r316 | HEADER_VERSION = 1103 | ||
BEGIN_ON_START = True | ||||
REFRESH_RATE = 1 | ||||
exp_type = models.PositiveIntegerField( | ||||
verbose_name='Experiment Type', choices=EXPERIMENT_TYPE, default=0) | ||||
cards_number = models.PositiveIntegerField(verbose_name='Number of Cards', validators=[ | ||||
MinValueValidator(1), MaxValueValidator(4)], default=1) | ||||
channels_number = models.PositiveIntegerField(verbose_name='Number of Channels', validators=[ | ||||
MinValueValidator(1), MaxValueValidator(8)], default=5) | ||||
channels = models.CharField( | ||||
verbose_name='Channels', max_length=15, default='1,2,3,4,5') | ||||
data_type = models.PositiveIntegerField( | ||||
verbose_name='Data Type', choices=DATA_TYPE, default=0) | ||||
raw_data_blocks = models.PositiveIntegerField( | ||||
verbose_name='Raw Data Blocks', validators=[MaxValueValidator(5000)], default=60) | ||||
profiles_block = models.PositiveIntegerField( | ||||
verbose_name='Profiles Per Block', default=400) | ||||
acq_profiles = models.PositiveIntegerField( | ||||
verbose_name='Acquired Profiles', default=400) | ||||
ftp_interval = models.PositiveIntegerField( | ||||
verbose_name='FTP Interval', default=60) | ||||
fftpoints = models.PositiveIntegerField( | ||||
verbose_name='FFT Points', default=16) | ||||
cohe_integr_str = models.PositiveIntegerField( | ||||
verbose_name='Coh. Int. Stride', validators=[MinValueValidator(1)], default=30) | ||||
cohe_integr = models.PositiveIntegerField( | ||||
verbose_name='Coherent Integrations', validators=[MinValueValidator(1)], default=30) | ||||
incohe_integr = models.PositiveIntegerField( | ||||
verbose_name='Incoherent Integrations', validators=[MinValueValidator(1)], default=30) | ||||
decode_data = models.PositiveIntegerField( | ||||
verbose_name='Decode Data', choices=DECODE_TYPE, default=0) | ||||
post_coh_int = models.BooleanField( | ||||
verbose_name='Post Coherent Integration', default=False) | ||||
spectral_number = models.PositiveIntegerField( | ||||
verbose_name='# Spectral Combinations', validators=[MinValueValidator(1)], default=1) | ||||
spectral = models.CharField( | ||||
verbose_name='Combinations', max_length=5000, default='[0, 0],') | ||||
create_directory = models.BooleanField( | ||||
verbose_name='Create Directory Per Day', default=True) | ||||
include_expname = models.BooleanField( | ||||
verbose_name='Experiment Name in Directory', default=False) | ||||
|
r157 | #view_raw_data = models.BooleanField(verbose_name='View Raw Data', default=True) | ||
|
r316 | save_ch_dc = models.BooleanField( | ||
verbose_name='Save Channels DC', default=True) | ||||
save_data = models.BooleanField(verbose_name='Save Data', default=True) | ||||
filter_parms = models.CharField( | ||||
max_length=10000, default=FILTER) | ||||
filter = models.ForeignKey( | ||||
'JARSFilter', verbose_name='Filter', null=True, blank=True, on_delete=models.CASCADE) | ||||
|
r6 | |||
class Meta: | ||||
db_table = 'jars_configurations' | ||||
|
r172 | |||
|
r269 | def filter_resolution(self): | ||
|
r316 | filter_parms = json.loads(self.filter_parms) | ||
clock = float(filter_parms['clock']) | ||||
cic_2 = filter_parms['cic_2'] | ||||
cic_5 = filter_parms['cic_5'] | ||||
fir = filter_parms['fir'] | ||||
resolution = round((clock/(cic_2*cic_5*fir)), 2) | ||||
|
r269 | return resolution | ||
|
r263 | def dict_to_parms(self, params, id=None): | ||
|
r316 | |||
|
r263 | if id is not None: | ||
data = Params(params).get_conf(id_conf=id) | ||||
else: | ||||
data = Params(params).get_conf(dtype='jars') | ||||
|
r275 | data['filter_parms'] = params['filter_parms'] | ||
|
r316 | |||
# self.name = data['name'] | ||||
self.exp_type = data['exp_type'] | ||||
|
r275 | #----PDATA---- | ||
if self.exp_type == 1: | ||||
|
r316 | self.incohe_integr = data['incohe_integr'] | ||
|
r275 | self.spectral_number = data['spectral_number'] | ||
|
r316 | self.spectral = data['spectral'] | ||
self.fftpoints = data['fftpoints'] | ||||
self.save_ch_dc = data['save_ch_dc'] | ||||
|
r275 | else: | ||
self.raw_data_blocks = data['raw_data_blocks'] | ||||
|
r316 | #----PDATA---- | ||
self.cards_number = data['cards_number'] | ||||
|
r263 | self.channels_number = data['channels_number'] | ||
|
r316 | self.channels = data['channels'] | ||
self.data_type = data['data_type'] | ||||
self.profiles_block = data['profiles_block'] | ||||
self.acq_profiles = data['acq_profiles'] | ||||
self.ftp_interval = data['ftp_interval'] | ||||
|
r275 | self.cohe_integr_str = data['cohe_integr_str'] | ||
|
r316 | self.cohe_integr = data['cohe_integr'] | ||
|
r275 | #----DECODE---- | ||
|
r316 | self.decode_data = data['decode_data'] | ||
self.post_coh_int = data['post_coh_int'] | ||||
|
r275 | #----DECODE---- | ||
|
r263 | self.create_directory = data['create_directory'] | ||
|
r316 | self.include_expname = data['include_expname'] | ||
self.save_data = data['save_data'] | ||||
self.filter_parms = json.dumps(data['filter_parms']) | ||||
|
r263 | self.save() | ||
|
r172 | |||
|
r276 | def parms_to_text(self, file_format='jars'): | ||
data = self.experiment.parms_to_dict() | ||||
for key in data['configurations']['allIds']: | ||||
if data['configurations']['byId'][key]['device_type'] in ('dds', 'cgs'): | ||||
data['configurations']['allIds'].remove(key) | ||||
data['configurations']['byId'].pop(key) | ||||
elif data['configurations']['byId'][key]['device_type'] == 'jars': | ||||
|
r316 | data['configurations']['byId'][key] = self.parms_to_dict( | ||
)['configurations']['byId'][str(self.pk)] | ||||
|
r276 | elif data['configurations']['byId'][key]['device_type'] == 'rc': | ||
data['configurations']['byId'][key]['pulses'] = '' | ||||
data['configurations']['byId'][key]['delays'] = '' | ||||
|
r316 | rc_ids = [pk for pk in data['configurations']['allIds'] | ||
if data['configurations']['byId'][pk]['device_type'] == 'rc'] | ||||
mix_ids = [pk for pk in rc_ids if data['configurations'] | ||||
['byId'][pk]['mix']] | ||||
|
r276 | if mix_ids: | ||
params = data['configurations']['byId'][mix_ids[0]]['parameters'] | ||||
|
r316 | rc = data['configurations']['byId'][params.split( | ||
'-')[0].split('|')[0]] | ||||
|
r276 | rc['mix'] = True | ||
data['configurations']['byId'][rc['id']] = rc | ||||
|
r316 | elif len(rc_ids) == 0: | ||
|
r276 | self.message = 'File needs RC configuration' | ||
return '' | ||||
json_data = json.dumps(data) | ||||
racp_file, filter_file = create_jarsfiles(json_data) | ||||
|
r316 | if file_format == 'racp': | ||
|
r276 | return racp_file | ||
return filter_file | ||||
|
r263 | def request(self, cmd, method='get', **kwargs): | ||
req = getattr(requests, method)(self.device.url(cmd), **kwargs) | ||||
payload = req.json() | ||||
return payload | ||||
|
r172 | |||
|
r123 | def status_device(self): | ||
|
r172 | |||
|
r209 | try: | ||
|
r263 | payload = self.request('status', | ||
params={'name': self.experiment.name}) | ||||
self.device.status = payload['status'] | ||||
|
r209 | self.device.save() | ||
|
r263 | self.message = payload['message'] | ||
|
r209 | except Exception as e: | ||
self.device.status = 0 | ||||
self.message = str(e) | ||||
self.device.save() | ||||
return False | ||||
return True | ||||
|
r172 | |||
|
r157 | def stop_device(self): | ||
|
r172 | |||
|
r209 | try: | ||
|
r263 | payload = self.request('stop', 'post') | ||
self.device.status = payload['status'] | ||||
|
r209 | self.device.save() | ||
|
r263 | self.message = payload['message'] | ||
|
r209 | except Exception as e: | ||
self.device.status = 0 | ||||
self.message = str(e) | ||||
self.device.save() | ||||
return False | ||||
return True | ||||
|
r172 | |||
|
r123 | def read_device(self): | ||
|
r172 | |||
|
r157 | try: | ||
|
r316 | payload = self.request( | ||
'read', params={'name': self.experiment.name}) | ||||
|
r263 | self.message = 'Configuration loaded' | ||
|
r157 | except: | ||
self.device.status = 0 | ||||
self.device.save() | ||||
self.message = 'Could not read JARS configuration.' | ||||
|
r209 | return False | ||
|
r172 | |||
|
r263 | return payload | ||
|
r172 | |||
|
r123 | def write_device(self): | ||
|
r172 | |||
|
r209 | if self.device.status == 3: | ||
self.message = 'Could not configure device. Software Acquisition is running' | ||||
return False | ||||
|
r157 | data = self.experiment.parms_to_dict() | ||
|
r172 | |||
|
r263 | for key in data['configurations']['allIds']: | ||
if data['configurations']['byId'][key]['device_type'] in ('dds', 'cgs'): | ||||
data['configurations']['allIds'].remove(key) | ||||
data['configurations']['byId'].pop(key) | ||||
elif data['configurations']['byId'][key]['device_type'] == 'rc': | ||||
data['configurations']['byId'][key]['pulses'] = '' | ||||
data['configurations']['byId'][key]['delays'] = '' | ||||
|
r316 | rc_ids = [pk for pk in data['configurations']['allIds'] | ||
if data['configurations']['byId'][pk]['device_type'] == 'rc'] | ||||
if len(rc_ids) == 0: | ||||
|
r263 | self.message = 'Missing RC configuration' | ||
return False | ||||
json_data = json.dumps(data) | ||||
|
r316 | |||
|
r209 | try: | ||
|
r263 | payload = self.request('write', 'post', json=json_data) | ||
self.device.status = payload['status'] | ||||
self.message = payload['message'] | ||||
|
r209 | self.device.save() | ||
if self.device.status == 1: | ||||
return False | ||||
|
r172 | |||
|
r209 | except Exception as e: | ||
self.device.status = 0 | ||||
self.message = str(e) | ||||
self.device.save() | ||||
return False | ||||
|
r172 | |||
|
r209 | return True | ||
|
r172 | |||
|
r158 | def start_device(self): | ||
|
r172 | |||
|
r209 | try: | ||
|
r263 | payload = self.request('start', 'post', | ||
json={'name': self.experiment.name}) | ||||
self.device.status = payload['status'] | ||||
self.message = payload['message'] | ||||
|
r209 | self.device.save() | ||
if self.device.status == 1: | ||||
return False | ||||
except Exception as e: | ||||
self.device.status = 0 | ||||
self.message = str(e) | ||||
self.device.save() | ||||
return False | ||||
return True | ||||
|
r172 | |||
|
r274 | def get_log(self): | ||
payload = None | ||||
try: | ||||
|
r316 | payload = requests.get(self.device.url('get_log'), params={ | ||
'name': self.experiment.name}) | ||||
|
r274 | except: | ||
self.device.status = 0 | ||||
self.device.save() | ||||
self.message = 'Jars API is not running.' | ||||
return False | ||||
|
r316 | |||
|
r274 | self.message = 'Jars API is running' | ||
return payload | ||||
|
r263 | def update_from_file(self, filename): | ||
|
r172 | |||
|
r263 | f = JARSFile(filename) | ||
self.dict_to_parms(f.data) | ||||
|
r157 | self.save() | ||
|
r172 | |||
|
r157 | def get_absolute_url_import(self): | ||
return reverse('url_import_jars_conf', args=[str(self.id)]) | ||||
|
r172 | |||
|
r157 | def get_absolute_url_read(self): | ||
|
r172 | return reverse('url_read_jars_conf', args=[str(self.id)]) | ||
|
r274 | |||
def get_absolute_url_log(self): | ||||
return reverse('url_get_jars_log', args=[str(self.id)]) | ||||