Skip to content

Commit

Permalink
API for loading .mod files selectively
Browse files Browse the repository at this point in the history
  • Loading branch information
vvbragin committed Dec 28, 2023
1 parent 9d53354 commit 0becdf6
Show file tree
Hide file tree
Showing 5 changed files with 128 additions and 36 deletions.
2 changes: 2 additions & 0 deletions CHANGES.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@

- Examples based on the CA3 model using the 'colorbyPhase' option in the plotRaster

- API for loading .mod files selectively

**Bug fixes**

- Fixed loading point cell params from legacy models (issue 607)
Expand Down
3 changes: 3 additions & 0 deletions netpyne/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,3 +31,6 @@
from netpyne import specs
from netpyne import support
from netpyne import tests

import neuron
neuron._netpyne_mech_hashes = {}
149 changes: 114 additions & 35 deletions netpyne/sim/load.py
Original file line number Diff line number Diff line change
Expand Up @@ -483,8 +483,7 @@ def loadFromIndexFile(index):
return loadModel(index, loadMechs=False)


def loadModel(path, loadMechs=True, ignoreMechAlreadyExistsError=False):

def loadModel(path, loadMechs=True, forceCompileMechs=False, returnLoadMechStatus=False):
import __main__
import json
from .. import sim
Expand All @@ -511,7 +510,9 @@ def loadModel(path, loadMechs=True, ignoreMechAlreadyExistsError=False):
modFolder = indexData.get('mod_folder')
if modFolder:
modFolderPath = os.path.join(dir, modFolder)
__processMod(modFolderPath, ignoreMechAlreadyExistsError)
loadedMods, skippedMods = processModFiles(modFolderPath, forceCompile=forceCompileMechs)
else:
loadedMods, skippedMods = [], [] # for consistency

os.chdir(dir)

Expand Down Expand Up @@ -545,53 +546,71 @@ def loadModel(path, loadMechs=True, ignoreMechAlreadyExistsError=False):

os.chdir(originalDir)

return cfg, netParams
res = cfg, netParams
if returnLoadMechStatus:
res += loadedMods, skippedMods
return res


def __processMod(modFolderPath, ignoreMechAlreadyExistsError):
import os, subprocess, shutil, time, glob
def processModFiles(modFolderPath, forceCompile=False):
import os, glob
import neuron
from neuron import h

if not os.path.exists(modFolderPath):
print(f"Warning: specified 'mod_folder' path {modFolderPath} doesn't exist")
return
return None, None

originalDir = os.getcwd()
os.chdir(modFolderPath)
shutil.rmtree('x86_64', ignore_errors=True)

if not ignoreMechAlreadyExistsError:
# compile and load all the mods from folder. If some of them already exist,
# it will raise the error
subprocess.call(["nrnivmodl"])
neuron.load_mechanisms(os.path.abspath('.'))
# all mod file names in this dir (abs path)
allMods = glob.glob(f"{os.getcwd()}/*.mod")

# mapping from file name to mechanism name
filesAndMechs = {file: __extractModelName(file) for file in allMods}

# first, try to identify which mechs are already loaded
modsToSkip = []
modsToSkipStatus = []
for fileName in allMods:
mechName = filesAndMechs[fileName]

if not hasattr(h, mechName):
continue

# this mech is already loaded, so it will be skipped
modsToSkip.append(fileName)

# now try to figure out if it's differs from loaded version
thisFileHash = utils.fileDigest(fileName)
alreadyLoadedModHash = neuron._netpyne_mech_hashes.get(mechName)

if not alreadyLoadedModHash:
modsToSkipStatus.append('unknown')
elif thisFileHash == alreadyLoadedModHash:
modsToSkipStatus.append('unmodified')
else:
modsToSkipStatus.append('modified')

modsToLoad = set(allMods) - set(modsToSkip)

if not modsToLoad:
print('All mods are already loaded. Skipping..')

elif len(modsToLoad) == len(allMods):
# no mods to skip, compile/load all at once
_compileAndLoadMechanisms(forceCompile=forceCompile, filesToMechsMapping=filesAndMechs)
else:
# parse mod files and load only those not yet loaded
mods = glob.glob("*.mod")
# proceed only with files not loaded before
_compileAndLoadMechanisms(files=modsToLoad, forceCompile=forceCompile, filesToMechsMapping=filesAndMechs)

alreadyLoadedMods = []
for file in mods:
name = __extractModelName(file)
try:
getattr(h, name)
alreadyLoadedMods.append(file)
except:
pass

modsToLoad = set(mods) - set(alreadyLoadedMods)
if len(modsToLoad) > 0:
# compile into dir other than x86_64 to bypass possibly cached data
tmpDir = f'tmp_{os.getpid()}_{time.time()}'
os.mkdir(tmpDir)
os.chdir(tmpDir)
modsToLoad = list(map(lambda path: f'../{path}', modsToLoad))
subprocess.call(['nrnivmodl'] + list(modsToLoad))
neuron.load_mechanisms(os.path.abspath('.'))
os.chdir('..')
os.system(f'rm -rf {tmpDir}')
loadedMods = [(fileName, filesAndMechs[fileName]) for fileName in modsToLoad]
skippedMods = [(fileName, filesAndMechs[fileName], status) for fileName, status in zip(modsToSkip, modsToSkipStatus)]

os.chdir(originalDir)
return [loadedMods, skippedMods]


def __extractModelName(modFile):
import re
Expand All @@ -610,6 +629,66 @@ def __extractModelName(modFile):
return None


def _compileAndLoadMechanisms(path=None, files=None, forceCompile=False, filesToMechsMapping={}):

import neuron, os, subprocess, shutil, platform, time

origPath = os.getcwd()

if path:
os.chdir(path)

if files:
files = [os.path.abspath(file) for file in files]

tmpDir = None
if hasattr(neuron, 'nrn_dll_loaded') \
and (os.getcwd() in neuron.nrn_dll_loaded):
# this path is already stored in NEURON's internal cache, which makes it not possible to load anything else from it.
# need to re-compile into another folder to load from there
compile = True

tmpDir = f'tmp_{os.getpid()}_{time.time()}'
os.mkdir(tmpDir)

os.chdir(tmpDir)
elif not os.path.exists(platform.machine()):
# there is no folder with compiled files (e.g. 'x86_64') at given path
compile = True
elif files:
# if about to load a specific subset of files, compilation is necessary
compile = True
else:
compile = forceCompile

if compile:
if files: # compile specific files
cmd = ['nrnivmodl'] + files
elif tmpDir: # being in temp dir, compile all files in parent folder
cmd = ['nrnivmodl', '..']
else: # compile all files in current folder
cmd = ['nrnivmodl']

subprocess.call(cmd)

neuron.load_mechanisms(os.path.abspath('.')) # load compiled dlls

# save md5 hashes of just loaded mod files
if not files:
import glob
files = glob.glob(f"{os.getcwd()}/*.mod")

for file in files:
mech = filesToMechsMapping.get(file, __extractModelName(file))
neuron._netpyne_mech_hashes[mech] = utils.fileDigest(file)

if tmpDir:
os.chdir('..')
os.system(f'rm -rf {tmpDir}')

os.chdir(origPath)


# ------------------------------------------------------------------------------
# Convert compact (list-based) to long (dict-based) conn format
# ------------------------------------------------------------------------------
Expand Down
8 changes: 8 additions & 0 deletions netpyne/sim/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -297,6 +297,14 @@ def hashList(obj):
return int(hashlib.md5(array.array(chr(ord('L')), obj)).hexdigest()[0:8], 16)


# ------------------------------------------------------------------------------
# Hash function for file (md5 hex)
# ------------------------------------------------------------------------------
def fileDigest(file):
fileStr = open(file,'rb').read()
return hashlib.md5(fileStr).hexdigest()


# ------------------------------------------------------------------------------
# Initialize the stim randomizer
# ------------------------------------------------------------------------------
Expand Down
2 changes: 1 addition & 1 deletion netpyne/sim/validator.py
Original file line number Diff line number Diff line change
Expand Up @@ -1051,7 +1051,7 @@ def checkValidation():

def checkModelValid(index):
print(f'PROCESSSING {index}')
_, netParams = sim.loadModel(index, loadMechs=True, ignoreMechAlreadyExistsError=True)
_, netParams = sim.loadModel(index, loadMechs=True)
valid, failed = validator.validateNetParams(net_params=netParams)
if failed:
print(f'FOUND {len(failed)} ERRORS IN {index}')
Expand Down

0 comments on commit 0becdf6

Please sign in to comment.