2015-08-07 15:51:56 +08:00
|
|
|
import asyncio
|
2015-02-22 11:34:31 +08:00
|
|
|
import os
|
2015-08-07 15:51:56 +08:00
|
|
|
import tempfile
|
|
|
|
import shutil
|
2015-07-15 16:54:44 +08:00
|
|
|
import logging
|
2015-02-22 11:34:31 +08:00
|
|
|
|
|
|
|
from artiq.protocols.sync_struct import Notifier
|
2016-01-24 10:23:02 +08:00
|
|
|
from artiq.master.worker import (Worker, WorkerInternalException,
|
|
|
|
log_worker_exception)
|
2015-12-09 19:13:57 +08:00
|
|
|
from artiq.tools import get_windows_drives, exc_to_warning
|
2015-02-22 11:34:31 +08:00
|
|
|
|
|
|
|
|
2015-07-15 16:54:44 +08:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2015-11-16 22:46:40 +08:00
|
|
|
async def _get_repository_entries(entry_dict,
|
2016-04-02 23:05:16 +08:00
|
|
|
root, filename, worker_handlers):
|
|
|
|
worker = Worker(worker_handlers)
|
2015-11-14 00:06:52 +08:00
|
|
|
try:
|
2016-01-27 03:31:42 +08:00
|
|
|
description = await worker.examine("scan", os.path.join(root, filename))
|
2016-01-24 10:23:02 +08:00
|
|
|
except:
|
|
|
|
log_worker_exception()
|
|
|
|
raise
|
2015-11-14 00:06:52 +08:00
|
|
|
finally:
|
|
|
|
await worker.close()
|
|
|
|
for class_name, class_desc in description.items():
|
|
|
|
name = class_desc["name"]
|
2015-12-06 17:27:15 +08:00
|
|
|
arginfo = class_desc["arginfo"]
|
2015-11-16 22:46:40 +08:00
|
|
|
if "/" in name:
|
|
|
|
logger.warning("Character '/' is not allowed in experiment "
|
|
|
|
"name (%s)", name)
|
|
|
|
name = name.replace("/", "_")
|
|
|
|
if name in entry_dict:
|
2015-11-14 00:06:52 +08:00
|
|
|
logger.warning("Duplicate experiment name: '%s'", name)
|
|
|
|
basename = name
|
|
|
|
i = 1
|
2015-11-16 22:46:40 +08:00
|
|
|
while name in entry_dict:
|
2015-11-14 00:06:52 +08:00
|
|
|
name = basename + str(i)
|
|
|
|
i += 1
|
|
|
|
entry = {
|
|
|
|
"file": filename,
|
|
|
|
"class_name": class_name,
|
2015-12-06 17:27:15 +08:00
|
|
|
"arginfo": arginfo
|
2015-11-14 00:06:52 +08:00
|
|
|
}
|
2015-11-16 22:46:40 +08:00
|
|
|
entry_dict[name] = entry
|
2015-11-14 00:06:52 +08:00
|
|
|
|
|
|
|
|
2016-04-02 23:05:16 +08:00
|
|
|
async def _scan_experiments(root, worker_handlers, subdir=""):
|
2015-11-16 22:46:40 +08:00
|
|
|
entry_dict = dict()
|
2015-11-14 00:06:52 +08:00
|
|
|
for de in os.scandir(os.path.join(root, subdir)):
|
|
|
|
if de.name.startswith("."):
|
|
|
|
continue
|
|
|
|
if de.is_file() and de.name.endswith(".py"):
|
|
|
|
filename = os.path.join(subdir, de.name)
|
2015-02-22 11:34:31 +08:00
|
|
|
try:
|
2015-11-16 22:46:40 +08:00
|
|
|
await _get_repository_entries(
|
2016-04-02 23:05:16 +08:00
|
|
|
entry_dict, root, filename, worker_handlers)
|
2016-01-24 10:23:02 +08:00
|
|
|
except Exception as exc:
|
|
|
|
logger.warning("Skipping file '%s'", filename,
|
|
|
|
exc_info=not isinstance(exc, WorkerInternalException))
|
2015-11-14 00:06:52 +08:00
|
|
|
if de.is_dir():
|
|
|
|
subentries = await _scan_experiments(
|
2016-04-02 23:05:16 +08:00
|
|
|
root, worker_handlers,
|
2015-11-14 00:06:52 +08:00
|
|
|
os.path.join(subdir, de.name))
|
|
|
|
entries = {de.name + "/" + k: v for k, v in subentries.items()}
|
2015-11-16 22:46:40 +08:00
|
|
|
entry_dict.update(entries)
|
|
|
|
return entry_dict
|
2015-02-22 11:34:31 +08:00
|
|
|
|
|
|
|
|
2015-07-15 16:54:44 +08:00
|
|
|
def _sync_explist(target, source):
|
|
|
|
for k in list(target.read.keys()):
|
|
|
|
if k not in source:
|
|
|
|
del target[k]
|
|
|
|
for k in source.keys():
|
|
|
|
if k not in target.read or target.read[k] != source[k]:
|
|
|
|
target[k] = source[k]
|
|
|
|
|
|
|
|
|
2015-12-06 18:39:27 +08:00
|
|
|
class ExperimentDB:
|
2016-04-02 23:05:16 +08:00
|
|
|
def __init__(self, repo_backend, worker_handlers):
|
2015-12-06 18:39:27 +08:00
|
|
|
self.repo_backend = repo_backend
|
2016-04-02 23:05:16 +08:00
|
|
|
self.worker_handlers = worker_handlers
|
2015-08-07 15:51:56 +08:00
|
|
|
|
2015-12-06 18:39:27 +08:00
|
|
|
self.cur_rev = self.repo_backend.get_head_rev()
|
|
|
|
self.repo_backend.request_rev(self.cur_rev)
|
2015-07-15 16:54:44 +08:00
|
|
|
self.explist = Notifier(dict())
|
2015-07-18 00:55:48 +08:00
|
|
|
self._scanning = False
|
2015-02-22 11:34:31 +08:00
|
|
|
|
2016-03-18 00:40:17 +08:00
|
|
|
self.status = Notifier({
|
|
|
|
"scanning": False,
|
|
|
|
"cur_rev": self.cur_rev
|
|
|
|
})
|
|
|
|
|
2015-08-08 11:44:19 +08:00
|
|
|
def close(self):
|
|
|
|
# The object cannot be used anymore after calling this method.
|
2015-12-06 18:39:27 +08:00
|
|
|
self.repo_backend.release_rev(self.cur_rev)
|
2015-08-08 11:44:19 +08:00
|
|
|
|
2015-12-06 18:39:27 +08:00
|
|
|
async def scan_repository(self, new_cur_rev=None):
|
2015-07-18 00:55:48 +08:00
|
|
|
if self._scanning:
|
|
|
|
return
|
|
|
|
self._scanning = True
|
2016-03-18 00:40:17 +08:00
|
|
|
self.status["scanning"] = True
|
2015-08-08 23:36:12 +08:00
|
|
|
try:
|
|
|
|
if new_cur_rev is None:
|
2015-12-06 18:39:27 +08:00
|
|
|
new_cur_rev = self.repo_backend.get_head_rev()
|
|
|
|
wd, _ = self.repo_backend.request_rev(new_cur_rev)
|
|
|
|
self.repo_backend.release_rev(self.cur_rev)
|
2015-08-08 23:36:12 +08:00
|
|
|
self.cur_rev = new_cur_rev
|
2016-03-18 00:40:17 +08:00
|
|
|
self.status["cur_rev"] = new_cur_rev
|
2016-04-02 23:05:16 +08:00
|
|
|
new_explist = await _scan_experiments(wd, self.worker_handlers)
|
2015-08-08 23:36:12 +08:00
|
|
|
|
|
|
|
_sync_explist(self.explist, new_explist)
|
|
|
|
finally:
|
|
|
|
self._scanning = False
|
2016-03-18 00:40:17 +08:00
|
|
|
self.status["scanning"] = False
|
2015-07-18 00:55:48 +08:00
|
|
|
|
2015-12-06 18:39:27 +08:00
|
|
|
def scan_repository_async(self, new_cur_rev=None):
|
2015-12-08 19:24:04 +08:00
|
|
|
asyncio.ensure_future(
|
|
|
|
exc_to_warning(self.scan_repository(new_cur_rev)))
|
2015-08-07 15:51:56 +08:00
|
|
|
|
2015-12-06 17:27:15 +08:00
|
|
|
async def examine(self, filename, use_repository=True):
|
|
|
|
if use_repository:
|
|
|
|
revision = self.cur_rev
|
2015-12-06 18:39:27 +08:00
|
|
|
wd, _ = self.repo_backend.request_rev(revision)
|
2015-12-06 17:27:15 +08:00
|
|
|
filename = os.path.join(wd, filename)
|
2016-04-02 23:05:16 +08:00
|
|
|
worker = Worker(self.worker_handlers)
|
2015-12-06 17:27:15 +08:00
|
|
|
try:
|
2016-01-27 03:31:42 +08:00
|
|
|
description = await worker.examine("examine", filename)
|
2015-12-06 17:27:15 +08:00
|
|
|
finally:
|
|
|
|
await worker.close()
|
|
|
|
if use_repository:
|
2015-12-06 18:39:27 +08:00
|
|
|
self.repo_backend.release_rev(revision)
|
2015-12-06 17:27:15 +08:00
|
|
|
return description
|
|
|
|
|
2015-12-08 19:24:04 +08:00
|
|
|
def list_directory(self, directory):
|
2015-12-09 19:13:57 +08:00
|
|
|
r = []
|
|
|
|
prefix = ""
|
|
|
|
if not directory:
|
|
|
|
if os.name == "nt":
|
|
|
|
drives = get_windows_drives()
|
|
|
|
return [drive + ":\\" for drive in drives]
|
|
|
|
else:
|
|
|
|
directory = "/"
|
|
|
|
prefix = "/"
|
|
|
|
for de in os.scandir(directory):
|
|
|
|
if de.is_dir():
|
|
|
|
r.append(prefix + de.name + os.path.sep)
|
|
|
|
else:
|
|
|
|
r.append(prefix + de.name)
|
|
|
|
return r
|
2015-12-08 19:24:04 +08:00
|
|
|
|
2015-08-07 15:51:56 +08:00
|
|
|
|
|
|
|
class FilesystemBackend:
|
|
|
|
def __init__(self, root):
|
|
|
|
self.root = os.path.abspath(root)
|
|
|
|
|
|
|
|
def get_head_rev(self):
|
|
|
|
return "N/A"
|
|
|
|
|
|
|
|
def request_rev(self, rev):
|
2015-08-08 11:08:04 +08:00
|
|
|
return self.root, None
|
2015-08-07 15:51:56 +08:00
|
|
|
|
|
|
|
def release_rev(self, rev):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class _GitCheckout:
|
|
|
|
def __init__(self, git, rev):
|
|
|
|
self.path = tempfile.mkdtemp()
|
2015-08-08 11:08:04 +08:00
|
|
|
commit = git.get(rev)
|
|
|
|
git.checkout_tree(commit, directory=self.path)
|
|
|
|
self.message = commit.message.strip()
|
2015-08-07 15:51:56 +08:00
|
|
|
self.ref_count = 1
|
|
|
|
logger.info("checked out revision %s into %s", rev, self.path)
|
|
|
|
|
|
|
|
def dispose(self):
|
|
|
|
logger.info("disposing of checkout in folder %s", self.path)
|
|
|
|
shutil.rmtree(self.path)
|
|
|
|
|
|
|
|
|
|
|
|
class GitBackend:
|
|
|
|
def __init__(self, root):
|
|
|
|
# lazy import - make dependency optional
|
|
|
|
import pygit2
|
|
|
|
|
|
|
|
self.git = pygit2.Repository(root)
|
|
|
|
self.checkouts = dict()
|
|
|
|
|
|
|
|
def get_head_rev(self):
|
|
|
|
return str(self.git.head.target)
|
|
|
|
|
|
|
|
def request_rev(self, rev):
|
|
|
|
if rev in self.checkouts:
|
|
|
|
co = self.checkouts[rev]
|
|
|
|
co.ref_count += 1
|
|
|
|
else:
|
|
|
|
co = _GitCheckout(self.git, rev)
|
|
|
|
self.checkouts[rev] = co
|
2015-08-08 11:08:04 +08:00
|
|
|
return co.path, co.message
|
2015-08-07 15:51:56 +08:00
|
|
|
|
|
|
|
def release_rev(self, rev):
|
|
|
|
co = self.checkouts[rev]
|
|
|
|
co.ref_count -= 1
|
|
|
|
if not co.ref_count:
|
|
|
|
co.dispose()
|
|
|
|
del self.checkouts[rev]
|