Source code for one.converters

"""A module for inter-converting experiment identifiers.

There are multiple ways to uniquely identify an experiment:
    - eid (str) : An experiment UUID as a string
    - np (int64) : An experiment UUID encoded as 2 int64s
    - path (Path) : A pathlib ALF path of the form <lab>/Subjects/<subject>/<date>/<number>
    - ref (str) : An experiment reference string of the form yyyy-mm-dd_n_subject
    - url (str) : An remote http session path of the form <lab>/Subjects/<subject>/<date>/<number>
"""
import re
import functools
import datetime
from uuid import UUID
from inspect import unwrap
from pathlib import Path, PurePosixPath
from urllib.parse import urlsplit
from typing import Optional, Union, Mapping, List, Iterable as Iter

import numpy as np
import pandas as pd
from iblutil.io import parquet
from iblutil.util import Bunch

import one.alf.io as alfio
from one.alf.spec import is_session_path, is_uuid_string
from one.alf.files import get_session_path, add_uuid_string
from .util import Listable


[docs]def recurse(func): @functools.wraps(func) def wrapper_decorator(*args, **kwargs): if len(args) <= 1: return func(*args, **kwargs) obj, first = args[:2] exclude = (str, Mapping, pd.Series, pd.DataFrame) if isinstance(first, Iter) and not isinstance(first, exclude): return [func(obj, item, *args[2:], **kwargs) for item in first] else: return func(obj, first, *args[2:], **kwargs) return wrapper_decorator
[docs]def parse_values(func): """Convert str values in reference dict to appropriate type""" def parse_ref(ref): if ref: if isinstance(ref['date'], str): if len(ref['date']) == 10: ref['date'] = datetime.date.fromisoformat(ref['date']) else: ref['date'] = datetime.datetime.fromisoformat(ref['date']).date() ref['sequence'] = int(ref['sequence']) return ref @functools.wraps(func) def wrapper_decorator(*args, **kwargs): parse = kwargs.pop('parse', True) ref = func(*args, **kwargs) if not parse or isinstance(ref, str): return ref elif isinstance(ref, list): return list(map(parse_ref, ref)) else: return parse_ref(ref) return wrapper_decorator
[docs]class ConversionMixin: def __init__(self): self._cache = None self._par = None
[docs] @recurse def to_eid(self, id: Listable(Union[str, Path, UUID, dict]) = None, cache_dir: Optional[Union[str, Path]] = None) -> Listable(str): # TODO Could add np2str here # if isinstance(id, (list, tuple)): # Recurse # return [self.to_eid(i, cache_dir) for i in id] if id is None: return elif isinstance(id, UUID): return str(id) elif self.is_exp_ref(id): return self.ref2eid(id) elif isinstance(id, dict): assert {'subject', 'number', 'start_time', 'lab'}.issubset(id) root = Path(cache_dir or self._cache_dir) id = root.joinpath( id['lab'], 'Subjects', id['subject'], id['start_time'][:10], ('%03d' % id['number'])) if isinstance(id, Path): return self.path2eid(id) elif isinstance(id, str): if is_session_path(id) or get_session_path(id): return self.path2eid(id) if len(id) > 36: id = id[-36:] if not is_uuid_string(id): raise ValueError('Invalid experiment ID') else: return id else: raise ValueError('Unrecognized experiment ID')
[docs] @recurse def eid2path(self, eid: str) -> Optional[Listable(Path)]: """ From an experiment id or a list of experiment ids, gets the local cache path Parameters ---------- eid : str, uuid.UUID Experiment ID (UUID) or list of UUIDs Returns ------- A pathlib.Path of the session """ # If eid is a list of eIDs recurse through list and return the results if isinstance(eid, list): path_list = [] for p in eid: path_list.append(self.eid2path(p)) return path_list # If not valid return None if not is_uuid_string(eid): raise ValueError(eid + " is not a valid eID/UUID string") if self._cache['sessions'].size == 0: return # load path from cache if self._index_type() is int: eid = parquet.str2np(eid).tolist() try: ses = self._cache['sessions'].loc[eid] assert len(ses) == 1, 'Duplicate eids in sessions table' ses, = ses.to_dict('records') return Path(self._cache_dir).joinpath( ses['lab'], 'Subjects', ses['subject'], str(ses['date']), str(ses['number']).zfill(3)) except KeyError: return
[docs] @recurse def path2eid(self, path_obj): """ From a local path, gets the experiment id Parameters ---------- path_obj : pathlib.Path, str Local path or list of local paths Returns ------- Experiment ID (eid) string or list of eids """ # else ensure the path ends with mouse,date, number session_path = get_session_path(path_obj) sessions = self._cache['sessions'] # if path does not have a date and a number, or cache is empty return None if session_path is None or sessions.size == 0: return None # reduce session records from cache toDate = datetime.date.fromisoformat subject, date, number = session_path.parts[-3:] for col, val in zip(('subject', 'date', 'number'), (subject, toDate(date), int(number))): sessions = sessions[sessions[col] == val] if sessions.size == 0: return assert len(sessions) == 1 eid, = sessions.index.values if isinstance(eid, tuple): eid = parquet.np2str(np.array(eid)) return eid
[docs] @recurse def path2record(self, filepath): """ TODO Return Series instead of DataFrame NB: Assumes <lab>/Subjects/<subject>/<date>/<number> pattern Parameters ---------- filepath : str, pathlib.Path File path or HTTP URL Returns ------- pandas.Series corresponding to the file """ if isinstance(filepath, str) and filepath.startswith('http'): # Remove the UUID from path filepath = urlsplit(filepath).path.strip('/') filepath = alfio.remove_uuid_file(PurePosixPath(filepath), dry=True) session_path = get_session_path(filepath).as_posix() else: # No way of knowing root session path parts without cache tables eid = self.path2eid(filepath) session_path, *_ = self.list_datasets(eid, details=True).session_path rec = self._cache['datasets'] if rec.empty: return # if (rec := self._cache['datasets']).empty: # py 3.8 # return rec = rec[rec['session_path'] == session_path] rec = rec[rec['rel_path'].apply(lambda x: filepath.as_posix().endswith(x))] assert len(rec) < 2, 'Multiple records found' return None if rec.empty else rec
[docs] @recurse def path2url(self, filepath): """ Given a local file path, constructs the URL of the remote file. Parameters ---------- filepath : str, pathlib.Path A local file path Returns ------- A remote URL string """ record = self.path2record(filepath) if record is None: return return unwrap(self.record2url)(record)
[docs] def record2url(self, dataset): assert self._web_client # FIXME Should be OneAlyx converter only # TODO Document # for i, rec in dataset.iterrows(): if isinstance(dataset, pd.Series): uuid, = parquet.np2str(np.array([dataset.name])) else: assert len(dataset) == 1 uuid, = parquet.np2str(dataset.reset_index()[['id_0', 'id_1']]) session_path, rel_path = dataset[['session_path', 'rel_path']].to_numpy().flatten() url = PurePosixPath(session_path, rel_path) return self._web_client.rel_path2url(add_uuid_string(url, uuid).as_posix())
[docs] def record2path(self, dataset) -> Optional[Path]: """ Given a set of dataset records, checks the corresponding exists flag in the cache correctly reflects the files system Parameters ---------- dataset : pd.DataFrame, pd.Series A datasets dataframe slice Returns ------- File path for the record """ assert isinstance(dataset, pd.Series) or len(dataset) == 1 session_path, rel_path = dataset[['session_path', 'rel_path']].to_numpy().flatten() file = Path(self._cache_dir, session_path, rel_path) return file # files[0] if len(datasets) == 1 else files
[docs] @recurse def eid2ref(self, eid: Union[str, Iter], as_dict=True, parse=True) \ -> Union[str, Mapping, List]: """ Get human-readable session ref from path Parameters ---------- eid : str, uuid.UUID The experiment uuid to find reference for as_dict : bool If false a string is returned in the form 'subject_sequence_yyyy-mm-dd' parse : bool If true, the reference date and sequence are parsed from strings to their respective data types Returns ------- One or more objects with keys ('subject', 'date', 'sequence'), or strings with the form yyyy-mm-dd_n_subject Examples -------- >>> eid = '4e0b3320-47b7-416e-b842-c34dc9004cf8' >>> one.eid2ref(eid) {'subject': 'flowers', 'date': datetime.date(2018, 7, 13), 'sequence': 1} >>> one.eid2ref(eid, parse=False) {'subject': 'flowers', 'date': '2018-07-13', 'sequence': '001'} >>> one.eid2ref(eid, as_dict=False) '2018-07-13_1_flowers' >>> one.eid2ref(eid, as_dict=False, parse=False) '2018-07-13_001_flowers' >>> one.eid2ref([eid, '7dc3c44b-225f-4083-be3d-07b8562885f4']) [{'subject': 'flowers', 'date': datetime.date(2018, 7, 13), 'sequence': 1}, {'subject': 'KS005', 'date': datetime.date(2019, 4, 11), 'sequence': 1}] """ d = self.get_details(eid) if parse: date = datetime.datetime.fromisoformat(d['start_time']).date() ref = {'subject': d['subject'], 'date': date, 'sequence': d['number']} format_str = '{date:%Y-%m-%d}_{sequence:d}_{subject:s}' else: date = d['start_time'][:10] ref = {'subject': d['subject'], 'date': date, 'sequence': '%03d' % d['number']} format_str = '{date:s}_{sequence:s}_{subject:s}' return Bunch(ref) if as_dict else format_str.format(**ref)
[docs] @recurse def ref2eid(self, ref: Union[Mapping, str, Iter]) -> Union[str, List]: """ Returns experiment uuid, given one or more experiment references Parameters ---------- ref : str, dict, list One or more objects with keys ('subject', 'date', 'sequence'), or strings with the form yyyy-mm-dd_n_subject Returns ------- One or more experiment uuid strings Examples -------- >>> base = 'https://test.alyx.internationalbrainlab.org' >>> one = ONE(username='test_user', password='TapetesBloc18', base_url=base) Connected to... >>> ref = {'date': datetime(2018, 7, 13).date(), 'sequence': 1, 'subject': 'flowers'} >>> one.ref2eid(ref) '4e0b3320-47b7-416e-b842-c34dc9004cf8' >>> one.ref2eid(['2018-07-13_1_flowers', '2019-04-11_1_KS005']) ['4e0b3320-47b7-416e-b842-c34dc9004cf8', '7dc3c44b-225f-4083-be3d-07b8562885f4'] """ ref = self.ref2dict(ref, parse=False) # Ensure dict session = self.search( subject=ref['subject'], date_range=str(ref['date']), number=ref['sequence']) assert len(session) == 1, 'session not found' return session[0]
[docs] @recurse def ref2path(self, ref): """ Convert one or more experiment references to session path(s) Parameters ---------- ref : str, dict, list One or more objects with keys ('subject', 'date', 'sequence'), or strings with the form yyyy-mm-dd_n_subject Returns ------- Path object(s) for the experiment session(s) Examples -------- >>> base = 'https://test.alyx.internationalbrainlab.org' >>> one = ONE(username='test_user', password='TapetesBloc18', base_url=base) Connected to... >>> ref = {'subject': 'flowers', 'date': datetime(2018, 7, 13).date(), 'sequence': 1} >>> one.ref2path(ref) WindowsPath('E:/FlatIron/zadorlab/Subjects/flowers/2018-07-13/001') >>> one.ref2path(['2018-07-13_1_flowers', '2019-04-11_1_KS005']) [WindowsPath('E:/FlatIron/zadorlab/Subjects/flowers/2018-07-13/001'), WindowsPath('E:/FlatIron/cortexlab/Subjects/KS005/2019-04-11/001')] """ eid2path = unwrap(self.eid2path) ref2eid = unwrap(self.ref2eid) return eid2path(self, ref2eid(self, ref))
[docs] @staticmethod @parse_values def path2ref(path_str: Union[str, Path, Iter], as_dict=True) -> Union[Bunch, List]: """ Returns a human readable experiment reference, given a session path. The path need not exist. Parameters ---------- path_str : str A path to a given session as_dict : bool If True a Bunch is returned, otherwise a string Returns ------- One or more objects with keys ('subject', 'date', 'sequence') Examples -------- >>> path_str = Path('E:/FlatIron/Subjects/zadorlab/flowers/2018-07-13/001') >>> path2ref(path_str) {'subject': 'flowers', 'date': datetime.date(2018, 7, 13), 'sequence': 1} >>> path2ref(path_str, parse=False) {'subject': 'flowers', 'date': '2018-07-13', 'sequence': '001'} >>> path_str2 = Path('E:/FlatIron/Subjects/churchlandlab/CSHL046/2020-06-20/002') >>> path2ref([path_str, path_str2]) [{'subject': 'flowers', 'date': datetime.date(2018, 7, 13), 'sequence': 1}, {'subject': 'CSHL046', 'date': datetime.date(2020, 6, 20), 'sequence': 2}] """ if isinstance(path_str, (list, tuple)): return [unwrap(ConversionMixin.path2ref)(x) for x in path_str] pattern = r'(?P<subject>[\w-]+)([\\/])(?P<date>\d{4}-\d{2}-\d{2})(\2)(?P<sequence>\d{3})' match = re.search(pattern, str(path_str)) if match: ref = match.groupdict() return Bunch(ref) if as_dict else '{date:s}_{sequence:s}_{subject:s}'.format(**ref)
[docs] def ref2dj(self, ref: Union[str, Mapping, Iter]): """ Return an ibl-pipeline sessions table, restricted by experiment reference(s) Parameters ---------- ref : str, list, dict One or more objects with keys ('subject', 'date', 'sequence'), or strings with the form yyyy-mm-dd_n_subject Returns ------- An acquisition.Session table Examples -------- >>> ref2dj('2020-06-20_2_CSHL046').fetch1() Connecting... {'subject_uuid': UUID('dffc24bc-bd97-4c2a-bef3-3e9320dc3dd7'), 'session_start_time': datetime.datetime(2020, 6, 20, 13, 31, 47), 'session_number': 2, 'session_date': datetime.date(2020, 6, 20), 'subject_nickname': 'CSHL046'} >>> len(ref2dj({'date':'2020-06-20', 'sequence':'002', 'subject':'CSHL046'})) 1 >>> len(ref2dj(['2020-06-20_2_CSHL046', '2019-11-01_1_ibl_witten_13'])) 2 """ from ibl_pipeline import subject, acquisition sessions = acquisition.Session.proj('session_number', session_date='date(session_start_time)') sessions = sessions * subject.Subject.proj('subject_nickname') ref = self.ref2dict(ref) # Ensure dict-like @recurse def restrict(r): date, sequence, subject = dict(sorted(r.items())).values() # Unpack sorted restriction = { 'subject_nickname': subject, 'session_number': sequence, 'session_date': date} return restriction return sessions & restrict(ref)
[docs] @staticmethod def is_exp_ref(ref: Union[str, Mapping, Iter]) -> Union[bool, List[bool]]: """ Returns True is ref is a valid experiment reference Parameters ---------- ref : str, dict, list One or more objects with keys ('subject', 'date', 'sequence'), or strings with the form yyyy-mm-dd_n_subject Returns ------- True if ref is valid Examples -------- >>> ref = {'date': datetime(2018, 7, 13).date(), 'sequence': 1, 'subject': 'flowers'} >>> is_exp_ref(ref) True >>> is_exp_ref('2018-07-13_001_flowers') True >>> is_exp_ref('invalid_ref') False """ if isinstance(ref, (list, tuple)): return [ConversionMixin.is_exp_ref(x) for x in ref] if isinstance(ref, (Bunch, dict)): if not {'subject', 'date', 'sequence'}.issubset(ref): return False ref = '{date}_{sequence}_{subject}'.format(**ref) elif not isinstance(ref, str): return False return re.compile(r'\d{4}(-\d{2}){2}_(\d{1}|\d{3})_\w+').match(ref) is not None
[docs] @recurse def path2pid(self, path): """Returns a portion of the path that represents the session and probe label""" raise NotImplementedError() path = Path(path).as_posix()
[docs] @staticmethod @parse_values def ref2dict(ref: Union[str, Mapping, Iter]) -> Union[Bunch, List]: """ Returns a Bunch (dict-like) from a reference string (or list thereof) Parameters ---------- ref : str, list One or more experiment reference strings Returns ------- A Bunch in with keys ('subject', 'sequence', 'date') Examples -------- >>> ref2dict('2018-07-13_1_flowers') {'date': datetime.date(2018, 7, 13), 'sequence': 1, 'subject': 'flowers'} >>> ref2dict('2018-07-13_001_flowers', parse=False) {'date': '2018-07-13', 'sequence': '001', 'subject': 'flowers'} >>> ref2dict(['2018-07-13_1_flowers', '2020-01-23_002_ibl_witten_01']) [{'date': datetime.date(2018, 7, 13), 'sequence': 1, 'subject': 'flowers'}, {'date': datetime.date(2020, 1, 23), 'sequence': 2, 'subject': 'ibl_witten_01'}] """ if isinstance(ref, (list, tuple)): return [ConversionMixin.ref2dict(x) for x in ref] if isinstance(ref, (Bunch, dict)): return Bunch(ref) # Short circuit ref = dict(zip(['date', 'sequence', 'subject'], ref.split('_', 2))) return Bunch(ref)
[docs] @staticmethod def dict2ref(ref_dict): if isinstance(ref_dict, (list, tuple)): return [ConversionMixin.dict2ref(x) for x in ref_dict] if not ref_dict: return parsed = any(not isinstance(k, str) for k in ref_dict.values()) format_str = ('{date:%Y-%m-%d}_{sequence:d}_{subject:s}' if parsed else '{date:s}_{sequence:s}_{subject:s}') return format_str.format(**ref_dict)
[docs]def one_path_from_dataset(dset, one_cache): """ Returns local one file path from a dset record or a list of dsets records from REST. Unlike `to_eid`, this function does not require ONE, and the dataset may not exist. Parameters ---------- dset : dict, list Dataset dictionary or list of dictionaries from Alyx rest endpoint one_cache : str, pathlib.Path, pathlib.PurePath The local ONE data cache directory Returns ------- The local path for a given dataset """ return path_from_dataset(dset, root_path=one_cache, uuid=False)
[docs]def path_from_dataset(dset, root_path=PurePosixPath('/'), repository=None, uuid=False): """ Returns the local file path from a dset record from a REST query Unlike `to_eid`, this function does not require ONE, and the dataset may not exist. Parameters ---------- dset : dict, list Dataset dictionary or list of dictionaries from Alyx rest endpoint root_path : str, pathlib.Path, pathlib.PurePath The prefix path such as the ONE download directory or remote http server root repository : str, None Which data repository to use from the file_records list, defaults to first online repository uuid : bool If True, the file path will contain the dataset UUID Returns ------- File path or list of paths """ if isinstance(dset, list): return [path_from_dataset(d) for d in dset] if repository: fr = next((fr for fr in dset['file_records'] if fr['data_repository'] == repository)) else: fr = next((fr for fr in dset['file_records'] if fr['data_url'])) uuid = dset['url'][-36:] if uuid else None return path_from_filerecord(fr, root_path=root_path, uuid=uuid)
[docs]def path_from_filerecord(fr, root_path=PurePosixPath('/'), uuid=None): """ Returns a data file Path constructed from an Alyx file record. The Path type returned depends on the type of root_path: If root_path is a string a Path object is returned, otherwise if the root_path is a PurePath, the same path type is returned. Parameters ---------- fr : dict An Alyx file record dict root_path : str, pathlib.Path An optional root path uuid : str, uuid.UUID An optional dataset UUID to add to the file name Returns ------- A filepath as a pathlib object """ if isinstance(fr, list): return [path_from_filerecord(f) for f in fr] repo_path = fr['data_repository_path'] repo_path = repo_path[repo_path.startswith('/'):] # remove starting / if any # repo_path = (p := fr['data_repository_path'])[p[0] == '/':] # py3.8 Remove slash at start file_path = PurePosixPath(repo_path, fr['relative_path']) if root_path: # NB: By checking for string we won't cast any PurePaths if isinstance(root_path, str): root_path = Path(root_path) file_path = root_path / file_path if uuid: file_path = add_uuid_string(file_path, uuid) return file_path