import warnings
import time
_t0 = time.time()
import sys
import os as _os
# ensure that any `import omfit.xxx` refers to this installation of OMFIT
sys.path.insert(0, _os.path.dirname(_os.path.abspath(__file__)))
from omfit_classes.startup_framework import *
# import utilities
from utils import _available_to_user_math, _available_to_user_util, _available_to_user_plot, _available_to_user_fusion
# ---------------------
# classes under the `class` folder and starting with OMFIT
# ---------------------
_loaded_omfit_class_files = []
import omfit_classes.omfit_base
from omfit_classes.omfit_base import *
from omfit_classes.omfit_base import _moduleAttrs, itemTagsValues
from omfit_classes import unix_os as os
from omfit_classes.omfit_clusters import OMFITclusters
# override sys.exit to prevent external packages from quitting the session
_exit = sys.exit
sys.exit = lambda *args, **kw: None
import omfit_classes.omfit_python
from omfit_classes.omfit_python import *
_k = 0
for _file in sorted(glob.glob(OMFITsrc + '/omfit_classes/omfit_*.py')):
    _python_module = os.path.splitext(os.path.split(_file)[1])[0]
    try:
        exec("import omfit_classes." + _python_module, globals(), locals())
        try:
            getattr(omfit_classes, _python_module).__all__
        except AttributeError:
            raise OMFITexception(
                'Exception in omfit/%s: OMFIT does not allow modules under omfit/classes'
                'not to have an __all__ attribute which explicitly specifies what symbols '
                'will be exported when from <module> import * is used on the module' % _python_module
            )
        # all classes must be available in omfit_base and omfit_python since that's where the OMFITtree is defined
        if _python_module != 'omfit_base':
            for item in getattr(omfit_classes, _python_module).__all__:
                setattr(omfit_classes.omfit_base, item, getattr(getattr(omfit_classes, _python_module), item))
        if _python_module != 'omfit_python':
            for item in getattr(omfit_classes, _python_module).__all__:
                setattr(omfit_classes.omfit_python, item, getattr(getattr(omfit_classes, _python_module), item))
        exec("from omfit_classes import " + _python_module, globals(), locals())
        exec("from omfit_classes." + _python_module + ' import *', globals(), locals())
        _k += 1
        _loaded_omfit_class_files.append(('%2d)%s' % (_k, _python_module.replace('omfit_', ''))).ljust(20))
    except Exception:
        printe('Offending module: ' + _python_module)
        raise
sys.exit = _exit
for item in omfit_classes.omfit_python.__all__:
    setattr(omfit_classes.omfit_base, item, getattr(omfit_python, item))
# backward compatibility with old classes location in `classes` folder instead of `omfit_classes`
for item in list(sys.modules.keys()):
    if item.startswith('omfit_classes.'):
        # allow `import classes.omfit_classname` # how Python environment with local OMFIT installation can use OMFIT classes
        sys.modules[re.sub('^omfit_classes\.', 'classes.', item)] = sys.modules[item]
print('Loaded OMFIT classes:')
while len(_loaded_omfit_class_files):
    print(''.join(_loaded_omfit_class_files[:4]))
    _loaded_omfit_class_files = _loaded_omfit_class_files[4:]
[docs]def reload_python(moduleName, quiet=False):
    """
    This function extends the Python builtin `reload` function to easily reload OMFIT classes
    >>> reload_python(omfit_classes.omfit_onetwo)
    :param moduleName: module or module name
    :param quiet: bool
        Suppress print statements listing reloaded objects
    """
    from matplotlib.cbook.deprecation import MatplotlibDeprecationWarning
    warnings.filterwarnings('ignore', category=MatplotlibDeprecationWarning)
    if not isinstance(moduleName, str):
        moduleName = moduleName.__name__
    if not (moduleName.startswith('omfit_') or moduleName.startswith('omfit_classes.omfit_')):
        printe('reload_python is meant for class containing modules (in omfit_classes) that start with `omfit_`')
        printe('Try instead:\n\nreload({0})\nfrom {0} import *'.format(moduleName))
        return
    if moduleName in ['omfit_tree', 'omfit_gui']:
        printe('Can not reload omfit_tree or omfit_gui')
        return
    tmp = {}
    exec("import %s" % moduleName, tmp)
    from importlib import reload
    reload(eval(moduleName))
    exec("from " + moduleName + " import *", tmp)
    # Keep track of reloaded objects to avoid storing them in the persistent OMFITconsoleDict namespace
    OMFITreloadedDict.update(tmp)
    # Find objects in the OMFIT tree that should be reloaded
    OMtr = traverse(OMFIT, string='OMFIT', onlyDict=True, skipDynaLoad=True)
    OMtr_cls = {}
    for loc in OMtr:
        if hasattr(eval(loc), '__class__') and eval(loc).__class__.__name__ in list(tmp.keys()):
            OMtr_cls.setdefault(eval(loc).__class__.__name__, []).append(loc)
    # Select only the items that should be reloaded
    items = list(tmp.keys())
    if hasattr(eval(moduleName), '__all__'):
        items = eval(moduleName).__all__
    if '__builtins__' in items:
        items.remove('__builtins__')
    # Loop through items to reload
    updated = {}
    for item in items:
        # Loop through python modules
        for mod in list(sys.modules.keys()):
            if hasattr(sys.modules[mod], '__dict__') and item in sys.modules[mod].__dict__:
                sys.modules[mod].__dict__[item] = tmp[item]
                updated.setdefault(mod, []).append(item)
        # Loop through OMFIT objects
        if item in OMtr_cls:
            for loc in OMtr_cls[item]:
                if not quiet:
                    print(loc)
                eval(loc).__class__ = tmp[item]
    # Update OMFIT data types
    omfit_classes.omfit_base._updateTypes()
    if not quiet:
        printi('*' * 20)
        printi('Updated python modules')
        printi('*' * 20)
        pprinti(updated)
        printi('')
        printi('*' * 20)
        printi('Updated OMFIT tree entries')
        printi('*' * 20)
        pprinti(OMtr_cls)
    return 
omfit_classes.omfit_python.reload_python = reload_python
# ---------------------
# OMFIT main tree
# ---------------------
_availableModulesCache = {}
[docs]class OMFITmaintree(OMFITproject):
    _save_method = '_save_with_info'
    def __init__(self, filename=''):
        OMFITproject.__init__(self, filename)
        self._OMFITparent = None
        self._OMFITkeyName = ''
        self.prj_options = {}
[docs]    def start(self, filename=''):
        self.clear()
        self.filename = filename
        self.reset()
        self.onlyRunningCopy()
        OMFIT['MainSettings'].sort() 
[docs]    def projectName(self):
        if not len(self.filename):
            return ''
        if re.findall('OMFITsave.txt', self.filename):
            return os.path.split(self.filename.split(os.sep + 'OMFITsave.txt')[0])[1]
        else:
            return os.path.splitext(os.path.split(self.filename)[1])[0] 
[docs]    def onlyRunningCopy(self, deletePID=False):
        """
        :param delete: whether to remove PID from list of running OMFIT processes (this should be done only at exit)
        :return: return True/False wether this is the only running copy of OMFIT on this computer
        """
        filename = os.sep.join([OMFITsessionsDir, str(os.getpid())])
        # OMFITsessionsDir should always exist! -- if not return False
        if not os.path.exists(OMFITsessionsDir):
            printe('Something is wrong! OMFITsessionsDir (%s) has been deleted!' % OMFITsessionsDir)
            return False
        else:
            try:
                # this is in a try/except because
                # in some extreme circumstances there can be
                # errors if the system cannot allocate memory
                if not os.path.exists(filename):
                    open(filename, 'w').close()
                pids = []
                for file in glob.glob(os.sep.join([OMFITsessionsDir, '*'])):
                    pid = os.path.split(file)[1]
                    if is_running(pid):
                        pids.append(pid)
                    elif deletePID:
                        try:
                            os.remove(file)
                        except OSError:
                            pass
                if deletePID and os.path.exists(filename):
                    os.remove(filename)
                return len(pids) == 1
            except OSError:
                return False 
[docs]    def reset(self):
        # always use saving as .zip as the default
        self.zip = True
        # clear
        self.clear()
        OMFITconsoleDict.clear()
        OMFITscriptsDict.clear()
        # remove all files under current temporary directory
        try:
            shutil.rmtree(OMFITcwd)
        except Exception:
            pass
        if not os.path.exists(OMFITcwd):
            os.makedirs(OMFITcwd)
        os.chdir(OMFITcwd)
        # fill special locations
        super().__setitem__('scratch', OMFITmainscratch())
        super().__setitem__('commandBox', OMFITconsoleDict)
        super().__setitem__('scriptsRun', OMFITscriptsDict)
        super().__setitem__('shotBookmarks', OMFITshotBookmarks)
        # create main settings
        self.addMainSettings(restore='user')
        OMFIT['MainSettings']['EXPERIMENT']['runid'] = 'sim1'
        self['MainSettings']['SETUP']['version'] = repo_active_branch_commit
        self['MainSettings']['SETUP']['python_environment'] = SortedDict(python_environment())
        # initialize main scratch
        self['scratch'].initialize()
        # set the localhost
        SERVER.setLocalhost() 
[docs]    def newProvenanceID(self):
        self['MainSettings']['EXPERIMENT']['provenanceID'] = omfit_hash(utils_base.now("%Y-%m-%d_%H_%M_%S_%f")) 
[docs]    def newProjectID(self):
        self['MainSettings']['EXPERIMENT']['projectID'] = 'projectID__'
        self['MainSettings']['EXPERIMENT']['projectID'] += repo_active_branch_commit + '__'
        self['MainSettings']['EXPERIMENT']['projectID'] += utils_base.now("%Y-%m-%d_%H_%M_%S_%f")
        self['MainSettings']['EXPERIMENT']['projectID'] = self['MainSettings']['EXPERIMENT']['projectID'].replace(' ', '') 
[docs]    def addMainSettings(self, updateUserSettings=False, restore=''):
        # read the user namelist file
        self.userMainSettings = OMFITsettingsDir + os.sep + 'MainSettings.txt'
        self.userMainSettingsNamelistDump = OMFITsettingsDir + os.sep + 'MainSettingsNamelistDump.txt'
        self.userMainSettingsNamelist = OMFITsettingsDir + os.sep + 'MainSettingsNamelist.txt'
        if not os.path.exists(os.path.split(self.userMainSettings)[0]):
            os.makedirs(os.path.split(self.userMainSettings)[0])
        if not os.path.exists(self.userMainSettings):
            open(self.userMainSettings, 'w').close()
            updateUserSettings = True
        if not os.path.exists(self.userMainSettingsNamelist):
            open(self.userMainSettingsNamelist, 'w').close()
            updateUserSettings = True
        # keep project options
        for k in list(OMFIT.prj_options_choices.keys()):
            if k not in self.prj_options:
                if k == 'persistent_projectID':
                    self.prj_options[k] = False
                else:
                    self.prj_options[k] = ''
        # force restore of skel main settings if necessary
        self.apply_bindings()
        if 'MainSettings' not in self or restore != '':
            # skeleton settings
            skelMainSettings = OMFITsrc + os.sep + 'omfit_classes' + os.sep + 'skeleton' + os.sep + 'skeletonMainSettings.txt'
            self.tmpSkel = OMFITtree(skelMainSettings, quiet=True)
            if platform.system() == 'Darwin':
                skelMainSettings = (
                    OMFITsrc + os.sep + 'omfit_classes' + os.sep + 'skeleton' + os.sep + 'skeletonMainSettingsNamelistOSX.txt'
                )
            else:
                skelMainSettings = (
                    OMFITsrc + os.sep + 'omfit_classes' + os.sep + 'skeleton' + os.sep + 'skeletonMainSettingsNamelistUNIX.txt'
                )
            self.tmpSkel['MainSettings'].recursiveUpdate(namelist.NamelistFile(skelMainSettings), overwrite=True)
            # institution settings
            if os.path.exists(os.environ.get('OMFIT_INSTITUTION_FILE', '/does not exist')):
                tmp = namelist.NamelistFile(os.environ['OMFIT_INSTITUTION_FILE'])
            else:
                filename = os.sep.join([OMFITsrc, '..', 'institution'])
                if os.path.exists(filename):
                    tmp = namelist.NamelistFile(filename)
                else:
                    institution_files = glob.glob(os.sep.join([OMFITsrc, '..', 'institutions', '']) + '*')
                    institution_files.append(None)
                    for filename in institution_files:
                        tmp = namelist.NamelistFile(filename)
                        if 'SETUP' in tmp and 'stats_file' in tmp['SETUP'] and os.path.exists(tmp['SETUP']['stats_file']):
                            break
            self.tmpSkel['MainSettings'].recursiveUpdate(tmp, overwrite=True)
            # user settings
            self.tmpUser = OMFITtree(self.userMainSettings, quiet=True)
            if 'MainSettings' not in self:
                self['MainSettings'] = OMFITmainSettings()
            self['MainSettings'].filename = OMFITcwd + os.sep + os.path.split(self.userMainSettingsNamelist)[1]
            if not len(self.tmpUser):
                restore = 'skel'
                updateUserSettings = True
        # The main settings are built starting from the skeleton, the user local preference in ~/.OMFIT/ and the edits that the users have made
        if restore == '':
            self['MainSettings'].recursiveUpdate(self.tmpSkel['MainSettings'])
            def f_traverse(me):
                for kid in list(me.keys()):
                    if isinstance(me[kid], dict):
                        f_traverse(me[kid])
                    if re.match(hide_ptrn, kid):
                        del me[kid]
            f_traverse(self['MainSettings'])
            self.add_bindings_to_main_settings()
        elif restore.startswith('diff_'):
            restore = restore[5:]
            tmp = OMFITmainSettings()
            if restore == 'skel':
                tmp.recursiveUpdate(self.tmpSkel['MainSettings'])
            elif restore == 'user' and 'MainSettings' in self.tmpUser:
                tmp.recursiveUpdate(self.tmpSkel['MainSettings'])
                tmp.recursiveUpdate(self.tmpUser['MainSettings'], overwrite=True)
            elif restore == 'S3':
                tmp.recursiveUpdate(self.tmpSkel['MainSettings'])
                tmp.recursiveUpdate(self.tmpUser['MainSettings'], overwrite=True)
                tmp1 = OMFITobject_fromS3(tmp['SETUP']['email'] + "_" + 'MainSettings.txt', s3bucket='omfit')
                tmp2 = OMFITobject_fromS3(tmp['SETUP']['email'] + "_" + 'MainSettingsNamelist.txt', s3bucket='omfit')
                shutil.copy2(tmp2.filename, os.path.split(tmp1.filename)[0] + os.sep + os.path.split(tmp2.filename)[1])
                tmp0 = OMFITtree(tmp1.filename, quiet=True)
                tmp.recursiveUpdate(tmp0['MainSettings'], overwrite=True)
                printi('Loaded MainSettings from the cloud')
            diffTreeGUI(OMFIT['MainSettings'], tmp)
        else:
            self['MainSettings'].clear()
            if restore == 'skel':
                self['MainSettings'].recursiveUpdate(self.tmpSkel['MainSettings'])
            elif restore == 'user' and 'MainSettings' in self.tmpUser:
                self['MainSettings'].recursiveUpdate(self.tmpSkel['MainSettings'])
                self['MainSettings'].recursiveUpdate(self.tmpUser['MainSettings'], overwrite=True)
            elif restore == 'S3':
                self['MainSettings'].recursiveUpdate(self.tmpSkel['MainSettings'])
                self['MainSettings'].recursiveUpdate(self.tmpUser['MainSettings'], overwrite=True)
                tmp1 = OMFITobject_fromS3(self['MainSettings']['SETUP']['email'] + "_" + 'MainSettings.txt', s3bucket='omfit')
                tmp2 = OMFITobject_fromS3(self['MainSettings']['SETUP']['email'] + "_" + 'MainSettingsNamelist.txt', s3bucket='omfit')
                shutil.copy2(tmp2.filename, os.path.split(tmp1.filename)[0] + os.sep + os.path.split(tmp2.filename)[1])
                tmp = OMFITtree(tmp1.filename, quiet=True)
                self['MainSettings'].recursiveUpdate(tmp['MainSettings'], overwrite=True)
                printi('Restored MainSettings from the cloud')
            self.apply_bindings()
        # generate unique projectID if that's not already there
        if 'projectID' not in self['MainSettings']['EXPERIMENT']:
            self.newProjectID()
        # generate unique provenanceID if that's not already there
        if 'provenanceID' not in self['MainSettings']['EXPERIMENT']:
            self.newProvenanceID()
        # make sure auto-save is not more often than 15 minutes
        try:
            self['MainSettings']['SETUP']['autosave_minutes'] = int(self['MainSettings']['SETUP']['autosave_minutes'])
            if self['MainSettings']['SETUP']['autosave_minutes'] < 15:
                raise OMFITexception('Auto-save time must be >= than 15 minutes')
        except Exception as _excp:
            printe('Error in setting Auto-save time\n' + repr(_excp))
            self['MainSettings']['SETUP']['autosave_minutes'] = 15
        # if default_tunnel is None, then set it based on what servers the user has access
        if self['MainSettings']['SERVER']['default_tunnel'] is None:
            favorite_tunnels = ['cybele', 'portal', 'cmodws', 'shenma', 'itm_gateway', 'iter_login', 'cfetr']
            default_tunnel = 'cybele'
            if os.path.exists(os.environ['HOME'] + '/.ssh/known_hosts'):
                with open(os.environ['HOME'] + '/.ssh/known_hosts', 'r') as f:
                    lines = [_f for _f in f.read().strip().split('\n') if _f]
                known_tunnels = []
                for line in lines:
                    try:
                        for server in line.split()[0].split(','):
                            try:
                                SERVER[server]
                                if SERVER(server) in favorite_tunnels:
                                    known_tunnels.append(SERVER(server))
                            except Exception:
                                pass
                    except IndexError:
                        pass
                for tunnel in favorite_tunnels:
                    if tunnel in known_tunnels:
                        default_tunnel = tunnel
                        break
            self['MainSettings']['SERVER']['default_tunnel'] = default_tunnel
        if updateUserSettings:
            # find differences of current MainSettings with respect to skeleton
            tmpU = OMFITtree()
            tmpU['MainSettings'] = copy.deepcopy(OMFIT['MainSettings'])
            ptrn = self.tmpSkel.pretty_diff(tmpU)
            # do not store projectID, provenanceID, or runid
            for k in ['projectID', 'provenanceID', 'runid']:
                if k in list(ptrn['MainSettings']['EXPERIMENT'].keys()):
                    del ptrn['MainSettings']['EXPERIMENT'][k]
            # keep only differences
            tmpU = prune_mask(tmpU, ptrn)
            # remove entries that are deprecated
            _clearDeprecatedMainSettings(tmpU['MainSettings'])
            # remove entries that are not in the skeleton
            for item in list(tmpU['MainSettings'].keys()):
                if item in list(tmpU['MainSettings'].keys()) and item not in self.tmpSkel['MainSettings']:
                    del tmpU['MainSettings'][item]
                if item in list(tmpU['MainSettings'].keys()) and item != 'SERVER':
                    for subitem in list(tmpU['MainSettings'][item].keys()):
                        if (
                            subitem in list(tmpU['MainSettings'][item].keys())
                            and subitem not in self.tmpSkel['MainSettings'][item]
                            and subitem != 'KeyBindings'
                        ):
                            del tmpU['MainSettings'][item][subitem]
                        elif subitem == 'KeyBindings':
                            for desc, event in list(tmpU['MainSettings'][item][subitem].items()):
                                if (
                                    desc in global_event_bindings.default_desc_event
                                    and global_event_bindings.default_desc_event[desc] == event
                                ):
                                    del tmpU['MainSettings'][item][subitem][desc]
            # save only differences (if there are any)
            if len(tmpU):
                tmpU._save(filename=self.userMainSettings, only="['MainSettings']", onlyOMFITsave=True)
                global OMFITexpressionsReturnNone
                tmpExp = OMFITexpressionsReturnNone
                OMFITexpressionsReturnNone = True
                try:
                    tmpU['MainSettings'].saveas(self.userMainSettingsNamelist)
                finally:
                    OMFITexpressionsReturnNone = tmpExp
                self.tmpUser = tmpU
                printi(self.userMainSettings + ' has been updated')
            if updateUserSettings == 'S3':
                OMFITascii(OMFITsettingsDir + os.sep + 'MainSettings.txt').deploy(
                    self['MainSettings']['SETUP']['email'] + "_" + 'MainSettings.txt', s3bucket='omfit'
                )
                OMFITascii(OMFITsettingsDir + os.sep + 'MainSettingsNamelist.txt').deploy(
                    self['MainSettings']['SETUP']['email'] + "_" + 'MainSettingsNamelist.txt', s3bucket='omfit'
                )
                printi('Saved user MainSettings in the cloud')
        # save a version of MainSettings in plain text
        if updateUserSettings or not os.path.exists(self.userMainSettingsNamelistDump):
            self['MainSettings'].deploy(self.userMainSettingsNamelistDump) 
    def __setitem__(self, key, value):
        # TODO: handle 'MainSettings' the same way
        if key in ['scratch', 'commandBox', 'scriptsRun', 'shotBookmarks']:
            raise RuntimeError(f"OMFIT['{key}'] is write-protected")
        else:
            super().__setitem__(key, value)
    def __delitem__(self, key):
        if key in ['scratch', 'commandBox', 'scriptsRun', 'shotBookmarks', 'MainSettings']:
            raise RuntimeError(f"OMFIT['{key}'] is delete-protected")
        else:
            super().__delitem__(key)
[docs]    def add_bindings_to_main_settings(self):
        """
        Take the descriptions and events from global_events_bindings and insert
        them in self['MainSettings']['SETUP']['KeyBindings'][desc] = <event>
        """
        printd('Calling add_bindings_to_main_settings', level=2, topic='keybindings')
        if 'KeyBindings' not in self['MainSettings']['SETUP']:
            self['MainSettings']['SETUP']['KeyBindings'] = namelist.NamelistName()
        for di, d in enumerate(global_event_bindings.descs):
            self['MainSettings']['SETUP']['KeyBindings'][d.replace(' ', '_')] = global_event_bindings.events[di] 
[docs]    def apply_bindings(self):
        """
        Take the descriptions and events from
        self['MainSettings']['SETUP']['KeyBindings']
        and use them to update the global_event_bindings
        """
        if ('MainSettings' not in self) or ('SETUP' not in self['MainSettings']) or ('KeyBindings' not in self['MainSettings']['SETUP']):
            printd("No key bindings in MainSettings['SETUP']['KeyBindings'] -> unable to update", topic='keybindings')
            return
        for d, e in list(self['MainSettings']['SETUP']['KeyBindings'].items()):
            try:
                global_event_bindings.set(d.replace('_', ' '), e)
            except ValueError:
                del self['MainSettings']['SETUP']['KeyBindings'][d]
        self.add_bindings_to_main_settings() 
[docs]    def save(self, quiet=None, skip_save_errors=False):
        """
        Writes the content of the OMFIT tree to the filesystem
        using the same filename and zip options of the last saveas
        :param quiet: whether to print save progress to screen
        :param skip_save_errors: skip errors when saving objects
        """
        if quiet is None:
            quiet = bool(eval(os.environ.get('OMFIT_PROGRESS_BAR_QUIET', '0')))
        self.filename = self._save_with_info(self.filename, zip=self.zip, quiet=quiet, skip_save_errors=skip_save_errors)
        return self.filename 
    def _save_with_info(self, filename, zip=False, quiet=False, updateExistingDir=False, skip_save_errors=False):
        """
        Wrapper function around the OMFITtree._save method which
        handles the prj_options dictionary that is only part of the main OMFIT tree
        """
        # this method should have the same signature of the ._save() method
        if filename:
            # by default the first time the projects directory is created,
            # assign permissions so that it is readeable but not writeable
            # by other users. Users can always override these default permissions
            # once the directory is created.
            projpath = os.path.split(os.path.split(filename)[0])[0]
            if not os.path.exists(projpath):
                # create projects directory
                os.makedirs(projpath)
                # by default, remove group and others write permission
                for perm in [stat.S_IWGRP, stat.S_IWOTH]:
                    st_mode = os.lstat(projpath).st_mode
                    os.chmod(projpath, st_mode & ~perm)
                # by default, add group and others read and browse permission
                for perm in [stat.S_IRGRP, stat.S_IROTH, stat.S_IXGRP, stat.S_IXOTH]:
                    st_mode = os.lstat(projpath).st_mode
                    os.chmod(projpath, st_mode | perm)
            # update list of shots/times from mainSettings and modules settings
            modules = self.moduleDict()
            for k in ['device', 'shot', 'time']:
                self.prj_options[k] = []
            for k in ['device', 'shot', 'shots', 'time', 'times']:
                self.prj_options[k.rstrip('s')].extend(tolist(evalExpr(self['MainSettings']['EXPERIMENT'][k])))
                try:
                    for module in list(modules.keys()):
                        mod = eval('OMFIT' + module)
                        self.prj_options.setdefault(k.rstrip('s'), []).extend(
                            tolist(evalExpr(mod['SETTINGS'].get('EXPERIMENT', {}).get(k, None)))
                        )
                    self.prj_options[k.rstrip('s')] = list([_f for _f in self.prj_options[k.rstrip('s')] if _f])
                except Exception as _excp:
                    printe('Error accessing `%s` of module `OMFIT%s` : %s' % (k, module, repr(_excp)))
            for k in ['device', 'shot', 'time']:
                self.prj_options[k] = np.unique(list([_f for _f in self.prj_options[k.rstrip('s')] if _f])).tolist()
            # if the GUI is open update the GUI elements info
            if OMFITaux['GUI']:
                commands = []
                commandNames = []
                for k in range(len(OMFITaux['GUI'].command)):
                    tmp = OMFITaux['GUI'].command[k].get(1.0, tk.END).strip()
                    if len(tmp):
                        commands.append(OMFITaux['GUI'].command[k].get(1.0, tk.END))
                        commandNames.append(OMFITaux['GUI'].commandNames[k])
                if not len(commands):
                    commands.append('')
                    commandNames.append('1')
                OMFIT.prj_options['commands'] = commands
                OMFIT.prj_options['commandNames'] = commandNames
                OMFIT.prj_options['namespace'] = OMFITaux['GUI'].commandBoxNamespace
                OMFIT.prj_options['notes'] = OMFITaux['GUI'].notes.get(1.0, tk.END).strip()
                OMFIT.prj_options['console'] = OMFITaux['console'].get()
                OMFIT.prj_options.setdefault('color', '')
                if OMFIT.prj_options['color'] not in list(OMFIT.prj_options_choices['color'].keys()):
                    OMFIT.prj_options['color'] = ''
                if OMFIT.prj_options['type'] not in OMFIT.prj_options_choices['type']:
                    OMFIT.prj_options['type'] = ''
            try:
                OMFITaux['hardLinks'] = True
                self['__GUISAVE__'] = self.prj_options
                cb = OMFITtree()
                for k, v in enumerate(self['__GUISAVE__']['commands']):
                    if not v.strip():
                        continue
                    try:
                        cb['%d.py' % (k + 1)] = OMFITascii(filename='%d.py' % (k + 1), fromString=v.rstrip())
                    except Exception as _excp:
                        printe(_excp)
                if list(cb.keys()):
                    self['__COMMANDBOX__'] = cb
                orig_version = self['MainSettings']['SETUP']['version']
                orig_environ = self['MainSettings']['SETUP']['python_environment']
                self['MainSettings']['SETUP']['version'] = repo_active_branch_commit
                self['MainSettings']['SETUP']['python_environment'] = SortedDict(python_environment())
                return self._save(
                    filename,
                    zip=zip,
                    quiet=quiet,
                    skipStorage=False,
                    updateExistingDir=updateExistingDir,
                    skip_save_errors=skip_save_errors,
                )
            except Exception:
                self['MainSettings']['SETUP']['version'] = orig_version
                self['MainSettings']['SETUP']['python_environment'] = orig_environ
                raise
            finally:
                if '__GUISAVE__' in self:
                    del self['__GUISAVE__']
                if '__COMMANDBOX__' in self:
                    del self['__COMMANDBOX__']
                OMFITaux['hardLinks'] = False
        else:
            raise IOError('Last saved directory was not defined')
[docs]    def saveas(self, filename, zip=None, quiet=None, skip_save_errors=False):
        """
        Writes the content of the OMFIT tree to the filesystem
        and permanently changes the name of the project
        :param filename: project filename to save to
        :param zip: whether the save should occur as a zip file
        :param quiet: whether to print save progress to screen
        :param skip_save_errors: skip errors when saving objects
        """
        if quiet is None:
            quiet = bool(eval(os.environ.get('OMFIT_PROGRESS_BAR_QUIET', '0')))
        oldZip = self.zip
        oldFilename = self.filename
        oldPrj_options = self.prj_options
        if zip is not None:
            self.zip = zip
        self.filename = filename
        try:
            return self.save(quiet=quiet, skip_save_errors=skip_save_errors)
        except Exception:
            self.zip = oldZip
            self.filename = oldFilename
            self.prj_options = oldPrj_options
            raise 
[docs]    def loadModule(
        self,
        filename,
        location=None,
        withSubmodules=True,
        availableModulesList=None,
        checkLicense=True,
        developerMode=None,
        depth=0,
        quiet=False,
        startup_lib=None,
        **kw,
    ):
        r"""
        Load a module in OMFIT
        :param filename: * full path to the module to be loaded
                         * if just the module name is provided, this will be loaded from the public modules
                         * remote/branch:module format will load a module from a specific git remote and branch
                         * module:remote/branch format will load a module from a specific git remote and branch
        :param location: string with the location where to place the module in the OMFIT tree
        :param withSubmodules: load submodules or not
        :param availableModulesList: list of available modules generated by ``OMFIT.availableModules()``
                                     If this list is not passed, then the availableModulesList is generated internally
        :param checkLicense: Check license files at load
        :param developerMode: Load module with developer mode option (ie. scripts loaded as modifyOriginal)
                              if None then default behavior is set by ``OMFIT['MainSettings']['SETUP']['developer_mode_by_default']``
                              Note: public OMFIT installation cannot be loaded in developer mode
        :param depth: parameter used internally by for keeping track of the recursion depth
        :param quiet: load modules silently or not
        :param startup_lib: Used internally for executing OMFITlib_startup scripts
        :param \**kw: additional keywords passed to OMFITmodule() class
        :return: instance of the loaded module
        """
        if os.sep not in filename or ':' in filename:
            # if there is a semicolumn in the filename, then a specific remote/branch is been requested
            if ':' in filename:
                work_repo = repo.clone()
                remote, branch = [x for x in filename.split(':') if '/' in x][0].split('/')
                filename = [x for x in filename.split(':') if '/' not in x][0]
                work_repo.switch_branch(branch, remote)
                filename = os.sep.join([work_repo.git_dir, 'modules', filename, 'OMFITsave.txt'])
            else:
                # if it's just a module name, load it from the first of the OMFITmodule directories
                filename = os.sep.join([OMFITmodule.directories()[0], filename, 'OMFITsave.txt'])
        filename = os.path.abspath(filename)
        # get list of available modules
        if availableModulesList is None:
            availableModulesList = self.availableModules(quiet=True, same_path_as=filename)
        # handle developerMode switch
        OMFITsrc = os.path.split(os.path.split(os.path.split(filename)[0])[0])[0] + os.sep + 'omfit'
        if os.path.exists(os.sep.join([OMFITsrc, '..', 'public'])):
            developerMode = False
        if developerMode is None:
            developerMode = OMFIT['MainSettings']['SETUP']['developer_mode_by_default']
        # load the module
        moduleName = OMFITmodule.info(filename).get('ID', 'unknown_module')
        if location is None:
            location = "self['" + moduleName + "']"
        # handle printing
        kw.pop('quiet', None)
        quiet_module = True
        if not quiet:
            quiet_module = {
                'newline': False,
                'clean': False,
                'quiet': bool(eval(os.environ.get('OMFIT_PROGRESS_BAR_QUIET', '0'))),
                'style': ' [{sfill}{svoid}] {perc:3.2f}% ' + '*' * (depth + 1) + ' ' + moduleName + '{mess}',
            }
        # load module
        tmp = OMFITmodule(filename, developerMode=developerMode, quiet=quiet_module, **kw)
        # remove the module filename as there is no point of keeping it once it's loaded from the repo
        # OMFITtree (and derived classes) filenames are used for saving them external to the project
        tmp.filename = ''
        # if loading as a submodule, then place at the top
        loc = parseLocation(location)
        if isinstance(eval(buildLocation(loc[:-1])), OMFITmodule):
            eval(buildLocation(loc[:-1])).insert(0, loc[-1], tmp)
        else:
            eval(buildLocation(loc[:-1]))[loc[-1]] = tmp
        # add extra commit info to MODULE
        if filename in availableModulesList:
            for k in _moduleAttrs:
                if k in availableModulesList[filename]:
                    tmp['SETTINGS']['MODULE'][k] = availableModulesList[filename][k]
        # fill in missing infos (e.g. when starting a new module)
        if not tmp['SETTINGS']['MODULE']['commit']:
            tmp['SETTINGS']['MODULE']['commit'] = repo.get_hash('HEAD')
        if tmp['SETTINGS']['MODULE']['contact'] is None:
            tmp['SETTINGS']['MODULE']['contact'] = []
        if not len(tmp['SETTINGS']['MODULE']['contact']) and OMFIT['MainSettings']['SETUP']['email']:
            tmp['SETTINGS']['MODULE']['contact'].append(OMFIT['MainSettings']['SETUP']['email'])
        # Check license
        if checkLicense and 'license' in eval(location):
            license = eval(location)['license']
            email_dict = {}
            with open(license.filename, 'r') as f:
                for k in f.readlines():
                    k = str(k)
                    if 'OMFIT_license_email_notify' in k:
                        notify = [x.strip() for x in re.sub(r'OMFIT_license_email_notify\:(.*)', r'\1', k).strip().split(',')]
                        email_dict = {
                            'email_address': notify,
                            'email_prompt': 'Please describe planned research using ' + moduleName,
                            'fromm': self['MainSettings']['SETUP']['email'],
                        }
                        break
            try:
                OMFITlicenses[moduleName] = License(moduleName, license.filename, email_dict=email_dict, rootGUI=OMFITaux['rootGUI'])
                if OMFITaux['rootGUI']:
                    OMFITlicenses[moduleName].check()
            except LicenseException:
                exec('del ' + location, globals(), locals())
                raise
            except tk.TclError:  # If there is no DISPLAY, allow access
                pass
            except IndexError:  # If gui element fails to load, allow access
                pass
        # load the submodules (will set DEPENDENCIES, SETTINGS['EXPERIMENT'], and honor LIB['OMFITlib_startup'] scripts)
        if withSubmodules:
            # find the submodules (one level deep)
            moduleDict = eval(location).moduleDict(level=0)
            # save the dependencies
            depsDict = {}
            for subMod in moduleDict:
                depsDict[subMod] = eval(location + subMod)['SETTINGS']['DEPENDENCIES']
            # load the submodules
            if startup_lib is None:
                startup_lib = []
            for subMod in moduleDict:
                found = False
                for avMod in availableModulesList:
                    if moduleDict[subMod]['ID'] == availableModulesList[avMod]['ID']:
                        OMFIT.loadModule(
                            availableModulesList[avMod]['path'],
                            location + subMod,
                            withSubmodules=True,
                            availableModulesList=availableModulesList,
                            checkLicense=checkLicense,
                            developerMode=developerMode,
                            depth=depth + 1,
                            quiet=quiet,
                            startup_lib=startup_lib,
                            **kw,
                        )
                        found = True
                        break
                if not found:
                    raise IOError(
                        'Module %s is a submodule of %s, and it could not found in %s'
                        % (moduleDict[subMod]['ID'], moduleName, os.path.split(os.path.split(filename)[0])[0])
                    )
            # restore the dependencies
            for subMod in moduleDict:
                for avMod in availableModulesList:
                    if moduleDict[subMod]['ID'] == availableModulesList[avMod]['ID']:
                        eval(location + subMod)['SETTINGS']['DEPENDENCIES'].update(depsDict[subMod])
                        break
            # freeze the root['SETTINGS']['EXPERIMENT'] entries if this is a top-level import that is not a submodule
            if depth == 0 and 'SETTINGS' in eval(location) and 'EXPERIMENT' in eval(location)['SETTINGS']:
                freeze = True
                for tmpName in reversed(traverseLocation(location)[:-1]):
                    if eval(tmpName).__class__ is OMFITmodule and tmpName != 'OMFIT':
                        freeze = False
                if freeze:
                    for item in list(OMFITaux['moduleSkeletonCache']['SETTINGS']['EXPERIMENT'].keys()):
                        eval(location)['SETTINGS']['EXPERIMENT'][item] = evalExpr(eval(location)['SETTINGS']['EXPERIMENT'][item])
            # execute startup script (the OMFITmodulePath variable is setup to point to the OMFITsave.txt file of the module that is being loaded)
            if 'LIB' in eval(location) and 'OMFITlib_startup' in eval(location)['LIB']:
                startup_lib.append((moduleName, location, eval(location)['LIB']['OMFITlib_startup']))
            if depth == 0:
                for k, (mname, loc, st_lib) in enumerate(startup_lib):
                    if not quiet:
                        if k == 0:
                            printi("Running OMFITlib_startup:", end='')
                        printi(f' {mname}', end='')
                    try:
                        with quiet_environment():
                            st_lib.runNoGUI()
                    except Exception as _excp:
                        printe("Error in %s['LIB']['OMFITlib_startup'] : " % (loc) + repr(_excp))
                if not quiet:
                    printi('')
            # store settings at load time
            # NOTE: this is done both here and in OMFITmodule.__init__ because this one will get execution of OMFITlib_startup and the other instead is needed when reloading a module
            eval(location)._save_settings_at_import()
        # return the module
        return eval(location) 
[docs]    def load(self, filename, persistent_projectID=False):
        """
        loads an OMFITproject in OMFIT
        :param filename: filename of the project to load (if `-1` then the most recent project is loaded)
        :param persistent_projectID: whether the projectID should remain the same
        """
        try:
            # open the last project
            if filename == '-1':
                projects = OMFIT.recentProjects()
                if len(projects):
                    filename = list(projects.keys())[0]
                    persistent_projectID = projects[filename].get('persistent_projectID', False)
            # keep aside user 'SETUP','SERVER'
            tmp = {}
            for item in ['SETUP', 'SERVER']:
                if item in self['MainSettings']:
                    tmp[item] = copy.deepcopy(self['MainSettings'][item])
            self.filename = os.path.abspath(filename)
            oldDir = os.getcwd()
            TMPnoCopyToCWD = OMFITaux['noCopyToCWD']
            OMFITaux['noCopyToCWD'] = False
            try:
                self._load(filename)
            finally:
                os.chdir(oldDir)
                OMFITaux['noCopyToCWD'] = TMPnoCopyToCWD
            if zipfile.is_zipfile(self.filename):
                self.zip = True
                projectName = os.path.split(self.filename)[1]
            else:
                self.zip = False
                projectName = os.path.split(os.path.split(self.filename)[0])[1]
            # for older projects we calculate provenanceID based on institution and projectName
            # very old projects do not even have institution info
            if 'provenanceID' not in self['MainSettings']['EXPERIMENT']:
                self['MainSettings']['EXPERIMENT']['provenanceID'] = omfit_hash(
                    self['MainSettings']['SETUP'].get('institution', 'old_project') + '_' + projectName
                )
            # SETUP is overwritten (but not version nor python_environment)
            project_version = self['MainSettings']['SETUP'].get('version', '?')
            project_environ = self['MainSettings']['SETUP'].get('python_environment', {})
            self['MainSettings']['SETUP'] = tmp['SETUP']
            self['MainSettings']['SETUP']['version'] = project_version
            self['MainSettings']['SETUP']['python_environment'] = project_environ
            # SERVER is updated
            self['MainSettings'].setdefault('SERVER', NamelistName()).update(tmp.get('SERVER', {}))
            # load project options
            self.prj_options.clear()
            tmp = OMFITproject.info(self.filename)
            for k in OMFIT.prj_options_choices:
                self.prj_options[k] = tmp.get(k, '')
            for whereSave in ['__GUISAVE__', '__COMMANDBOX__']:
                if whereSave in OMFIT:
                    del OMFIT[whereSave]
            if self.prj_options['color'] not in list(self.prj_options_choices['color'].keys()):
                self.prj_options['color'] = ''
            if self.prj_options['type'] not in self.prj_options_choices['type']:
                self.prj_options['type'] = ''
            if 'namespace' not in self.prj_options or not self.prj_options['namespace']:
                self.prj_options['namespace'] = 'OMFIT'
            # update projectID
            self.prj_options['persistent_projectID'] = persistent_projectID
            if not persistent_projectID:
                self.newProjectID()
            # keep MainSettings in check
            self.updateMainSettings()
            self['MainSettings'].sort()
            # set the localhost
            SERVER.setLocalhost()
            # remote projects are loaded in OMFITtmpDir
            # reset the filename to force users to decide how/where to save it
            if OMFITtmpDir in self.filename:
                self.filename = ''
        except Exception:
            self.start()
            raise 
[docs]    def updateCWD(self):
        global OMFITcwd
        self._save(filename=OMFITcwd + os.sep + 'OMFITsave.txt', skipStorage=False) 
[docs]    def updateMainSettings(self):
        self.addMainSettings()
        self.keyOrder.remove('scratch')
        self.keyOrder.append('scratch')
        self.keyOrder.remove('commandBox')
        self.keyOrder.append('commandBox')
        self.keyOrder.remove('scriptsRun')
        self.keyOrder.append('scriptsRun')
        self.keyOrder.remove('shotBookmarks')
        self.keyOrder.append('shotBookmarks')
        self.keyOrder.remove('MainSettings')
        self.keyOrder.append('MainSettings') 
[docs]    def saveMainSettingsSkeleton(self):
        """
        This utility function updates the MainSettingsSkeleton for the current OMFIT installation
        """
        self['MainSettings'].sort()
        tmp = copy.deepcopy(self['MainSettings'])
        self.addMainSettings(restore='skel')
        keys = self['MainSettings'].keys()
        for k in keys:
            if k.startswith('__comment'):
                del self['MainSettings'][k]
        self['MainSettings'].sort()
        try:
            self['MainSettings'].filename = self['MainSettings']['SETUP']['workDir'] + '../' + 'skeletonMainSettingsNamelist.txt'
            self['MainSettings']['SETUP']['version'] = '?'
            self['MainSettings']['SETUP']['python_environment'] = {}
            del self['MainSettings']['EXPERIMENT']['projectID']
            del self['MainSettings']['EXPERIMENT']['provenanceID']
            self['MainSettings']['SETUP']['browser'] = None
            self['MainSettings']['SETUP']['Extensions'].clear()
            # Avoid changes made by institution file
            self['MainSettings']['SETUP']['stats_file'] = OMFITexpression("os.path.abspath(OMFITsettingsDir+'/OMFITstats.txt')")
            self['MainSettings']['SETUP']['institution'] = 'PERSONAL'
            if 'institutionProjectsDir' in self['MainSettings']['SETUP']:
                del self['MainSettings']['SETUP']['institutionProjectsDir']
            self['MainSettings']['EXPERIMENT']['device'] = None
            for k in tmp['SERVER']:
                if k not in self['MainSettings']['SERVER']:
                    self['MainSettings']['SERVER'][k] = tmp['SERVER'][k]
                if isinstance(tmp['SERVER'][k], namelist.NamelistName):
                    self['MainSettings']['SERVER'][k].update(tmp['SERVER'][k])
            self['MainSettings']['SERVER']['localhost'] = namelist.NamelistName()
            self['MainSettings']['SERVER']['localhost']['workDir'] = OMFITexpression("OMFITcwd+os.sep+'remote_runs'+os.sep")
            self['MainSettings']['SERVER']['localhost']['server'] = 'localhost'
            self['MainSettings']['SERVER']['localhost']['tunnel'] = ''
            self['MainSettings']['SERVER']['localhost']['idl'] = ''
            self['MainSettings']['SERVER']['localhost']['matlab'] = ''
            self['MainSettings']['SERVER']['default_tunnel'] = None
        finally:
            self['MainSettings'].sort()
            self._save(
                filename=OMFITsrc + os.sep + 'omfit_classes' + os.sep + 'skeleton' + os.sep + 'skeletonMainSettings.txt',
                only="['MainSettings']",
                updateExistingDir=True,
            )
            self['MainSettings'] = tmp 
[docs]    def availableModules(self=None, quiet=None, directories=None, same_path_as=None, force=False):
        """
        Index available OMFIT modules in directories
        :param quiet: verbosity
        :param directories: list of directories to index. If `None` this is taken from OMFIT['MainSettings']['SETUP']['modulesDir']
        :param same_path_as: sample OMFITsave.txt path to set directory to index
        :param force: force rebuild of .modulesInfoCache file
        :return: This method returns a dictionary with the available OMFIT modules.
                 Each element in the dictionary is a dictionary itself with the details of the available modules.
        """
        if quiet is None:
            quiet = bool(eval(os.environ.get('OMFIT_PROGRESS_BAR_QUIET', '0')))
        if directories is None:
            directories = OMFITmodule.directories()
        else:
            directories = tolist(directories)
        if same_path_as is None:
            pass
        elif 'OMFITsave.txt' in same_path_as and os.path.exists(same_path_as):
            directories = [os.path.split(os.path.split(same_path_as)[0])[0]]
        else:
            raise IOError('%s file is not a valid OMFITsave.txt file' % same_path_as)
        updatePersistentCache = []
        moduleDict = {}
        for directory in directories:  # the user inputs the directory where the modules are
            if not quiet:
                tmp = 'Modules directory: ' + directory
                printi('=' * len(tmp))
                printi(tmp)
                printi('=' * len(tmp))
                print('', end='')
            if not force and os.path.exists(directory + os.sep + '.modulesInfoCache'):
                try:
                    with open(directory + os.sep + '.modulesInfoCache', 'rb') as f:
                        _availableModulesCache[directory] = pickle.load(f)
                except Exception:
                    _availableModulesCache[directory] = {}
                    printw('-- Failed to load persistent git modules cache --')
            else:
                _availableModulesCache[directory] = {}
            # modules in directory
            modules_list_in_dir = sorted(
                [os.path.abspath(x) for x in glob.glob(os.sep.join([directory, '*', 'OMFITsave.txt']))], key=lambda x: x.lower()
            )
            # try handling git repository of modules twice
            # first as if git root directory parent directory of modules directory
            # second as if git root directory is modules directory
            cases_errors = []
            cases = []
            if os.path.split(directory)[1] == 'modules':
                cases.append([os.path.split(directory)[0], 'modules'])
            cases.append([directory, ''])
            for repo_dir, modules_subpath in cases:
                try:
                    # try to access as a git repository
                    repo = OMFITgit(repo_dir)
                    commit = repo("log -1 --pretty='%H'")
                    branch = repo('rev-parse --abbrev-ref HEAD')
                    remote = repo.get_branches().get(branch, {'remote': ''})['remote']
                    changed_modules = []
                    # if same remote/branch and commit
                    if (
                        np.all([k in _availableModulesCache[directory] for k in ['branch', 'remote', commit]])
                        and _availableModulesCache[directory]['remote'] == remote
                        and _availableModulesCache[directory]['branch'] == branch
                    ):
                        changes = list([_f for _f in repo("ls-files -m --others --exclude-standard " + directory).split('\n') if _f])
                        for mod_path in modules_list_in_dir:
                            if (
                                mod_path not in _availableModulesCache[directory][commit]
                                or len(_availableModulesCache[directory][commit][mod_path]['modified'])
                                or len(_availableModulesCache[directory][commit][mod_path]['untracked'])
                            ):
                                changes.append(os.sep.join(mod_path.split(os.sep)[[-2, -3][len(modules_subpath) > 0] :]))
                        if len(changes):
                            if not quiet:
                                printi('-- some modules changes are not part of git --')
                            for mod in tolist(np.unique([x.split(os.sep)[[0, 1][len(modules_subpath) > 0]] for x in changes])):
                                mod_path = directory + os.sep + mod
                                if not os.path.exists(mod_path):
                                    raise Exception(mod_path + ' does not exist')
                                tmp = repo.get_module_params(quiet=quiet, path=modules_subpath, key='path', modules=[mod_path])
                                _availableModulesCache[directory][commit].update(tmp)
                                changed_modules.append(os.path.abspath(mod_path + os.sep + 'OMFITsave.txt'))
                        elif not quiet:
                            printi('-- modules cache is valid --')
                    else:
                        # if same remote/branch then try to update since recorded commit (faster: only need to loop over modules that changed)
                        if (
                            np.all([k in _availableModulesCache[directory] for k in ['branch', 'remote']])
                            and len(_availableModulesCache[directory]) == 3
                            and _availableModulesCache[directory]['remote'] == remote
                            and _availableModulesCache[directory]['branch'] == branch
                        ):
                            printi('-- updating modules cache since last commit --')
                            try:
                                old_commit = [
                                    item for item in list(_availableModulesCache[directory].keys()) if item not in ['branch', 'remote']
                                ][0]
                                changes = np.unique(
                                    [
                                        modules_subpath + '/' + re.sub('^' + modules_subpath + '/', '', x).split(os.sep)[0]
                                        for x in repo('diff --name-only %s %s -- %s' % (old_commit, commit, modules_subpath)).split()
                                    ]
                                ).tolist()
                                changed_modules = [os.sep.join([repo_dir, x, 'OMFITsave.txt']) for x in changes]
                                _availableModulesCache[directory][commit] = _availableModulesCache[directory][old_commit]
                                del _availableModulesCache[directory][old_commit]
                                _availableModulesCache[directory][commit].update(
                                    repo.get_module_params(quiet=quiet, path=modules_subpath, key='path', modules=changed_modules)
                                )
                            except Exception as _excp:
                                printw('Failed to update available modules cache: ' + repr(_excp))
                                del _availableModulesCache[directory]
                        else:
                            printi('-- rebuilding modules cache from scratch --')
                            if directory in _availableModulesCache:
                                del _availableModulesCache[directory]
                        # if directory is not in _availableModulesCache then must build everything from scratch
                        if directory not in _availableModulesCache:
                            _availableModulesCache[directory] = {}
                            _availableModulesCache[directory]['branch'] = branch
                            _availableModulesCache[directory]['remote'] = remote
                            _availableModulesCache[directory][commit] = repo.get_module_params(
                                quiet=quiet, path=modules_subpath, key='path'
                            )
                        updatePersistentCache.append(directory)
                    # delete modules from cache if these are not found on hard drive
                    for mod_path in list(_availableModulesCache[directory][commit].keys()):
                        if not os.path.exists(_availableModulesCache[directory][commit][mod_path]['path']):
                            del _availableModulesCache[directory][commit][mod_path]
                    # update list of modules
                    moduleDict.update(_availableModulesCache[directory][commit])
                    # do not try 2nd case if 1st one worked
                    break
                except Exception as _excp:
                    cases_errors.append(_excp)
                    if len(cases_errors) == len(cases):
                        _availableModulesCache[directory].clear()
                        # if the path is not a valid git repository or the data was modified then get modules by hand
                        for mod_path in modules_list_in_dir:
                            if mod_path in moduleDict:
                                continue
                            mod = os.path.split(mod_path)[0]
                            mod_name = os.path.split(mod)[1]
                            if not quiet:
                                printi(mod_name.ljust(20) + ' (untracked)')
                            moduleDict[mod_path] = {}
                            moduleDict[mod_path]['path'] = mod_path
                            moduleDict[mod_path]['untracked'] = mod_path  # just for compatibility with data returned by `get_module_params`
                            moduleDict[mod_path]['modified'] = mod_path  # just for compatibility with data returned by `get_module_params`
                            moduleDict[mod_path]['edited_by'] = os.environ['USER']
                            moduleDict[mod_path]['date'] = convertDateFormat(time.time())
                            moduleDict[mod_path]['commit'] = ''
                            moduleDict[mod_path]['description'] = ''
                            info = OMFITmodule.info(moduleDict[mod_path]['path'])
                            info.pop('date', None)
                            info.pop('edited_by', None)
                            info.pop('commit', None)
                            moduleDict[mod_path].update(info)
            # save modules info cache
            if directory in updatePersistentCache or not os.path.exists(directory + os.sep + '.modulesInfoCache'):
                try:
                    with open(directory + os.sep + '.modulesInfoCache', 'wb') as f:
                        pickle.dump(_availableModulesCache[directory], f, pickle.OMFIT_PROTOCOL)
                    printi('-- Updated persistent git modules cache --')
                except IOError:
                    pass
        return moduleDict 
[docs]    def quit(self, deepClean=False):
        """
        Cleanup current OMFIT session directory `OMFITsessionDir` and PID from `OMFITsessionsDir`
        Also close all SSH related tunnels and connections
        :param deepClean: if deepClean is True, then the `OMFITtmpDir` and `OMFITsessionsDir` get deleted
        """
        if OMFITcwd is None:
            return
        onlyRunningCopy = OMFIT.onlyRunningCopy(deletePID=True)
        if onlyRunningCopy:
            OMFIT.reset_connections()  # reset_connections() will also close_mds_connections()
        else:
            omfit_classes.omfit_mds.close_mds_connections()
        try:
            if os.path.exists(OMFITsessionDir):
                shutil.rmtree(OMFITsessionDir)
        except Exception:
            printw('Some files and directories could not be deleted!')
        if deepClean:
            if os.path.exists(OMFITsessionsDir):
                try:
                    shutil.rmtree(OMFITsessionsDir)
                    os.makedirs(OMFITsessionsDir)
                except OSError:
                    printw('Some files and directories under %s could not be deleted!' % OMFITsessionsDir)
            if os.path.exists(OMFITtmpDir):
                try:
                    shutil.rmtree(OMFITtmpDir)
                except Exception:
                    printw('Some files and directories under %s could not be deleted!' % OMFITtmpDir)
        return onlyRunningCopy 
[docs]    def reset_connections(self, server=None, mds_cache=True):
        """
        Reset connections, stored passwords, and MDSplus cache
        :param server: only reset SSH connections to this server
        """
        import utils_widgets
        serverPicker = ''
        if server is None:
            printi('Reset OMFIT stored passwords')
            utils_widgets.stored_passwords.clear()
            printi('Reset OMFIT server cache')
            SERVER_cache.clear()
            if mds_cache:
                printi('Purge OMFIT MDSplus cache')
                OMFITmdsCache().purge()
            else:
                printi('OMFIT MDSplus cache not purged; use OMFITmdsCache().purge() to purge it')
            printi('Reset OMFIT SSH  connections')
            printi('Reset OMFIT MDSplus connections')
            omfit_classes.omfit_mds.close_mds_connections()
            printi('Reset OMFIT SQL  connections')
            for k in ['MDSserverReachable', 'RDBserverReachable', 'batch_js', 'sshTunnel', 'sysinfo']:
                OMFITaux[k] = copy.deepcopy(OMFITaux.defaults[k])
        else:
            if server:
                serverPicker = parse_server(SERVER[server]['server'])[2]
            printi('Reset OMFIT SSH  connections to %s' % serverPicker)
        sshServices = ['OMFITssh', 'OMFITsshTunnel', 'OMFITsshControlmaster', 'OMFITsshGit']
        # Have to create tmpdir since OMFIT tmp directory may have been cleaned up already
        import tempfile, shutil
        tmpdir = tempfile.mkdtemp()
        s = subprocess.Popen("ps xw", stdout=subprocess.PIPE, shell=True, cwd=tmpdir)
        std_out, std_err = s.communicate()
        shutil.rmtree(tmpdir)
        for item in b2s(std_out).split('\n'):
            service = [s for s in sshServices if re.search(s, item)]
            if len(service) and re.search(serverPicker, item):
                pid = int([_f for _f in item.split(' ') if _f][0])
                try:
                    printi('kill %s' % (item))
                    os.kill(int(pid), signal.SIGKILL)
                except OSError as e:
                    printw('Failed! %s' % repr(e)) 
[docs]    def recentProjects(self, only_read=False):
        """
        This routine keeps track of the recent projects
        and is also in charge of deleting auto-saves if they do not appear in the project list.
        :param read_only: only read projects file (don't do any maintenance on it)
        """
        # do not log projects that are in OMFITtmpDir
        if OMFITtmpDir in OMFIT.filename:
            return
        filename_old = OMFITsettingsDir + os.sep + 'OMFITrecentProjects.txt'
        filename = OMFITsettingsDir + os.sep + 'OMFITprojects.txt'
        autoSaveLinkDir = os.path.abspath(OMFIT['MainSettings']['SETUP']['projectsDir'].rstrip('/\\')) + os.sep + 'auto-save' + os.sep
        if not os.path.exists(filename) and os.path.exists(filename_old):
            shutil.copyfile(filename_old, filename)
        # read the projects
        projects = SortedDict()
        if os.path.exists(filename):
            with open(filename, 'r') as f:
                lines = f.readlines()
            for k, line in enumerate(lines):
                line = line.strip()
                tmp = line.split('{')
                if len(tmp) == 1:
                    path = tmp[0].strip()
                    opts = {}
                else:
                    path = tmp[0].strip()
                    opts = eval('{' + '{'.join(tmp[1:]))
                if os.path.exists(path):
                    projects[path] = opts
        # add extra info (eg. file size)
        for item in list(projects.keys()):
            if 'size' not in projects[item] and os.path.exists(item) and item.endswith('.zip') and zipfile.is_zipfile(item):
                projects[item]['size'] = os.stat(item).st_size
        if only_read:
            return projects
        # this switch will be set to True if the recent projects file needs to be updated
        doWrite = False
        # delete `Self-Destruct` projects that have not been accessed in more than 30 days
        for item in list(projects.keys()):
            if projects[item].get('type', '') == 'Self-Destruct' and os.stat(item).st_atime < (time.time() - 30 * 86400):
                if os.path.isdir(item):
                    shutil.rmtree(item)
                else:
                    os.remove(item)
                del projects[item]
                doWrite = True
        # add projects which for some reason were not already there at the bottom of the list
        if os.path.exists(os.path.abspath(evalExpr(OMFIT['MainSettings']['SETUP']['projectsDir']))):
            for item in glob.glob(os.path.abspath(evalExpr(OMFIT['MainSettings']['SETUP']['projectsDir'])) + os.sep + '*'):
                if os.path.isdir(item):
                    if os.path.exists(item + os.sep + 'OMFITsave.txt'):
                        item = item + os.sep + 'OMFITsave.txt'
                    else:
                        continue
                if item not in projects:
                    projects[item] = {}
                    doWrite = True
        # try to read prj_options for the projects which do not have them
        for item in list(projects.keys()):
            if 'device' not in projects[item]:
                try:
                    tmp = OMFITproject.info(item)
                except Exception as _excp:
                    printe('%s looks like an invalid OMFIT project' % (item))
                    del projects[item]
                    continue
                tmp.setdefault('notes', '')
                for opt in OMFIT.prj_options_choices:
                    if opt in tmp:
                        projects[item][opt] = tmp[opt]
                        doWrite = True
        # if the current project has been saved put it at the top of the list
        if (
            OMFIT.filename is not None
            and len(OMFIT.filename)
            and autoSaveLinkDir not in OMFIT.filename
            and OMFITautosaveDir not in OMFIT.filename
        ):
            item = os.path.abspath(OMFIT.filename)
            if item in list(projects.keys()):
                del projects[item]
            projects.insert(0, item, copy.deepcopy(OMFIT.prj_options))
            if item.endswith('.zip') and zipfile.is_zipfile(item):
                projects[item]['size'] = os.stat(item).st_size
            elif item.endswith('OMFITsave.txt'):
                projects[item]['size'] = size_of_dir(os.path.split(item)[0])
            doWrite = True
        # do not store console output, commands, and namespace in OMFITprojects.txt
        for item in list(projects.keys()):
            for opt in ['console', 'commands', 'namespace']:
                if opt in projects[item]:
                    del projects[item][opt]
                    doWrite = True
        # only store modules locations
        for item in list(projects.keys()):
            if isinstance(projects[item]['modules'], dict):
                projects[item]['modules'] = list(projects[item]['modules'].keys())
        # write OMFITprojects.txt~ then move to OMFITprojects.txt if successful
        # this is to have as much of an atomic operation as possible
        if doWrite:
            with open(filename_old + '~', 'w') as f_old, open(filename + '~', 'w') as f:
                for item in list(projects.keys()):
                    f.write('%s %s\n' % (item, projects[item]))
                    f_old.write('%s\n' % (item))
            shutil.move(filename + '~', filename)
            shutil.move(filename_old + '~', filename_old)
        # delete auto-saves that are older than one week
        # process one old auto-save at the time ones that were accessed first (deleting large auto-saves can take some time)
        for dir in [OMFITautosaveDir, autoSaveLinkDir]:
            # sort auto-saves by time
            files = glob.glob(dir + os.sep + '*')
            # delete broken links (otherwise getatime will fail)
            remove = []
            for path in files:
                if os.path.islink(path) and not os.path.exists(path):
                    os.remove(path)
                    remove.append(path)
            files = list(set(files).difference(set(remove)))
            # sort files
            files.sort(key=lambda x: os.path.getatime(x))
            for path in files:
                # delete auto-saves that have not been accessed in more than one week
                if not os.path.islink(path) and os.path.isdir(path) and os.stat(path).st_atime < (time.time() - 7 * 86400):
                    shutil.rmtree(path)
                    break  # delete one old auto-save at the time
                # delete broken links
                if os.path.islink(path) and not os.path.exists(path):
                    os.remove(path)
        return projects 
[docs]    def showExecDiag(self):
        """
        display execution diagram
        """
        def printMe(tmp, depth):
            for child in tmp:
                filename = re.sub('__console__.py', '', os.path.split(child['filename'])[1])
                printi('* ' * (depth + 1) + " %s %3.3fs [%s]" % (filename, child['time'], sizeof_fmt(child['memory'])))
                printMe(child['child'], depth + 1)
        printi('\nExecution diagram:\n------------------')
        from omfit_classes.omfit_python import ExecDiagPtr
        printMe(ExecDiagPtr, 0)  
[docs]class OMFITmainscratch(OMFITtmp):
    def __init__(self):
        self.required = list(map(lambda x: os.path.splitext(os.path.basename(x))[0], glob.glob(OMFITsrc + '/framework_guis/*.py')))
[docs]    def initialize(self):
        for item in self.required:
            self.load_framework_gui(item) 
[docs]    def load_framework_gui(self, item):
        dev_acceptable = os.access(OMFITsrc + os.sep + 'framework_guis' + os.sep + item + '.py', os.W_OK) and not os.path.exists(
            os.sep.join([OMFITsrc, '..', 'public'])
        )
        self[f'__{item}__'] = OMFITpythonGUI(
            OMFITsrc + os.sep + 'framework_guis' + os.sep + item + '.py',
            modifyOriginal=OMFIT['MainSettings']['SETUP']['developer_mode_by_default'] & dev_acceptable,
        ) 
    def __delitem__(self, key):
        if key in self.required:
            self.load_framework_gui(key)
        else:
            super().__delitem__(key) 
# --------------------------------
# Setup the global variable OMFIT
# --------------------------------
OMFIT.__class__ = OMFITmaintree  # root of the OMFIT tree
OMFIT.modifyOriginal = False
OMFIT.readOnly = False
OMFIT._OMFITkeyName = 'OMFIT'
OMFIT.prj_options = {}
OMFIT.prj_options_choices = {
    'notes': '',
    'commands': '',
    'commandNames': [],
    'namespace': '',
    'console': '',
    'shot': '',
    'time': '',
    'device': '',
    'modules': '',
    'type': ['Self-Destruct', 'Test', 'Working', 'Done', 'VIP'],
    'persistent_projectID': False,
    'color': OrderedDict(
        (
            ('red', 'red2'),
            ('orange', 'dark orange'),
            ('yellow', 'orange'),
            ('green', 'forest green'),
            ('blue', 'mediumblue'),
            ('purple', 'medium orchid'),
            ('gray', 'gray25'),
        )
    ),
}
atexit.register(OMFIT.quit)  # Register final automatic cleanup
# ------------------------------------
# location from root requires knowing what `OMFIT` is
# ------------------------------------
[docs]def locationFromRoot(location):
    """
    :param location: tree location
    :return: tree location from the closest root
    """
    tmp = location
    location = parseBuildLocation(location)
    for k in 1 + np.array(list(range(len(location) - 1))):
        loc = eval('OMFIT' + parseBuildLocation(location[:k]))
        if loc.__class__ is OMFITmodule:
            tmp = "root" + parseBuildLocation(location[k:])
    return tmp 
[docs]def rootLocation(location, *args, **kw):
    """
    :param location: tree location
    :return: tree location of the root
    """
    if len(args) >= 1:
        OMFIT = args[0]
    if len(args) >= 2:
        kw['rootName'] = args[1]
    rootName = kw.setdefault('rootName', 'OMFIT')
    location = parseBuildLocation(location)
    root = 'OMFIT'
    for k in 1 + np.array(list(range(len(location) - 1))):
        loc = eval('OMFIT' + parseBuildLocation(location[:k]))
        if loc.__class__ is OMFITmodule:
            root = 'OMFIT' + parseBuildLocation(location[:k])
    root = rootName + root[5:]
    return root 
# ---------------------
# Import OMFIT APIs
# and make it available to users Python scripts and expressions
# ---------------------
from omfit_classes import OMFITx
setattr(omfit_classes.omfit_python, 'OMFITx', OMFITx)
setattr(omfit_classes.omfit_base, 'OMFITx', OMFITx)
# ---------------------
# graphics
# ---------------------
from omfit_plot import *
# ---------------------
# update classes
# ---------------------
omfit_classes.omfit_base._updateTypes()
# ---------------------
# S3
# ---------------------
[docs]def OMFITobject_fromS3(filename, s3bucket):
    """
    Recovers OMFITobject from S3 and reloads it with the right class and original keyword parameters
    :return: object loaded from S3
    """
    s3connection = boto3.resource('s3', **boto_credentials())
    obj = s3connection.Object(s3bucket, filename)
    cls = eval(eval(obj.metadata.get('__class__', "'OMFITobject'")))
    orig_filename = eval(obj.metadata.get('__filename__', filename))
    tmp = cls(filename, s3bucket=s3bucket, **{k: eval(obj.metadata[k]) for k in obj.metadata if k not in ['__class__', '__filename__']})
    if os.path.split(tmp.filename)[1] != orig_filename:
        shutil.move(tmp.filename, os.path.split(tmp.filename)[0] + os.sep + os.path.split(orig_filename)[1])
        tmp.filename = os.path.split(tmp.filename)[0] + os.sep + os.path.split(orig_filename)[1]
        tmp.originalFilename = tmp.filename
        tmp.link = tmp.filename
    return tmp 
# ---------------------
# backwards compatibility
# ---------------------
isNone = is_none
[docs]class OMFITfileASCII(OMFITascii):
    """
    Use of this class is deprecated: use OMFITascii instead
    """ 
[docs]class OMFIT_Ufile(OMFITuFile):
    """
    Use of this class is deprecated: use OMFITuFile instead
    """ 
[docs]class OMFITdict(SortedDict):
    """
    Use of this class is deprecated: use SortedDict instead
    """ 
expr_value = evalExpr
[docs]class chebyfit(omfit_classes.utils_fit.fitCH):
    """
    Use of this class is deprecated: use fitCH instead
    """ 
[docs]class OMFITdependenceException(RuntimeError):
    """
    Use of this class is deprecated: use RuntimeError instead
    """ 
# ------------------
# this version time (follows last version time in startup.py)
# ------------------
_filename = OMFITsettingsDir + os.sep + 'OMFITlastRun.txt'
if float(thisVersionTime) > float(latestVersionTime):
    with open(_filename, 'w') as _f:
        _f.write(str(thisVersionTime))
# --------------------------------------
# load main settings and override / fix
# older versions of users main settings
# NOTE: after this point on OMFIT['MainSettings'] is available and can be accessed
# --------------------------------------
def _clearDeprecatedMainSettings(MainSettings):
    # keep track if there has been a change
    updated = False
    keep = {}
    # clear if last version is before given time or if `tmpDir` is found within one month after given time
    # (this is done to manage transition between versions of OMFIT that are already running)
    if latestVersionTime < 1427916944 or (
        'SETUP' in MainSettings and 'tmpDir' in MainSettings['SETUP'] and thisVersionTime < (1427916944 + 60 * 60 * 24 * 30)
    ):
        keep['SETUP'] = {}
        keep['SETUP']['email'] = ''
        keep['SETUP']['modulesDir'] = ''
        keep['SETUP']['editor'] = ''
        keep['SETUP']['browser'] = ''
        keep['SETUP']['error_report'] = ''
        for k in ['Extensions', 'KeyBindings', 'GUIappearance']:
            if k in MainSettings['SETUP']:
                keep['SETUP'][k] = copy.deepcopy(MainSettings['SETUP'][k])
        keep['EXPERIMENT'] = {}
        keep['EXPERIMENT']['device'] = ''
        keep['EXPERIMENT']['shot'] = ''
        keep['EXPERIMENT']['time'] = ''
        keep['SERVER'] = {}
        keep['SERVER']['ALCF_username'] = ''
        keep['SERVER']['GA_username'] = ''
        keep['SERVER']['PPPL_username'] = ''
        keep['SERVER']['NERSC_username'] = ''
        keep['SERVER']['UCSD_username'] = ''
        keep['SERVER']['ITM_username'] = ''
        keep['SERVER']['MIT_username'] = ''
        keep['SERVER']['default'] = ''
        keep['SERVER']['idl'] = ''
        keep['SERVER']['matlab'] = ''
        tmp = prune_mask(MainSettings, keep)
        MainSettings.clear()
        MainSettings.update(tmp)
        updated = True
    if 'SERVER' in MainSettings:
        # delete deprecated servers and localhost (localhost entry will be re-created)
        for server in (
            ['thea', 'harvest', 'venus', 'venus_tmp', 'venus_scratch', 'hopper', 'zeus', 'lohan', 'philos', 'edison']
            + list(['lohan%d' % x for x in range(20)])
            + ['localhost']
        ):
            if server in list(MainSettings['SERVER'].keys()):
                del MainSettings['SERVER'][server]
                if server != 'localhost':
                    updated = '1. deprecated_server:' + server
            # delete references to the deprecated servers
            for item in list(MainSettings['SERVER'].keys()):
                if (
                    'localhost' not in [item, server]
                    and isinstance(MainSettings['SERVER'][item], str)
                    and MainSettings['SERVER'][item] == server
                ):
                    del MainSettings['SERVER'][item]
                    updated = '2. deprecated_server: (%s,%s)' % (server, item)
        # delete servers if any of their entries is set to None
        for server in list(MainSettings['SERVER'].keys()):
            if isinstance(MainSettings['SERVER'][server], namelist.NamelistName):
                for item in list(MainSettings['SERVER'][server].keys()):
                    if MainSettings['SERVER'][server][item] == None:
                        del MainSettings['SERVER'][server]
                        updated = 'none_server'
                        break
        # freeze usernames
        username_keys = [item for item in MainSettings['SERVER'] if item.endswith('_username')]
        for username_key in username_keys:
            # if OMFIT is running at a recognized institution set username to match os.environ['USER'] if OMFITexpression or empty string
            if (
                (isinstance(MainSettings['SERVER'][username_key], OMFITexpression) or not MainSettings['SERVER'][username_key])
                and ('institution' in MainSettings['SETUP'])
                and username_key == (MainSettings['SETUP']['institution'] + '_username')
            ):
                MainSettings['SERVER'][username_key] = os.environ['USER']
                updated = 'freeze_username_institution'
            # for all other servers set username to empty string if username is an OMFITexpression
            elif isinstance(MainSettings['SERVER'][username_key], OMFITexpression):
                MainSettings['SERVER'][username_key] = ''
                updated = 'freeze_username_expression'
            # NFRI renamed themselves to KFE
            if (
                'KFE_username' in MainSettings['SERVER']
                and not MainSettings['SERVER']['KFE_username']
                and 'NFRI_username' in MainSettings['SERVER']
                and MainSettings['SERVER']['NFRI_username']
            ):
                MainSettings['SERVER']['KFE_username'] = MainSettings['SERVER']['NFRI_username']
    # prevent users to change workDir in bad ways
    if 'SETUP' in MainSettings and 'workDir' in MainSettings['SETUP'] and OMFITcwd not in str(MainSettings['SETUP']['workDir']):
        MainSettings['SETUP']['workDir'] = OMFITexpression("OMFITcwd+os.sep+'runs'+os.sep")
        updated = 'workdir'
    if (
        'SETUP' in MainSettings
        and 'GUIappearance' in MainSettings['SETUP']
        and 'persistent_projectID' in MainSettings['SETUP']['GUIappearance']
    ):
        del MainSettings['SETUP']['GUIappearance']['persistent_projectID']
        updated = 'persistent_project_id'
    if 'EXPERIMENT' in MainSettings and 'runid' in MainSettings['EXPERIMENT'] and MainSettings['EXPERIMENT']['runid'] == None:
        MainSettings['EXPERIMENT']['runid'] = 'sim1'
        updated = 'experiment'
    return updated
OMFIT.reset()
_updated = _clearDeprecatedMainSettings(OMFIT['MainSettings'])
if _updated:
    OMFIT.addMainSettings(updateUserSettings=True)
# set OMFIT repos directory where the users projects are stored
# rationale: home directory have quotas, scratch directories get automatically cleaned
omfit_classes.utils_base.OMFITreposDir = (
    os.path.split(os.path.abspath(evalExpr(OMFIT['MainSettings']['SETUP']['projectsDir'])))[0] + os.sep + '.repos'
)
# ------------------------------------
# Initialize OMFIT tree
# ------------------------------------
if not np.any([('sphinx' in k and not 'sphinxcontrib' in k) for k in sys.modules]):
    OMFIT.start()
    print('Time to load omfit_tree: %g seconds' % (time.time() - _t0))
    # ------------------------------------
    # Set new modules skeleton
    # ------------------------------------
    if os.path.exists(OMFITsessionDir + os.sep + 'NEW_MODULE_skeleton'):
        shutil.rmtree(OMFITsessionDir + os.sep + 'NEW_MODULE_skeleton', ignore_errors=True)
    shutil.copytree(
        OMFITsrc + os.sep + 'omfit_classes' + os.sep + 'skeleton' + os.sep + 'NEW_MODULE', OMFITsessionDir + os.sep + 'NEW_MODULE_skeleton'
    )
    OMFITaux['moduleSkeletonCache'] = OMFITtree(
        OMFITsessionDir + os.sep + 'NEW_MODULE_skeleton' + os.sep + 'OMFITsave.txt', quiet=True, modifyOriginal=True, readOnly=True
    )