Developer's Guide ================= Adding new search paths ----------------------- All of the file path information is contained in the ``rhino_paths`` dictionary contained within :mod:`cmlreaders.constants`. The keys of the dictionary are the supported data types and the values are python lists whose elements correspond to the possible locations of where the data can be found. Paths within the list should appear in the preferred search order since :class:`cmlreaders.path_finder.PathFinder` is set up to return the first path where an existing file is found. When adding a new search path, place the new path in the desired location within the list of search paths. Adding new data types --------------------- To add support for a new data type, first add the data type shortcut name to :mod:`cmlreaders.constants` with a list of possible locations on rhino as the value. Next, add this data type to the appropriate list: - subject_files - localization_files - montage_files - session_files - host_pc_files - used_classifier_files These lists are used by :class:`cmlreaders.path_finder.PathFinder`. Ideally, a reader already exists that can manage the new data type. If this is the case, add the new data type to the ``data_types`` class member of the reader class that should be used. If a new reader is required, see the following section. Adding new readers ------------------ New readers are added by extending :class:`cmlreaders.readers.BaseCMLReader` and implementing one or more of the ``as_xyz`` methods. The default output format when calling ``load`` is set by using the class variable ``default_representation`` which defaults to ``dataframe``. For example, say you want to create a new reader that defaults to using a ``dict`` as output and should be used for some data type, X. At a minimum, you will need to define a ``data_types`` list that contains X, and set ``default_representation`` to ``dict``. If there are additional data types that should use this reader, those should also be added to the ``data_types`` list. .. code-block:: python class MyReader(BaseCMLReader): data_types = ['X'] default_representation = 'dict' def as_dict(self): return {'for': 'great justice'} By default, all known protocols (e.g., ``r1``, ``ltp``) are assumed to be supported. If a reader only works for a subset, specify the ``protocols`` class variable: .. code-block:: python class RamThingReader(BaseCMLReader): data_types = ["ram_thing"] protocols = ["r1"] Once the reader works, test cases for the data types using the new reader should be added to :mod:`cmlreaders.test.test_cmlreader`. These are in addition to the test cases that should already exist for the new reader. For examples, see :mod:`cmlreaders.test.test_readers`. Adding new EEG source formats ----------------------------- Support for new EEG formats is added by extending :class:`cmlreaders.readers.eeg.BaseEEGReader` and implementing the ``read`` method. Once this is done, edit :func:`cmlreaders.readers.eeg.EEGReader._get_reader_class` to return your new class (the class itself, not an instance of it) when given a filename of the appropriate kind. Releasing new versions and building conda packages -------------------------------------------------- When releasing a new version, be sure to increment the version number in ``cmlreaders/__init__.py``. Several maintenance tasks are handled using Invoke_ and are defined in ``tasks.py``. Building a conda package: .. code-block:: shell-session $ invoke build Uploading builds to Anaconda Cloud: .. code-block:: shell-session $ invoke upload .. note:: This requires that you have already logged in with ``anaconda login``. .. note:: Automated deployment is enabled on TravisCI for every tagged version, so the build and upload tasks are only necessary to be run manually for debugging purposes or if the automated deployment fails. Building documentation: .. code-block:: shell-session $ invoke docs Running tests: .. code-block:: shell-session $ invoke test .. _Invoke: http://www.pyinvoke.org/