2015-01-13 19:12:19 +08:00
|
|
|
from collections import OrderedDict
|
2015-01-12 18:51:23 +08:00
|
|
|
import importlib
|
2015-04-05 17:49:41 +08:00
|
|
|
import logging
|
2015-07-14 04:08:20 +08:00
|
|
|
import os
|
|
|
|
import time
|
|
|
|
import re
|
|
|
|
|
|
|
|
import numpy
|
|
|
|
import h5py
|
2015-01-12 18:51:23 +08:00
|
|
|
|
2015-01-17 19:38:20 +08:00
|
|
|
from artiq.protocols.sync_struct import Notifier
|
2015-02-12 00:54:54 +08:00
|
|
|
from artiq.protocols.pc_rpc import Client, BestEffortClient
|
2015-02-04 18:37:57 +08:00
|
|
|
|
|
|
|
|
2015-04-05 18:02:07 +08:00
|
|
|
logger = logging.getLogger(__name__)
|
2015-04-05 17:49:41 +08:00
|
|
|
|
|
|
|
|
2015-07-14 04:08:20 +08:00
|
|
|
def get_hdf5_output(start_time, rid, name):
|
|
|
|
dirname = os.path.join("results",
|
|
|
|
time.strftime("%Y-%m-%d", start_time),
|
|
|
|
time.strftime("%H-%M", start_time))
|
|
|
|
filename = "{:09}-{}.h5".format(rid, name)
|
|
|
|
os.makedirs(dirname, exist_ok=True)
|
|
|
|
return h5py.File(os.path.join(dirname, filename), "w")
|
|
|
|
|
|
|
|
|
|
|
|
def get_last_rid():
|
|
|
|
r = -1
|
|
|
|
try:
|
|
|
|
day_folders = os.listdir("results")
|
|
|
|
except:
|
|
|
|
return r
|
|
|
|
day_folders = filter(lambda x: re.fullmatch('\d\d\d\d-\d\d-\d\d', x),
|
|
|
|
day_folders)
|
|
|
|
for df in day_folders:
|
|
|
|
day_path = os.path.join("results", df)
|
2015-01-13 19:12:19 +08:00
|
|
|
try:
|
2015-07-14 04:08:20 +08:00
|
|
|
minute_folders = os.listdir(day_path)
|
|
|
|
except:
|
|
|
|
continue
|
|
|
|
minute_folders = filter(lambda x: re.fullmatch('\d\d-\d\d', x),
|
|
|
|
minute_folders)
|
|
|
|
for mf in minute_folders:
|
|
|
|
minute_path = os.path.join(day_path, mf)
|
|
|
|
try:
|
|
|
|
h5files = os.listdir(minute_path)
|
|
|
|
except:
|
|
|
|
continue
|
|
|
|
for x in h5files:
|
|
|
|
m = re.fullmatch('(\d\d\d\d\d\d\d\d\d)-.*\.h5', x)
|
|
|
|
rid = int(m.group(1))
|
|
|
|
if rid > r:
|
|
|
|
r = rid
|
|
|
|
return r
|
|
|
|
|
|
|
|
|
|
|
|
_type_to_hdf5 = {
|
|
|
|
int: h5py.h5t.STD_I64BE,
|
|
|
|
float: h5py.h5t.IEEE_F64BE
|
|
|
|
}
|
|
|
|
|
|
|
|
def result_dict_to_hdf5(f, rd):
|
|
|
|
for name, data in rd.items():
|
|
|
|
if isinstance(data, list):
|
|
|
|
el_ty = type(data[0])
|
|
|
|
for d in data:
|
|
|
|
if type(d) != el_ty:
|
|
|
|
raise TypeError("All list elements must have the same"
|
|
|
|
" type for HDF5 output")
|
|
|
|
try:
|
|
|
|
el_ty_h5 = _type_to_hdf5[el_ty]
|
|
|
|
except KeyError:
|
|
|
|
raise TypeError("List element type {} is not supported for"
|
|
|
|
" HDF5 output".format(el_ty))
|
|
|
|
dataset = f.create_dataset(name, (len(data), ), el_ty_h5)
|
|
|
|
dataset[:] = data
|
|
|
|
elif isinstance(data, numpy.ndarray):
|
|
|
|
f.create_dataset(name, data=data)
|
|
|
|
else:
|
|
|
|
ty = type(data)
|
2015-01-13 19:12:19 +08:00
|
|
|
try:
|
2015-07-14 04:08:20 +08:00
|
|
|
ty_h5 = _type_to_hdf5[ty]
|
2015-01-13 19:12:19 +08:00
|
|
|
except KeyError:
|
2015-07-14 04:08:20 +08:00
|
|
|
raise TypeError("Type {} is not supported for HDF5 output"
|
|
|
|
.format(ty))
|
|
|
|
dataset = f.create_dataset(name, (), ty_h5)
|
|
|
|
dataset[()] = data
|
2015-01-12 18:51:23 +08:00
|
|
|
|
2015-01-29 23:48:39 +08:00
|
|
|
|
2015-07-14 04:08:20 +08:00
|
|
|
class ResultDB:
|
|
|
|
def __init__(self):
|
|
|
|
self.rt = Notifier(dict())
|
|
|
|
self.nrt = dict()
|
2015-08-06 18:43:27 +08:00
|
|
|
self.store = set()
|
2015-07-14 04:08:20 +08:00
|
|
|
|
|
|
|
def get(self, key):
|
|
|
|
try:
|
|
|
|
return self.nrt[key]
|
|
|
|
except KeyError:
|
|
|
|
return self.rt[key].read
|
2015-01-12 18:51:23 +08:00
|
|
|
|
2015-08-06 18:43:27 +08:00
|
|
|
def set_store(self, key, store):
|
|
|
|
if store:
|
|
|
|
self.store.add(key)
|
|
|
|
else:
|
|
|
|
self.store.discard(key)
|
|
|
|
|
2015-02-04 18:37:57 +08:00
|
|
|
def write_hdf5(self, f):
|
2015-08-06 18:43:27 +08:00
|
|
|
result_dict_to_hdf5(
|
|
|
|
f, {k: v for k, v in self.rt.read.items() if k in self.store})
|
|
|
|
result_dict_to_hdf5(
|
|
|
|
f, {k: v for k, v in self.nrt.items() if k in self.store})
|
2015-02-04 18:37:57 +08:00
|
|
|
|
2015-01-12 18:51:23 +08:00
|
|
|
|
2015-07-17 16:49:55 +08:00
|
|
|
def _create_device(desc, dmgr):
|
2015-02-05 22:53:31 +08:00
|
|
|
ty = desc["type"]
|
|
|
|
if ty == "local":
|
|
|
|
module = importlib.import_module(desc["module"])
|
|
|
|
device_class = getattr(module, desc["class"])
|
2015-07-14 04:08:20 +08:00
|
|
|
return device_class(dmgr, **desc["arguments"])
|
2015-02-05 22:53:31 +08:00
|
|
|
elif ty == "controller":
|
2015-02-12 00:54:54 +08:00
|
|
|
if desc["best_effort"]:
|
|
|
|
cl = BestEffortClient
|
|
|
|
else:
|
|
|
|
cl = Client
|
|
|
|
return cl(desc["host"], desc["port"], desc["target_name"])
|
2015-02-05 22:53:31 +08:00
|
|
|
else:
|
|
|
|
raise ValueError("Unsupported type in device DB: " + ty)
|
2015-01-12 18:51:23 +08:00
|
|
|
|
|
|
|
|
2015-07-14 04:08:20 +08:00
|
|
|
class DeviceManager:
|
|
|
|
"""Handles creation and destruction of local device drivers and controller
|
|
|
|
RPC clients."""
|
|
|
|
def __init__(self, ddb, virtual_devices=dict()):
|
2015-01-12 18:51:23 +08:00
|
|
|
self.ddb = ddb
|
2015-07-14 04:08:20 +08:00
|
|
|
self.virtual_devices = virtual_devices
|
2015-01-12 18:51:23 +08:00
|
|
|
self.active_devices = OrderedDict()
|
|
|
|
|
2015-10-04 18:29:39 +08:00
|
|
|
def get_ddb(self):
|
|
|
|
"""Returns the full contents of the device database."""
|
|
|
|
return self.ddb.get_ddb()
|
|
|
|
|
2015-07-14 04:08:20 +08:00
|
|
|
def get(self, name):
|
|
|
|
"""Get the device driver or controller client corresponding to a
|
|
|
|
device database entry."""
|
|
|
|
if name in self.virtual_devices:
|
|
|
|
return self.virtual_devices[name]
|
2015-01-12 18:51:23 +08:00
|
|
|
if name in self.active_devices:
|
|
|
|
return self.active_devices[name]
|
|
|
|
else:
|
2015-07-14 04:08:20 +08:00
|
|
|
desc = self.ddb.get(name)
|
2015-01-12 18:51:23 +08:00
|
|
|
while isinstance(desc, str):
|
|
|
|
# alias
|
2015-07-14 04:08:20 +08:00
|
|
|
desc = self.ddb.get(desc)
|
2015-07-17 16:49:55 +08:00
|
|
|
dev = _create_device(desc, self)
|
2015-01-12 18:51:23 +08:00
|
|
|
self.active_devices[name] = dev
|
|
|
|
return dev
|
|
|
|
|
2015-03-20 19:01:34 +08:00
|
|
|
def close_devices(self):
|
2015-01-12 18:51:23 +08:00
|
|
|
"""Closes all active devices, in the opposite order as they were
|
2015-03-20 19:01:34 +08:00
|
|
|
requested."""
|
2015-01-12 18:51:23 +08:00
|
|
|
for dev in reversed(list(self.active_devices.values())):
|
2015-04-05 17:49:41 +08:00
|
|
|
try:
|
|
|
|
if isinstance(dev, (Client, BestEffortClient)):
|
|
|
|
dev.close_rpc()
|
|
|
|
elif hasattr(dev, "close"):
|
|
|
|
dev.close()
|
|
|
|
except Exception as e:
|
|
|
|
logger.warning("Exception %r when closing device %r", e, dev)
|
2015-04-05 10:40:31 +08:00
|
|
|
self.active_devices.clear()
|