diff --git a/.gitignore b/.gitignore index f8f1a114a0b60291dbc4c43af79b374d066e2dd2..4c9cc242ca00f1f359b8d6017dbba6b13dbec9b4 100644 --- a/.gitignore +++ b/.gitignore @@ -9,6 +9,7 @@ __pycache__ .cache .vagrant .env +*.tar src/python/OBis/build/ src/python/OBis/dist/ src/python/PyBis/build/ diff --git a/src/python/PyBis/pybis/pybis.py b/src/python/PyBis/pybis/pybis.py index 72cba1d2680bdd173b8311828f7d868b77c73d05..348f84ec4aac44605c0014b95980e3305ee3b3b5 100644 --- a/src/python/PyBis/pybis/pybis.py +++ b/src/python/PyBis/pybis/pybis.py @@ -643,8 +643,8 @@ class Openbis: 'new_space(name, description)', 'new_project(space, code, description)', 'new_experiment(type, code, project, props={})', - 'new_sample(type, space, project, experiment)', - 'new_object(type, space, project, experiment)', # 'new_sample(type, space, project, experiment)' alias + 'new_sample(type, space, project, experiment, parents)', + 'new_object(type, space, project, experiment, parents)', # 'new_sample(type, space, project, experiment)' alias 'new_dataset(type, parent, experiment, sample, files=[], folder, props={})', 'new_semantic_annotation(entityType, propertyType)', 'update_sample(sampleId, space, project, experiment, parents, children, components, properties, tagIds, attachments)', @@ -1167,12 +1167,13 @@ class Openbis: data = resp[id] ) - def new_experiment(self, type, code, props=None, **kwargs): + def new_experiment(self, type, code, project, props=None, **kwargs): """ Creates a new experiment of a given experiment type. """ return Experiment( openbis_obj = self, type = self.get_experiment_type(type), + project = project, data = None, props = props, code = code, @@ -2950,7 +2951,12 @@ class AttrHolder(): new_sample.space = 'MATERIALS' new_sample.parents = ['/MATERIALS/YEAST747'] """ - if name in ["parents", "children", "components"]: + if name in ["parents", "parent", "children", "child", "components"]: + if name == "parent": + name = "parents" + if name == "child": + name = "children" + if not isinstance(value, list): value = [value] objs = [] @@ -3029,9 +3035,9 @@ class AttrHolder(): raise KeyError("you can not modify the {}".format(name)) elif name == "code": try: - if self._type.data['autoGeneratedCode']: - raise KeyError("for this {}Type you can not set a code".format(self.entity)) - except AttributeError: + if self._type['autoGeneratedCode']: + raise KeyError("This {}Type has auto-generated code. You cannot set a code".format(self.entity)) + except KeyError: pass self.__dict__['_code'] = value @@ -3415,6 +3421,14 @@ class Space(OpenBisObject): get_objects = get_samples #Alias + def get_sample(self, sample_code): + if is_identifier(sample_code) or is_permid(sample_code): + return self.openbis.get_sample(sample_code) + else: + # we assume we just got the code + return self.openbis.get_sample('/{}/{}'.format(self.code,sample_code) ) + + def get_projects(self, **kwargs): return self.openbis.get_projects(space=self.code, **kwargs) @@ -3597,7 +3611,7 @@ class Experiment(OpenBisObject): """ """ - def __init__(self, openbis_obj, type, data=None, props=None, code=None, **kwargs): + def __init__(self, openbis_obj, type, project=None, data=None, props=None, code=None, **kwargs): self.__dict__['openbis'] = openbis_obj self.__dict__['type'] = type self.__dict__['p'] = PropertyHolder(openbis_obj, type) @@ -3606,6 +3620,9 @@ class Experiment(OpenBisObject): if data is not None: self._set_data(data) + if project is not None: + setattr(self, 'project', project) + if props is not None: for key in props: setattr(self.p, key, props[key]) @@ -3639,7 +3656,9 @@ class Experiment(OpenBisObject): # the list of possible methods/attributes displayed # when invoking TAB-completition return [ - 'props', 'space', 'project', + 'code', 'permId', 'identifier', + 'type', 'project', + 'props.', 'project', 'tags', 'attachments', 'data', 'get_datasets()', 'get_samples()', 'set_tags()', 'add_tags()', 'del_tags()', @@ -3821,6 +3840,13 @@ class Project(OpenBisObject): def get_samples(self, **kwargs): return self.openbis.get_samples(project=self.permId, **kwargs) + def get_sample(self, sample_code): + if is_identifier(sample_code) or is_permid(sample_code): + return self.openbis.get_sample(sample_code) + else: + # we assume we just got the code + return self.openbis.get_sample(project=self, code=sample_code) + get_objects = get_samples # Alias def get_experiments(self): diff --git a/src/python/PyBis/setup.py b/src/python/PyBis/setup.py index 45887cf516fef476037cfcb361dccd285e04f108..70a2255cd572c894f5f99b862ae910009f6b6636 100644 --- a/src/python/PyBis/setup.py +++ b/src/python/PyBis/setup.py @@ -9,7 +9,7 @@ from setuptools import setup setup( name='PyBIS', - version= '1.3', + version= '1.4', description='openBIS connection and interaction, optimized for using with Jupyter', url='https://sissource.ethz.ch/sispub/pybis/', author='Swen Vermeul | ID SIS | ETH Zürich', diff --git a/src/python/PyBis/tests/conftest.py b/src/python/PyBis/tests/conftest.py index ca95953bdc1d6e384c52dc736ab18644b8568564..dca56d7cb7850eebccadf61c860fda60706c20cd 100644 --- a/src/python/PyBis/tests/conftest.py +++ b/src/python/PyBis/tests/conftest.py @@ -22,12 +22,7 @@ def space(): o = Openbis(url=openbis_url, verify_certificates=False) o.login(admin_username, admin_password) - timestamp = time.strftime('%a_%y%m%d_%H%M%S').upper() - space_name = 'test_space_' + timestamp - space = o.new_space(code=space_name) - space.save() - space_exists = o.get_space(code=space_name) + space_exists = o.get_space(code='DEFAULT') yield space_exists - space.delete("testing on {}".format(timestamp)) o.logout() diff --git a/src/python/PyBis/tests/test_experiment.py b/src/python/PyBis/tests/test_experiment.py new file mode 100644 index 0000000000000000000000000000000000000000..10ba05ef05618db415a6034a04d1d98a914245e0 --- /dev/null +++ b/src/python/PyBis/tests/test_experiment.py @@ -0,0 +1,45 @@ +import json +import random +import re + +import pytest +import time +from pybis import DataSet +from pybis import Openbis + + +def test_create_delete_project(space): + o=space.openbis + timestamp = time.strftime('%a_%y%m%d_%H%M%S').upper() + new_code='test_experiment_'+timestamp + + with pytest.raises(TypeError): + # experiments must be assigned to a project + e_new = o.new_experiment( + code=new_code, + type='DEFAULT_EXPERIMENT', + ) + + e_new = o.new_experiment( + code=new_code, + project='DEFAULT', + type='DEFAULT_EXPERIMENT', + ) + assert e_new.project is not None + assert e_new.permId is None + + e_new.save() + + assert e_new.permId is not None + assert e_new.code == new_code.upper() + assert e_new.identifier == '/DEFAULT/DEFAULT/'+new_code.upper() + + e_exists = o.get_experiment('/DEFAULT/DEFAULT/'+new_code.upper()) + assert e_exists is not None + + e_new.delete('delete test experiment '+new_code.upper()) + + with pytest.raises(ValueError): + e_no_longer_exists = o.get_experiment('/DEFAULT/DEFAULT/'+new_code.upper()) + + diff --git a/src/python/PyBis/tests/test_openbis.py b/src/python/PyBis/tests/test_openbis.py index 21c09b70100358a3b8b711e3e093943cc6adb4f3..02f65ba1732c9e9890d9156f79e6597dd1d3cc14 100644 --- a/src/python/PyBis/tests/test_openbis.py +++ b/src/python/PyBis/tests/test_openbis.py @@ -22,17 +22,6 @@ def test_wrong_login(openbis_instance): assert new_instance.token is None assert new_instance.is_session_active() is False -def test_create_delete_space(openbis_instance): - space_name = 'test_space_' + time.strftime('%a_%y%m%d_%H%M%S').upper() - space = openbis_instance.new_space(code=space_name) - space.save() - space_exists = openbis_instance.get_space(code=space_name) - assert space_exists is not None - - space.delete() - with pytest.raises(ValueError): - space_not_exists = openbis_instance.get_space(code=space_name) - def test_cached_token(openbis_instance): openbis_instance.save_token() @@ -46,64 +35,6 @@ def test_cached_token(openbis_instance): assert openbis_instance._get_cached_token() is None -def test_create_sample(openbis_instance): - # given - sample_code = 'test_create_' + time.strftime('%a_%y%m%d_%H%M%S').upper() - sample_type = 'UNKNOWN' - space = 'DEFAULT' - # when - sample = openbis_instance.new_sample(code=sample_code, type=sample_type, space=space) - # then - assert sample is not None - assert sample.space == space - assert sample.code == sample_code - - -def test_get_sample_by_id(openbis_instance): - # given - sample_code = 'test_get_by_id_' + time.strftime('%a_%y%m%d_%H%M%S').upper() - sample = openbis_instance.new_sample(code=sample_code, type='UNKNOWN', space='DEFAULT') - sample.save() - # when - persisted_sample = openbis_instance.get_sample('/DEFAULT/' + sample_code) - # then - assert persisted_sample is not None - assert persisted_sample.permId is not None - -def test_get_sample_by_permid(openbis_instance): - # given - sample_code = 'test_get_by_permId_' + time.strftime('%a_%y%m%d_%H%M%S').upper() - sample = openbis_instance.new_sample(code=sample_code, type='UNKNOWN', space='DEFAULT') - sample.save() - persisted_sample = openbis_instance.get_sample('/DEFAULT/' + sample_code) - permId = persisted_sample.permId - # when - sample_by_permId = openbis_instance.get_sample(permId) - # then - assert sample_by_permId is not None - assert sample_by_permId.permId == permId - - -def test_get_sample_parents(openbis_instance): - id = '/TEST/TEST-SAMPLE-2' - sample = openbis_instance.get_sample(id) - assert sample is not None - assert sample.parents is not None - assert sample.parents[0]['identifier']['identifier'] == '/TEST/TEST-SAMPLE-2-PARENT' - parents = sample.get_parents() - assert isinstance(parents, list) - assert parents[0].ident == '/TEST/TEST-SAMPLE-2-PARENT' - - -def test_get_sample_children(openbis_instance): - id = '/TEST/TEST-SAMPLE-2' - sample = openbis_instance.get_sample(id) - assert sample is not None - assert sample.children is not None - assert sample.children[0]['identifier']['identifier'] == '/TEST/TEST-SAMPLE-2-CHILD-1' - children = sample.get_children() - assert isinstance(children, list) - assert children[0].ident == '/TEST/TEST-SAMPLE-2-CHILD-1' def test_get_dataset_parents(openbis_instance): diff --git a/src/python/PyBis/tests/test_sample.py b/src/python/PyBis/tests/test_sample.py new file mode 100644 index 0000000000000000000000000000000000000000..037bf018be8ac06d5658a5592ca59546be46d226 --- /dev/null +++ b/src/python/PyBis/tests/test_sample.py @@ -0,0 +1,71 @@ +import json +import random +import re + +import pytest +import time +from pybis import DataSet +from pybis import Openbis + +def test_create_delete_sample(space): + o=space.openbis + + sample_type = 'UNKNOWN' + sample = o.new_sample(code='illegal sample name with spaces', type=sample_type, space=space) + with pytest.raises(ValueError): + sample.save() + assert "should not have been created" is None + + timestamp = time.strftime('%a_%y%m%d_%H%M%S').upper() + sample_code = 'test_sample_'+timestamp + sample = o.new_sample(code=sample_code, type=sample_type, space=space) + assert sample is not None + assert sample.space == space + assert sample.code == sample_code + + assert sample.permId is None + sample.save() + + # now there should appear a permId + assert sample.permId is not None + + # get it by permId + sample_by_permId = o.get_sample(sample.permId) + assert sample_by_permId is not None + + sample_by_permId = space.get_sample(sample.permId) + assert sample_by_permId is not None + + + # get it by identifier + sample_by_identifier = o.get_sample(sample.identifier) + assert sample_by_identifier is not None + + sample_by_identifier = space.get_sample(sample.identifier) + assert sample_by_identifier is not None + + sample.delete('sample creation test on '+timestamp) + + +def test_parent_child(space): + o=space.openbis + sample_type = 'UNKNOWN' + timestamp = time.strftime('%a_%y%m%d_%H%M%S').upper() + parent_code = 'parent_sample_{}'.format(timestamp) + sample_parent = o.new_sample(code=parent_code, type=sample_type, space=space) + sample_parent.save() + + child_code='child_sample_{}'.format(timestamp) + sample_child = o.new_sample(code=child_code, type=sample_type, space=space, parent=sample_parent) + sample_child.save() + time.sleep(5) + + ex_sample_parents = sample_child.get_parents() + ex_sample_parent = ex_sample_parents[0] + assert ex_sample_parent.identifier == '/DEFAULT/{}'.format(parent_code).upper() + + ex_sample_children = ex_sample_parent.get_children() + ex_sample_child = ex_sample_children[0] + assert ex_sample_child.identifier == '/DEFAULT/{}'.format(child_code).upper() + + diff --git a/src/vagrant/jupyter-bis/README.md b/src/vagrant/jupyter-bis/README.md index 23a1d760d47f843adac11aa195c7d8d0c2b63c89..acb88a17cedb0b9b24acb06d8481a6ae9e5e0b5e 100644 --- a/src/vagrant/jupyter-bis/README.md +++ b/src/vagrant/jupyter-bis/README.md @@ -67,6 +67,18 @@ ame` `` 10. copy the output and add it to /etc/hosts: `sudo vi /etc/hosts` so that Java is happy 11. `exit` -- log off the virtual machine +## upgrading openBIS + +Before upgrading, make sure your openBIS instance is not running. + +1. download [latest openBIS Sprint release](https://wiki-bsse.ethz.ch/display/bis/Sprint+Releases) +2. make sure your OS is not automatically unzipping it +3. move *.gz file from ~/Download to pybis/src/vagrant/jupyter-bis +4. cd pybis/src/vagrant/jupyter-bis +5. vagrant ssh +6. sudo su - openbis +7. cp /vagrant/openBIS-installation-standard-technologies-S267.0-r39027.tar.gz ~/servers +8. bin/upgrade.sh ## start openBIS and JupyterHub