Source code for hbp_nrp_cle.brainsim.nest.NestControlAdapter

# ---LICENSE-BEGIN - DO NOT CHANGE OR MOVE THIS HEADER
# This file is part of the Neurorobotics Platform software
# Copyright (C) 2014,2015,2016,2017 Human Brain Project
# https://www.humanbrainproject.eu
#
# The Human Brain Project is a European Commission funded project
# in the frame of the Horizon2020 FET Flagship plan.
# http://ec.europa.eu/programmes/horizon2020/en/h2020-section/fet-flagships
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
# ---LICENSE-END
"""
Implementation of NestControlAdapter
"""
from __future__ import division


from builtins import range
from past.utils import old_div
from future.utils import iteritems

from hbp_nrp_cle.brainsim import IBrainControlAdapter
from hbp_nrp_cle.brainsim.common import PythonBrainLoader as BrainLoader
from hbp_nrp_cle.brainsim.nest import NestPopulationInfo
import hbp_nrp_cle.brainsim as brainsim
from hbp_nrp_cle.cle.CLEInterface import BrainRuntimeException
from hbp_nrp_cle.brainsim.nest.NestInfo import is_population

import hbp_nrp_cle.tf_framework.config as tf_config

import logging
from os import path
import copy

logger = logging.getLogger(__name__)

__author__ = 'LorenzoVannucci'


[docs]class NestControlAdapter(IBrainControlAdapter): """ Represents a controller object for the neuronal simulator """ def __init__(self, sim): """ Initializes the NEST control adapter :param sim: The simulator module """ self.__is_initialized = False self.__is_alive = False self.__rank = None self._sim = sim
[docs] def load_populations(self, populations): """ Load (or reload) populations defined on the currently loaded brain. :param populations: A dictionary indexed by population names and containing neuron indices. Neuron indices can be defined by a single integer, list of integers or python slices. Python slices can be replaced by a dictionary containing the 'from', 'to' and 'step' values. :raise Exception: When no brain is currently loaded """ # check if a valid brain is loaded if tf_config.brain_root is None: raise Exception( "No brain is currently loaded, cannot add Populations.") if not hasattr(tf_config.brain_root, 'circuit'): raise AttributeError("No circuit is found in the currently loaded brain:" "Cannot add Populations.") # valid brain found, load populations tf_config.brain_populations = self.populations_using_json_slice( populations) if not hasattr(tf_config.brain_root, 'populations_keys'): tf_config.brain_root.populations_keys = [] BrainLoader.clear_populations(tf_config.brain_root) setup_access_to_population( tf_config.brain_root, **self.populations_using_python_slice(populations))
[docs] def load_brain(self, brain_file, populations=None): """ Loads the neuronal network contained in the given file :param brain_file: The path to the neuronal network file :param populations: Additional populations to create (Optional) """ extension = path.splitext(brain_file)[1] brainsim.simulator = self._sim if extension == ".py": self.__load_py_brain(brain_file) # populations: # is not None -> load populations # is None -> no new population to load -> reload current ones (if any) # is {} -> load empty dict of populations if populations is not None: pops = populations elif tf_config.brain_populations: pops = tf_config.brain_populations else: pops = {} self.load_populations(pops) else: msg = "Neuronal network format {0} not supported".format(extension) raise Exception(msg)
def __load_py_brain(self, brain_file): """ Loads the brain model specified in the given Python script :param brain_file: The Python file containing the network """ self.__is_alive = True if not self.__is_initialized: self.initialize() tf_config.brain_root = BrainLoader.load_py_network(brain_file) logger.info("Saving brain source") with open(brain_file) as source: tf_config.brain_source = source.read()
[docs] def initialize(self, **params): """ Initializes the neuronal simulator :param timestep: The timestep used for the neuronal simulation :param min_delay: The minimum delay :param max_delay: The maximum delay :param threads: The amount of threads that should be used to run the simulation :param rng_seeds: The rng seeds for the simulation :return: True if the simulator is initialized, otherwise False """ if not self.__is_initialized: pars = {"resolution": params.get('timestep', 0.1), "min_delay": params.get('min_delay', 0.1), "max_delay": params.get('max_delay', 20.0)} self._sim.ResetKernel() self._sim.SetKernelStatus(pars) self.__is_initialized = True logger.info("neuronal simulator initialized") else: logger.warn( "trying to initialize an already initialized controller") return self.__is_initialized
# pylint: disable=no-self-use def _is_population(self, candidate): """ Determines whether the candidate is a population (i.e. a list of int) :param candidate: The candidate """ return is_population(candidate) # pylint: disable=no-self-use def _create_population_info(self, population, name): """ Creates a population info object for the given population :param population: The population :param name: The name of the population """ try: celltype = str(self._sim.GetStatus(population, 'model')[0]) parameters = self._sim.GetDefaults(celltype) except self._sim.NESTError: parameters = {} return NestPopulationInfo(population, name, parameters) def __find_all_populations(self, candidate, member_name, populations): """ Finds all populations under the given object and adds them to the list of populations :param candidate: The object tree :param member_name: The base member name :param populations: The list of populations """ if self._is_population(candidate): populations.append( self._create_population_info(candidate, member_name)) elif isinstance(candidate, list): for index in range(len(candidate)): self.__find_all_populations(candidate[index], "{name}[{id}]".format(name=member_name, id=str(index)), populations)
[docs] def get_populations(self): """ Gets an information about the populations currently available :return: A list of population infos """ populations = [] for member in dir(tf_config.brain_root): candidate = getattr(tf_config.brain_root, member) self.__find_all_populations(candidate, member, populations) return populations
@property def is_initialized(self): """ Gets a value indicating whether initialize has been called """ return self.__is_initialized
[docs] def is_alive(self): # -> bool: """ Gets a status whether the neuronal simulator is still alive :return: True if the simulator is alive, otherwise False """ return self.__is_alive
[docs] def run_step(self, dt): # -> None: """ Runs the neuronal simulator for the given amount of simulated time :param dt: the simulated time in milliseconds """ try: self._sim.Simulate(dt) except Exception as e: raise BrainRuntimeException(str(e))
[docs] def shutdown(self): # -> None: """ Shuts down the neuronal simulator """ self.__is_alive = False self.__is_initialized = False logger.info("neuronal simulator ended")
[docs] def reset(self): # -> None: """ Resets the neuronal simulator """ logger.info("neuronal simulator reset")
[docs] @staticmethod def populations_using_json_slice(populations): """ Turn populations defined as python slices into python dicts to allow straightforward translation in to json. :param populations: a dictionary whose values are either python lists or slices. Slices can be of two types, either python slices, or dictionnaries of the form {'from': 1, 'to': 10, 'step': 2} :return: A dictionary where python slices have been replaced by dictionaries referred to as 'json slices'. """ result = copy.deepcopy(populations) for key, value in iteritems(populations): if isinstance(value, slice): p = {'from': value.start, 'to': value.stop, 'step': value.step} result[key] = p return result
[docs] @staticmethod def populations_using_python_slice(populations): """ Turn slices defined as python dicts into python slices. Populations of type list are left unchanged. :param populations: a dictionary whose values are either python lists or slices. Slices can be of two types, either python slices, or dictionaries of the form {'from': 1, 'to': 10, 'step': 2} :return: A dictionary where 'json slices' (plain python dicts) have been replaced by python slices. """ result = copy.deepcopy(populations) for key, value in iteritems(populations): if isinstance(value, dict): if 'from' in value and 'to' in value: step = value.get('step') result[key] = slice(value['from'], value['to'], step) elif 'list' in value: value_list = value.get('list') result[key] = list(value_list) return result
[docs] def get_Timeout(self): """ returns The maximum amount of time (in seconds) to wait for the end of this step """ return 5
[docs]def setup_access_to_population(brain_module, **populations): """ Sets up the access to the population :param brain_module: The brain module :param populations: A dictionary of the populations and their ids """ try: circuit = brain_module.circuit logger.debug("Found circuit") for p in populations: population = populations[p] neurons = circuit[population] logger.debug("Population '%s': %s", p, neurons) if isinstance(population, slice): expected_size = abs( old_div((population.start - population.stop), (population.step or 1))) if len(neurons) != expected_size: raise Exception("Population '%s' out of bounds" % p) brain_module.__dict__[p] = neurons brain_module.populations_keys.append(p) except AttributeError: if populations: raise Exception( "Could not initialize populations, no circuit found")