Skip to content
Snippets Groups Projects
pybis.py 98.8 KiB
Newer Older
  • Learn to ignore specific revisions
  • #!/usr/bin/env python
    # -*- coding: utf-8 -*-
    
    """
    pybis.py
    
    
    Swen Vermeul's avatar
    Swen Vermeul committed
    Work with openBIS from Python.
    
    import requests
    
    from requests.packages.urllib3.exceptions import InsecureRequestWarning
    requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
    
    
    import json
    import re
    
    from urllib.parse import urlparse
    
    import zlib
    
    from pybis.utils import parse_jackson, check_datatype, split_identifier, format_timestamp, is_identifier, is_permid
    from pybis.property import PropertyHolder, PropertyAssignments
    from pybis.masterdata import Vocabulary
    
    import pandas as pd
    from pandas import DataFrame, Series
    
    
    import threading
    from threading import Thread
    from queue import Queue
    
    DROPBOX_PLUGIN = "jupyter-uploader-api"
    
    def _definitions(entity):
    
            "Space": {
                "attrs_new": "code description".split(),
                "attrs_up": "description".split(),
                "attrs": "code permId description registrator registrationDate modificationDate".split(), "identifier": "spaceId",
            },
            "Project": {
                "attrs_new": "code description space attachments".split(),
                "attrs_up": "description space attachments".split(),
                "attrs": "code description permId identifier space leader registrator registrationDate modifier modificationDate attachments".split(),
                "multi": "".split(),
                "identifier": "projectId",
            },
            "Experiment": {
    
                "attrs_new": "code type project tags attachments".split(),
                "attrs_up": "project tags attachments".split(),
                "attrs": "code permId identifier type project tags attachments".split(),
    
                "multi": "tags attachments".split(),
                "identifier": "experimentId",
            },
    
                "attrs_new": "code type space project experiment tags attachments".split(),
                "attrs_up": "space project experiment tags attachments".split(),
                "attrs": "code permId identifier type space project experiment tags attachments".split(),
    
                "ids2type": {
                    'parentIds': { 'permId': { '@type': 'as.dto.sample.id.SamplePermId' } },
                    'childIds':  { 'permId': { '@type': 'as.dto.sample.id.SamplePermId' } },
                    'componentIds': { 'permId': {'@type': 'as.dto.sample.id.SamplePermId' } },
                },
                "identifier": "sampleId",
                "cre_type": "as.dto.sample.create.SampleCreation",
                "multi": "parents children components tags".split(),
            },
            "DataSet": {
    
                "attrs_new": "type experiment sample parents children container components tags".split(),
                "attrs_up": "experiment sample parents children container components tags".split(),
    
                "attrs": "code permId type experiment sample parents children container components tags".split(),
    
                    'parentIds':     { 'permId': { '@type': 'as.dto.dataset.id.DataSetPermId' } },
                    'childIds':      { 'permId': { '@type': 'as.dto.dataset.id.DataSetPermId' } }, 
                    'componentIds':  { 'permId': { '@type': 'as.dto.dataset.id.DataSetPermId' } }, 
                    'containerIds':  { 'permId': { '@type': 'as.dto.dataset.id.DataSetPermId' } }, 
    
                "multi": "".split(),
    
            "Material": {
                "attrs_new": "code description type creation tags".split(),
                "attrs" : "code description type creation registrator tags".split()
            },
            "Tag": {
                "attrs_new": "code description experiments samples dataSets materials".split(),
                "attrs": "code description experiments samples dataSets materials registrationDate".split(),
            },
    
                "project"    : "projectId",
    
                "sample"     : "sampleId",
                "samples"    : "sampleIds",
                "dataSet"    : "dataSetId",
                "dataSets"   : "dataSetIds",
                "experiment" : "experimentId",
                "experiments": "experimentIds",
                "material"   : "materialId",
                "materials"  : "materialIds",
    
                "container"  : "containerId",
                "component"  : "componentId",
                "components" : "componentIds",
                "parents"    : "parentIds",
                "children"   : "childIds",
                "tags"       : "tagIds",
            },
            "ids2type": {
                'spaceId': { 'permId': { '@type': 'as.dto.space.id.SpacePermId' } },
                'projectId': { 'permId': { '@type': 'as.dto.project.id.ProjectPermId' } },
                'experimentId': { 'permId': { '@type': 'as.dto.experiment.id.ExperimentPermId' } },
                'tagIds': { 'code': { '@type': 'as.dto.tag.id.TagCode' } },
            },
        }
    
        return entities[entity]
    
    search_criteria = {
    
        "space":      "as.dto.space.search.SpaceSearchCriteria",
        "project":    "as.dto.project.search.ProjectSearchCriteria",
        "experiment": "as.dto.experiment.search.ExperimentSearchCriteria",
    
        "sample":     "as.dto.sample.search.SampleSearchCriteria",
        "dataset":    "as.dto.dataset.search.DataSetSearchCriteria",
    
        "code":       "as.dto.common.search.CodeSearchCriteria",
    
    Swen Vermeul's avatar
    Swen Vermeul committed
        "sample_type":"as.dto.sample.search.SampleTypeSearchCriteria",
    
    Swen Vermeul's avatar
    Swen Vermeul committed
    fetch_option = {
    
        "space":        { "@type": "as.dto.space.fetchoptions.SpaceFetchOptions" },
        "project":      { "@type": "as.dto.project.fetchoptions.ProjectFetchOptions" },
    
        "experiment":   { "@type": "as.dto.experiment.fetchoptions.ExperimentFetchOptions" },
    
        "sample":       { "@type": "as.dto.sample.fetchoptions.SampleFetchOptions" },
    
        "samples":       { "@type": "as.dto.sample.fetchoptions.SampleFetchOptions" },
        "dataSets":    {
            "@type": "as.dto.dataset.fetchoptions.DataSetFetchOptions",
            "properties": { "@type": "as.dto.property.fetchoptions.PropertyFetchOptions" },
            "type": { "@type": "as.dto.dataset.fetchoptions.DataSetTypeFetchOptions" },
        },
    
        "physicalData": { "@type": "as.dto.dataset.fetchoptions.PhysicalDataFetchOptions" },
        "linkedData":   { "@type": "as.dto.dataset.fetchoptions.LinkedDataFetchOptions" },
    
    
        "properties":   { "@type": "as.dto.property.fetchoptions.PropertyFetchOptions" },
    
        "propertyAssignments" : {
            "@type" : "as.dto.property.fetchoptions.PropertyAssignmentFetchOptions",
            "propertyType": {
                "@type": "as.dto.property.fetchoptions.PropertyTypeFetchOptions"
            }
        },
    
        "tags":         { "@type": "as.dto.tag.fetchoptions.TagFetchOptions" },
    
        "registrator":  { "@type": "as.dto.person.fetchoptions.PersonFetchOptions" },
        "modifier":     { "@type": "as.dto.person.fetchoptions.PersonFetchOptions" },
        "leader":       { "@type": "as.dto.person.fetchoptions.PersonFetchOptions" },
    
        "attachments":  { "@type": "as.dto.attachment.fetchoptions.AttachmentFetchOptions" },
    
        "attachmentsWithContent": {
            "@type": "as.dto.attachment.fetchoptions.AttachmentFetchOptions",
            "content": {
                "@type": "as.dto.common.fetchoptions.EmptyFetchOptions"
            },
        },
        "history": { "@type": "as.dto.history.fetchoptions.HistoryEntryFetchOptions" },
        "dataStore": { "@type": "as.dto.datastore.fetchoptions.DataStoreFetchOptions" },
    
    def search_request_for_identifier(ident, entity):
    
            search_request = {
                "identifier": ident.upper(),
    
                "@type": "as.dto.{}.id.{}Identifier".format(entity.lower(), entity.capitalize())
    
                "@type": "as.dto.{}.id.{}PermId".format(entity.lower(), entity.capitalize())
    
    def extract_code(obj):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
        if not isinstance(obj, dict):
            return str(obj)
    
    def extract_deletion(obj):
        del_objs = []
        for deleted_object in obj['deletedObjects']:
            del_objs.append({
                "reason": obj['reason'],
                "permId": deleted_object["id"]["permId"],
                "type": deleted_object["id"]["@type"]
            })
        return del_objs
    
    
    def extract_identifier(ident):
        if not isinstance(ident, dict): 
            return str(ident)
        return ident['identifier']
    
    
    def extract_nested_identifier(ident):
        if not isinstance(ident, dict): 
            return str(ident)
        return ident['identifier']['identifier']
    
    
    Swen Vermeul's avatar
    Swen Vermeul committed
    def extract_permid(permid):
        if not isinstance(permid, dict):
            return str(permid)
        return permid['permId']
    
    def extract_nested_permid(permid):
        if not isinstance(permid, dict):
            return str(permid)
        return permid['permId']['permId']
    
    
    def extract_property_assignments(pas):
        pa_strings = []
        for pa in pas:
            if not isinstance(pa['propertyType'], dict):
                pa_strings.append(pa['propertyType'])
            else:
                pa_strings.append(pa['propertyType']['label'])
        return pa_strings
    
    
    def extract_person(person):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
        if not isinstance(person, dict):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
        return person['userId']
    
    def crc32(fileName):
    
        """since Python3 the zlib module returns unsigned integers (2.7: signed int)
        """
    
        prev = 0
        for eachLine in open(fileName,"rb"):
            prev = zlib.crc32(eachLine, prev)
        # return as hex
    
    
    def _create_tagIds(tags=None):
        if tags is None:
            return None
    
        if not isinstance(tags, list):
            tags = [tags]
    
        tagIds = []
        for tag in tags:
            tagIds.append({ "code": tag, "@type": "as.dto.tag.id.TagCode" })
    
    def _tagIds_for_tags(tags=None, action='Add'):
        """creates an action item to add or remove tags. Action is either 'Add', 'Remove' or 'Set'
        """
        if tags is None:
            return
        if not isinstance(tags, list):
            tags = [tags]
    
        items = []
        for tag in tags:
            items.append({
                "code": tag,
                "@type": "as.dto.tag.id.TagCode"
            })
    
        tagIds = {
            "actions": [
                {
                    "items": items,
                    "@type": "as.dto.common.update.ListUpdateAction{}".format(action.capitalize())
                }
            ],
            "@type": "as.dto.common.update.IdListUpdateValue"
        }
    
        return tagIds
    
    
    def _list_update(ids=None, entity=None, action='Add'):
        """creates an action item to add, set or remove ids. 
        """
        if ids is None:
            return
        if not isinstance(ids, list):
            ids = [ids]
    
        items = []
        for ids in ids:
            items.append({
                "code": ids,
                "@type": "as.dto.{}.id.{}Code".format(entity.lower(), entity)
            })
    
        list_update = {
            "actions": [
                {
                    "items": items,
                    "@type": "as.dto.common.update.ListUpdateAction{}".format(action.capitalize())
                }
            ],
            "@type": "as.dto.common.update.IdListUpdateValue"
        }
        return list_update 
    
    
    def _create_typeId(type):
        return {
            "permId": type.upper(),
            "@type": "as.dto.entitytype.id.EntityTypePermId"
        }
    
    
    def _create_projectId(ident):
        match = re.match('/', ident)
        if match:
            return {
                "identifier": ident,
                "@type": "as.dto.project.id.ProjectIdentifier"
            }
        else:
            return { 
                "permId": ident,
                "@type": "as.dto.project.id.ProjectPermId"
            }
    
    
    def _common_search(search_type, value, comparison="StringEqualToValue"):
        sreq = {
            "@type": search_type,                                                                                                       
            "fieldValue": {
                "value": value,
                "@type": "as.dto.common.search.{}".format(comparison)                                      
            }   
        }   
        return sreq
    
    def _criteria_for_code(code):
        return {
            "fieldValue": {
    
                "@type": "as.dto.common.search.StringEqualToValue"
            },
            "@type": "as.dto.common.search.CodeSearchCriteria"
        }
    
    
    def _subcriteria_for_type(code, entity):
    
            "@type": "as.dto.{}.search.{}TypeSearchCriteria".format(entity.lower(), entity),
    
              "criteria": [
                {
                  "@type": "as.dto.common.search.CodeSearchCriteria",
                  "fieldValue": {
                    "value": code.upper(),
                    "@type": "as.dto.common.search.StringEqualToValue"
                  }
                }
              ]
        }
    
    
    def _gen_search_criteria(req):
    
        sreq = {}
        for key, val in req.items():
            if key == "criteria":
                items = []
                for item in req['criteria']:
    
                    items.append(_gen_search_criteria(item))
    
                sreq['criteria'] = items
            elif key == "code":
    
                sreq["criteria"] = [_common_search(
                    "as.dto.common.search.CodeSearchCriteria", val.upper()
                )]
            elif key == "permid":
                sreq["criteria"] = [_common_search(
                    "as.dto.common.search.PermIdSearchCriteria", val
                )]
            elif key == "identifier":
    
                si = split_identifier(val)
    
                sreq["criteria"] = []
                if "space" in si:
                    sreq["criteria"].append(
    
                        _gen_search_criteria({ "space": "Space", "code": si["space"] })
    
                    )
                if "experiment" in si:
                    pass
    
                if "code" in si:
                    sreq["criteria"].append(
                        _common_search(
                            "as.dto.common.search.CodeSearchCriteria", si["code"].upper()
                        ) 
                    )
    
    
            elif key == "operator":
               sreq["operator"] = val.upper() 
            else:
                sreq["@type"] = "as.dto.{}.search.{}SearchCriteria".format(key, val)
        return sreq
    
    
    def _subcriteria_for_tags(tags):
        if not isinstance(tags, list):
            tags = [tags]
    
        criterias = []
        for tag in tags:
            criterias.append({
                "fieldName": "code",
                "fieldType": "ATTRIBUTE",
                "fieldValue": {
                    "value": tag,
                    "@type": "as.dto.common.search.StringEqualToValue"
                },
                "@type": "as.dto.common.search.CodeSearchCriteria"
            })
    
        return {
            "@type": "as.dto.tag.search.TagSearchCriteria",
            "operator": "AND",
            "criteria": criterias
        }
    
    def _subcriteria_for_is_finished(is_finished):
        return {
            "@type": "as.dto.common.search.StringPropertySearchCriteria",
            "fieldName": "FINISHED_FLAG",
            "fieldType": "PROPERTY",
            "fieldValue": {
                "value": is_finished,
                "@type": "as.dto.common.search.StringEqualToValue"
            }
        }
    
    def _subcriteria_for_properties(prop, val):
        return {
            "@type": "as.dto.common.search.StringPropertySearchCriteria",
            "fieldName": prop.upper(),
            "fieldType": "PROPERTY",
            "fieldValue": {
                "value": val,
                "@type": "as.dto.common.search.StringEqualToValue"
            }
        }
    
    
    def _subcriteria_for_permid(permids, entity, parents_or_children=''):
    
    
        if not isinstance(permids, list):
            permids = [permids]
    
        criterias = []
        for permid in permids:
            criterias.append( {
                "@type": "as.dto.common.search.PermIdSearchCriteria",
                "fieldValue": {
                    "value": permid,
                    "@type": "as.dto.common.search.StringEqualToValue"
                },
                "fieldType": "ATTRIBUTE",
                "fieldName": "code"
            } )
    
        criteria = {
            "criteria": criterias,
    
            "@type": "as.dto.{}.search.{}{}SearchCriteria".format(
    
                entity.lower(), entity, parents_or_children
    
    
    def _subcriteria_for_code(code, object_type):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
        if code is not None:
            if is_permid(code):
                fieldname = "permId"
                fieldtype = "as.dto.common.search.PermIdSearchCriteria"
            else:
                fieldname = "code"
                fieldtype = "as.dto.common.search.CodeSearchCriteria"
    
            criteria = {
                "criteria": [
                    {
                        "fieldName": fieldname,
                        "fieldType": "ATTRIBUTE",
                        "fieldValue": {
                            "value": code.upper(),
                            "@type": "as.dto.common.search.StringEqualToValue"
                        },
                        "@type": fieldtype 
                    }
                ],
                "@type": search_criteria[object_type.lower()],
                "operator": "AND"
            }
            return criteria
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            criteria = { "@type": search_criteria[object_type.lower()] }
            return criteria
    
    Swen Vermeul's avatar
    Swen Vermeul committed
    
    
        """Interface for communicating with openBIS. A current version of openBIS is needed.
        (minimum version 16.05).
    
        def __init__(self, url='https://localhost:8443', verify_certificates=True, token=None):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            """Initialize a new connection to an openBIS server.
    
    
            :param host:
    
    
            url_obj = urlparse(url)
            if  url_obj.netloc is None:
                raise ValueError("please provide the url in this format: https://openbis.host.ch:8443")
    
            self.url_obj = url_obj
            self.url     = url_obj.geturl()
            self.port    = url_obj.port
            self.hostname = url_obj.hostname
    
            self.as_v3 = '/openbis/openbis/rmi-application-server-v3.json'
            self.as_v1 = '/openbis/openbis/rmi-general-information-v1.json'
            self.reg_v1 = '/openbis/openbis/rmi-query-v1.json'
    
            self.datastores = []
    
            self.dataset_types = None
            self.sample_types = None
    
            self.token_path = None
    
            # use an existing token, if available
            if self.token is None:
    
                self.token = self._get_cached_token()
    
        @property
        def spaces(self):
            return self.get_spaces()
    
    
        @property
        def projects(self):
            return self.get_projects()
    
    
        def _get_cached_token(self):
    
            """Read the token from the cache, and set the token ivar to it, if there, otherwise None.
            If the token is not valid anymore, delete it. 
            """
            token_path = self.gen_token_path()
    
                    token = f.read()
                    if not self.is_token_valid(token):
    
        def gen_token_path(self, parent_folder=None):
            """generates a path to the token file.
            The token is usually saved in a file called
            ~/.pybis/hostname.token
            """
    
                # save token under ~/.pybis folder
                parent_folder = os.path.join(
                    os.path.expanduser("~"),
                    '.pybis'
                )
            path = os.path.join(parent_folder, self.hostname + '.token')
    
        def save_token(self, token=None, parent_folder=None):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            """ saves the session token to the disk, usually here: ~/.pybis/hostname.token. When a new Openbis instance is created, it tries to read this saved token by default.
    
            if token is None:
                token = self.token
    
            token_path = None;
            if parent_folder is None:
                token_path = self.gen_token_path()
            else:
                token_path = self.gen_token_path(parent_folder)
    
            # create the necessary directories, if they don't exist yet
    
            os.makedirs(os.path.dirname(token_path), exist_ok=True)
            with open(token_path, 'w') as f:
    
                f.write(token)
                self.token_path = token_path
    
    
        def delete_token(self, token_path=None):
    
            """ deletes a stored session token.
            """
    
            if token_path is None:
                token_path = self.token_path
            os.remove(token_path)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
        def _post_request(self, resource, request):
    
            """ internal method, used to handle all post requests and serializing / deserializing
            data
            """
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            if "id" not in request:
                request["id"] = "1"
            if "jsonrpc" not in request:
                request["jsonrpc"] = "2.0"
    
            if request["params"][0] is None:
                raise ValueError("Your session expired, please log in again")
    
            resp = requests.post(
                self.url + resource, 
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                json.dumps(request), 
    
                verify=self.verify_certificates
            )
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                resp = resp.json()
                if 'error' in resp:
                    print(json.dumps(request))
                    raise ValueError('an error has occured: ' + resp['error']['message'] )
                elif 'result' in resp:
                    return resp['result']
    
                else:
                    raise ValueError('request did not return either result nor error')
            else:
                raise ValueError('general error while performing post request')
    
    
        def logout(self):
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            """ Log out of openBIS. After logout, the session token is no longer valid.
    
    
            logout_request = {
                "method":"logout",
                "params":[self.token],
            }
    
            resp = self._post_request(self.as_v3, logout_request)
    
            return resp
    
        def login(self, username=None, password=None, save_token=False):
    
            Expects a username and a password and updates the token (session-ID).
            The token is then used for every request.
    
            Clients may want to store the credentials object in a credentials store after successful login.
    
            Throw a ValueError with the error message if login failed.
            """
    
    
            login_request = {
                "method":"login",
                "params":[username, password],
            }
    
            result = self._post_request(self.as_v3, login_request)
    
            if result is None:
                raise ValueError("login to openBIS failed")
            else:
                self.token = result
    
                if save_token:
                    self.save_token()
                return self.token
    
            """ Get a list of all available datastores. Usually there is only one, but in some cases
            there might be more. If you upload a file, you need to specifiy the datastore you want
            the file uploaded to.
            """
    
            if len(self.datastores) == 0: 
                request = {
                    "method": "listDataStores",
                    "params": [ self.token ],
                }
    
                resp = self._post_request(self.as_v1, request)
    
                if resp is not None:
                    self.datastores = DataFrame(resp)[['code','downloadUrl', 'hostUrl']]
                    return self.datastores
                else:
                    raise ValueError("No datastore found!")
            else:
                return self.datastores
    
    
    
        def get_spaces(self, code=None):
    
            """ Get a list of all available spaces (DataFrame object). To create a sample or a
            dataset, you need to specify in which space it should live.
            """
    
            request = {
                "method": "searchSpaces",
    
                "params": [ self.token, 
                    criteria,
                    options,
                ],
    
            }
            resp = self._post_request(self.as_v3, request)
            if resp is not None:
                spaces = DataFrame(resp['objects'])
                spaces['registrationDate']= spaces['registrationDate'].map(format_timestamp)
                spaces['modificationDate']= spaces['modificationDate'].map(format_timestamp)
                sp = Things(
                    self,
                    'space',
                    spaces[['code', 'description', 'registrationDate', 'modificationDate']]
                )
                return sp
    
                raise ValueError("No spaces found!")
    
    
    
        def get_space(self, spaceId):
    
            """ Returns a Space object for a given identifier (spaceId).
            """
    
    Swen Vermeul's avatar
    Swen Vermeul committed
    
            spaceId = str(spaceId).upper()
    
            fetchopts = { "@type": "as.dto.space.fetchoptions.SpaceFetchOptions" }
            for option in ['registrator']:
                fetchopts[option] = fetch_option[option]
    
    
            request = {
            "method": "getSpaces",
                "params": [ 
                self.token,
                [{ 
                    "permId": spaceId,
                    "@type": "as.dto.space.id.SpacePermId"
                }],
    
            resp = self._post_request(self.as_v3, request)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            if len(resp) == 0:
                raise ValueError("No such space: %s" % spaceId)
    
            return Space(self, None, resp[spaceId])
    
        def get_samples(self, code=None, permId=None, space=None, project=None, experiment=None, type=None,
    
                        withParents=None, withChildren=None, tags=None, **properties):
    
            """ Get a list of all samples for a given space/project/experiment (or any combination)
            """
    
            sub_criteria = []
            if space:
    
                sub_criteria.append(_gen_search_criteria({
    
                    "space": "Space",
                    "operator": "AND",
                    "code": space
                  })
                )
    
                exp_crit = _subcriteria_for_code(experiment, 'experiment')
                proj_crit = _subcriteria_for_code(project, 'project')
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                exp_crit['criteria'] = []
                exp_crit['criteria'].append(proj_crit)
                sub_criteria.append(exp_crit)
    
                sub_criteria.append(_subcriteria_for_code(experiment, 'experiment'))
    
            if properties is not None:
                for prop in properties:
                    sub_criteria.append(_subcriteria_for_properties(prop, properties[prop]))
    
            if type:
                sub_criteria.append(_subcriteria_for_code(type, 'sample_type'))
    
            if tags:
                sub_criteria.append(_subcriteria_for_tags(tags))
    
            if code:
                sub_criteria.append(_criteria_for_code(code))
    
            if permId:
                sub_criteria.append(_common_search("as.dto.common.search.PermIdSearchCriteria",permId))
    
                if not isinstance(withParents, list):
                    withParents = [withParents]
                for parent in withParents:
                    sub_criteria.append(
    
                            _gen_search_criteria({
    
                            "sample": "SampleParents",
                            "identifier": parent
                        })
                    )
    
                if not isinstance(withChildren, list):
                    withChildren = [withChildren]
                for child in withChildren:
                    sub_criteria.append(
    
                            _gen_search_criteria({
    
                            "sample": "SampleChildren",
                            "identifier": child
                        })
                    )
    
    
            criteria = {
                "criteria": sub_criteria,
                "@type": "as.dto.sample.search.SampleSearchCriteria",
                "operator": "AND"
            }
    
                "properties": { "@type": "as.dto.property.fetchoptions.PropertyFetchOptions" },
                "tags": { "@type": "as.dto.tag.fetchoptions.TagFetchOptions" },
                "registrator": { "@type": "as.dto.person.fetchoptions.PersonFetchOptions" },
                "modifier": { "@type": "as.dto.person.fetchoptions.PersonFetchOptions" },
                "experiment": { "@type": "as.dto.experiment.fetchoptions.ExperimentFetchOptions" },
                "type": { "@type": "as.dto.sample.fetchoptions.SampleTypeFetchOptions" },
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                "@type": "as.dto.sample.fetchoptions.SampleFetchOptions",
    
            request = {
                "method": "searchSamples",
                "params": [ self.token, 
                    criteria,
                    options,
                ],
            }
    
            resp = self._post_request(self.as_v3, request)
            if resp is not None:
                objects = resp['objects']
    
    
                samples = DataFrame(objects)
                if len(samples) is 0:
                    raise ValueError("No samples found!")
    
                samples['registrationDate']= samples['registrationDate'].map(format_timestamp)
                samples['modificationDate']= samples['modificationDate'].map(format_timestamp)
                samples['registrator'] = samples['registrator'].map(extract_person)
                samples['modifier'] = samples['modifier'].map(extract_person)
                samples['identifier'] = samples['identifier'].map(extract_identifier)
    
                samples['permId'] = samples['permId'].map(extract_permid)
    
                samples['experiment'] = samples['experiment'].map(extract_nested_identifier)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                samples['sample_type'] = samples['type'].map(extract_nested_permid)
    
                ss = samples[['identifier', 'permId', 'experiment', 'sample_type', 'registrator', 'registrationDate', 'modifier', 'modificationDate']]
    
                return Things(self, 'sample', ss, 'identifier')
    
            else:
                raise ValueError("No samples found!")
    
    
        def get_experiments(self, code=None, type=None, space=None, project=None, tags=None, is_finished=None, **properties):
    
            """ Get a list of all experiment for a given space or project (or any combination)
            """
    
            sub_criteria = []
            if space:
    
                sub_criteria.append(_subcriteria_for_code(space, 'space'))
    
                sub_criteria.append(_subcriteria_for_code(project, 'project'))
    
            if code:
    
                sub_criteria.append(_criteria_for_code(code))
    
            if type:
                sub_criteria.append(_subcriteria_for_type(type, 'Experiment'))
            if tags:
                sub_criteria.append(_subcriteria_for_tags(tags))
            if is_finished is not None:
                sub_criteria.append(_subcriteria_for_is_finished(is_finished))
            if properties is not None:
                for prop in properties:
                    sub_criteria.append(_subcriteria_for_properties(prop, properties[prop]))
    
    
            criteria = {
                "criteria": sub_criteria,
                "@type": "as.dto.experiment.search.ExperimentSearchCriteria",
                "operator": "AND"
            }
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            fetchopts = { "@type": "as.dto.experiment.fetchoptions.ExperimentFetchOptions" }
            for option in ['tags', 'properties', 'registrator', 'modifier', 'project']:
                fetchopts[option] = fetch_option[option]
    
    
            request = {
                "method": "searchExperiments",
                "params": [ self.token, 
                    criteria,
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                    fetchopts,
    
                ],
            }
            resp = self._post_request(self.as_v3, request)
    
            if len(resp['objects']) == 0:
                raise ValueError("No experiments found!")
    
            objects = resp['objects']
            parse_jackson(objects)
    
            experiments = DataFrame(objects)
            experiments['registrationDate']= experiments['registrationDate'].map(format_timestamp)
            experiments['modificationDate']= experiments['modificationDate'].map(format_timestamp)
            experiments['project']= experiments['project'].map(extract_code)
            experiments['registrator'] = experiments['registrator'].map(extract_person)
            experiments['modifier'] = experiments['modifier'].map(extract_person)
            experiments['identifier'] = experiments['identifier'].map(extract_identifier)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            experiments['permId'] = experiments['permId'].map(extract_permid)
    
            experiments['type'] = experiments['type'].map(extract_code)
    
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            exps = experiments[['identifier', 'permId', 'project', 'type', 'registrator', 
    
                'registrationDate', 'modifier', 'modificationDate']]
            return Things(self, 'experiment', exps, 'identifier')
    
    
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            code=None, type=None, withParents=None, withChildren=None,
    
            sample=None, experiment=None, project=None, tags=None
    
    
            sub_criteria = []
    
            if code:
                sub_criteria.append(_criteria_for_code(code))
            if type:
                sub_criteria.append(_subcriteria_for_type(type, 'DataSet'))
            if withParents:
                sub_criteria.append(_subcriteria_for_permid(withParents, 'DataSet', 'Parents'))
            if withChildren:
                sub_criteria.append(_subcriteria_for_permid(withChildren, 'DataSet', 'Children'))
    
    
            if sample:
                sub_criteria.append(_subcriteria_for_code(sample, 'Sample'))
            if experiment:
                sub_criteria.append(_subcriteria_for_code(experiment, 'Experiment'))
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            if project:
                exp_crit = _subcriteria_for_code(experiment, 'Experiment')
                proj_crit = _subcriteria_for_code(project, 'Project')
                exp_crit['criteria'] = []
                exp_crit['criteria'].append(proj_crit)
                sub_criteria.append(exp_crit)
    
            if tags:
                sub_criteria.append(_subcriteria_for_tags(tags))
    
            criteria = {
                "criteria": sub_criteria,
                "@type": "as.dto.dataset.search.DataSetSearchCriteria",
                "operator": "AND"
            }
    
            fetchopts = {
                "containers":   { "@type": "as.dto.dataset.fetchoptions.DataSetFetchOptions" },
                "type":         { "@type": "as.dto.dataset.fetchoptions.DataSetTypeFetchOptions" }
            }
    
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            for option in ['tags', 'properties', 'sample', 'experiment']:
    
                fetchopts[option] = fetch_option[option]
    
            request = {
                "method": "searchDataSets",
                "params": [ self.token, 
                    criteria,
                    fetchopts,
                ],
            }
            resp = self._post_request(self.as_v3, request)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
            objects = resp['objects']
            if len(objects) == 0:
                raise ValueError("no datasets found!")
            else:
    
                parse_jackson(objects)
                datasets = DataFrame(objects)
                datasets['registrationDate']= datasets['registrationDate'].map(format_timestamp)
                datasets['modificationDate']= datasets['modificationDate'].map(format_timestamp)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                datasets['experiment']= datasets['experiment'].map(extract_nested_identifier)
    
                datasets['sample']= datasets['sample'].map(extract_nested_identifier)
                datasets['type']= datasets['type'].map(extract_code)
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                datasets['permId'] = datasets['code']
    
    Swen Vermeul's avatar
    Swen Vermeul committed
                    datasets[['permId', 'properties', 'type', 'experiment', 'sample', 'registrationDate', 'modificationDate']],
                    'permId'
    
        def get_experiment(self, expId, withAttachments=False):
    
            """ Returns an experiment object for a given identifier (expId).
            """
    
    
                "@type": "as.dto.experiment.fetchoptions.ExperimentFetchOptions",
                "type": {
                    "@type": "as.dto.experiment.fetchoptions.ExperimentTypeFetchOptions",
                },
    
            search_request = search_request_for_identifier(expId, 'experiment')
    
            for option in ['tags', 'properties', 'attachments', 'project', 'samples']:
    
                fetchopts[option] = fetch_option[option]
    
            if withAttachments:
                fetchopts['attachments'] = fetch_option['attachmentsWithContent']
    
    
            request = {
            "method": "getExperiments",
                "params": [