|
@ -0,0 +1,115 @@ |
|
|
|
|
|
import pathlib as pl |
|
|
|
|
|
import json |
|
|
|
|
|
import os |
|
|
|
|
|
import multiprocessing |
|
|
|
|
|
import threading |
|
|
|
|
|
|
|
|
|
|
|
from . import batch |
|
|
|
|
|
|
|
|
|
|
|
class Plan: |
|
|
|
|
|
def __init__(self, experiment=None, lock=None): |
|
|
|
|
|
self.experiment = None |
|
|
|
|
|
self.file = None |
|
|
|
|
|
self.pending_instances = [] |
|
|
|
|
|
self.assigned_instances = [] |
|
|
|
|
|
self.__lock = threading.Lock() if lock == None else lock |
|
|
|
|
|
|
|
|
|
|
|
if experiment: |
|
|
|
|
|
self.create(experiment) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def create(self, experiment): |
|
|
|
|
|
self.experiment = pl.Path(experiment).resolve() |
|
|
|
|
|
self.__set_file() |
|
|
|
|
|
|
|
|
|
|
|
if self.__is_finished(): |
|
|
|
|
|
self.__create() |
|
|
|
|
|
else: |
|
|
|
|
|
self.__load() |
|
|
|
|
|
|
|
|
|
|
|
def __create(self): |
|
|
|
|
|
with open(self.experiment, "r") as expf: |
|
|
|
|
|
exp_obj = json.loads(expf.read()) |
|
|
|
|
|
|
|
|
|
|
|
instances = batch.load(pl.Path(exp_obj["batch"]).resolve()) |
|
|
|
|
|
|
|
|
|
|
|
self.pending_instances = instances |
|
|
|
|
|
self.__update_file() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def __set_file(self): |
|
|
|
|
|
if self.experiment == None: |
|
|
|
|
|
self.file = None |
|
|
|
|
|
else: |
|
|
|
|
|
exp_path = pl.Path(self.experiment) |
|
|
|
|
|
self.file = exp_path.parent / (exp_path.stem + ".plan") |
|
|
|
|
|
|
|
|
|
|
|
def __load(self): |
|
|
|
|
|
self.pending_instances = [] |
|
|
|
|
|
self.assigned_instances = [] |
|
|
|
|
|
|
|
|
|
|
|
if not self.file.is_file(): |
|
|
|
|
|
return |
|
|
|
|
|
|
|
|
|
|
|
with open(self.file, "r") as pfile: |
|
|
|
|
|
content = json.loads(pfile.read()) |
|
|
|
|
|
|
|
|
|
|
|
if "assigned" in content: |
|
|
|
|
|
self.assigned_instances = content["assigned"] |
|
|
|
|
|
|
|
|
|
|
|
if "pending" in content: |
|
|
|
|
|
self.pending_instances = content["pending"] |
|
|
|
|
|
|
|
|
|
|
|
def __is_finished(self): |
|
|
|
|
|
return False if self.file.is_file() else True |
|
|
|
|
|
|
|
|
|
|
|
def next(self): |
|
|
|
|
|
|
|
|
|
|
|
with self.__lock: |
|
|
|
|
|
self.__load() |
|
|
|
|
|
|
|
|
|
|
|
if len(self.pending_instances) == 0: |
|
|
|
|
|
return None |
|
|
|
|
|
|
|
|
|
|
|
next_instance = self.pending_instances.pop() |
|
|
|
|
|
self.assigned_instances.append(next_instance) |
|
|
|
|
|
|
|
|
|
|
|
self.__update_file() |
|
|
|
|
|
|
|
|
|
|
|
return next_instance |
|
|
|
|
|
|
|
|
|
|
|
def done_with(self, instance): |
|
|
|
|
|
|
|
|
|
|
|
with self.__lock: |
|
|
|
|
|
self.__load() |
|
|
|
|
|
|
|
|
|
|
|
if instance in self.assigned_instances: |
|
|
|
|
|
self.assigned_instances.remove(instance) |
|
|
|
|
|
|
|
|
|
|
|
self.__update_file() |
|
|
|
|
|
|
|
|
|
|
|
def __update_file(self): |
|
|
|
|
|
content = {} |
|
|
|
|
|
|
|
|
|
|
|
if len(self.assigned_instances) > 0: |
|
|
|
|
|
content["assigned"] = list(map(str, self.assigned_instances)) |
|
|
|
|
|
|
|
|
|
|
|
if len(self.pending_instances) > 0: |
|
|
|
|
|
content["pending"] = list(map(str, self.pending_instances)) |
|
|
|
|
|
|
|
|
|
|
|
if content: |
|
|
|
|
|
with open(self.file, "w") as pfile: |
|
|
|
|
|
pfile.write(json.dumps(content)) |
|
|
|
|
|
|
|
|
|
|
|
elif self.file.is_file(): |
|
|
|
|
|
self.file.unlink() |
|
|
|
|
|
|
|
|
|
|
|
def __del__(self): |
|
|
|
|
|
|
|
|
|
|
|
with self.__lock: |
|
|
|
|
|
self.__load() |
|
|
|
|
|
|
|
|
|
|
|
self.pending_instances.extend(self.assigned_instances) |
|
|
|
|
|
self.assigned_instances = [] |
|
|
|
|
|
|
|
|
|
|
|
self.__update_file() |