Source code for smif.cli

# -*- coding: utf-8 -*-
"""A command line interface to the system of systems framework

This command line interface implements a number of methods.

- `setup` creates an example project with the recommended folder structure
- `run` performs a simulation of an individual sector model, or the whole system
        of systems model
- `validate` performs a validation check of the configuration file
- `app` runs the graphical user interface, opening in a web browser

Folder structure
----------------

When configuring a project for the CLI, the folder structure below should be
used.  In this example, there is one system-of-systems model, combining a water
supply and an energy demand model::

    /config
        project.yaml
        /sector_models
            energy_demand.yml
            water_supply.yml
        /sos_models
            energy_water.yml
        /model_runs
            run_to_2050.yml
            short_test_run.yml
            ...
    /data
        /initial_conditions
            reservoirs.yml
        /interval_definitions
            annual_intervals.csv
        /interventions
            water_supply.yml
        /narratives
            high_tech_dsm.yml
        /region_definitions
            /oxfordshire
                regions.geojson
            /uk_nations_shp
                regions.shp
        /scenarios
            population.csv
            raininess.csv
        /water_supply
            /initial_system
    /models
        energy_demand.py
        water_supply.py
    /planning
        expected_to_2020.yaml
        national_infrastructure_pipeline.yml

The sector model implementations can be installed independently of the model run
configuration. The paths to python wrapper classes (implementing SectorModel)
should be specified in each ``sector_model/*.yml`` configuration.

The project.yaml file specifies the metadata shared by all elements of the
project; ``sos_models`` specify the combinations of ``sector_models`` and
``scenarios`` while individual ``model_runs`` specify the scenario, strategy
and narrative combinations to be used in each run of the models.

"""
from __future__ import print_function

try:
    import _thread
except ImportError:
    import thread as _thread

import logging
import logging.config
import os
import pkg_resources

try:
    import win32api
    USE_WIN32 = True
except ImportError:
    USE_WIN32 = False

from argparse import ArgumentParser
import sys

import smif
import smif.cli.log

from smif.controller import copy_project_folder, execute_model_run, ModelRunScheduler
from smif.http_api import create_app
from smif.data_layer import Store
from smif.data_layer.file import (CSVDataStore, FileMetadataStore, ParquetDataStore,
                                  YamlConfigStore)


__author__ = "Will Usher, Tom Russell"
__copyright__ = "Will Usher, Tom Russell"
__license__ = "mit"


LOGGER = logging.getLogger(__name__)


[docs]def list_model_runs(args): """List the model runs defined in the config """ store = _get_store(args) model_run_configs = store.read_model_runs() for run in model_run_configs: print(run['name'])
[docs]def run_model_runs(args): """Run the model runs as requested. Check if results exist and asks user for permission to overwrite Parameters ---------- args """ LOGGER.profiling_start('run_model_runs', '{:s}, {:s}, {:s}'.format( args.modelrun, args.interface, args.directory)) if args.batchfile: with open(args.modelrun, 'r') as f: model_run_ids = f.read().splitlines() else: model_run_ids = [args.modelrun] store = _get_store(args) execute_model_run(model_run_ids, store, args.warm) LOGGER.profiling_stop('run_model_runs', '{:s}, {:s}, {:s}'.format( args.modelrun, args.interface, args.directory)) LOGGER.summary()
def _get_store(args): """Contruct store as configured by arguments """ if args.interface == 'local_csv': store = Store( config_store=YamlConfigStore(args.directory), metadata_store=FileMetadataStore(args.directory), data_store=CSVDataStore(args.directory), model_base_folder=args.directory ) elif args.interface == 'local_binary': store = Store( config_store=YamlConfigStore(args.directory), metadata_store=FileMetadataStore(args.directory), data_store=ParquetDataStore(args.directory), model_base_folder=args.directory ) else: raise ValueError("Store interface type {} not recognised.".format(args.interface)) return store def _run_server(args): app_folder = pkg_resources.resource_filename('smif', 'app/dist') app = create_app( static_folder=app_folder, template_folder=app_folder, data_interface=_get_store(args), scheduler=ModelRunScheduler() ) port = 5000 print(" Opening smif app\n") print(" Copy/paste this URL into your web browser to connect:") print(" http://localhost:" + str(port) + "\n") # add flush to ensure that text is printed before server thread starts print(" Close your browser then type Control-C here to quit.", flush=True) app.run(host='0.0.0.0', port=port, threaded=True)
[docs]def run_app(args): """Run the client/server application Parameters ---------- args """ # avoid one of two error messages from 'forrtl error(200)' when running # on windows cmd - seems related to scipy's underlying Fortran os.environ['FOR_DISABLE_CONSOLE_CTRL_HANDLER'] = 'T' if USE_WIN32: # Set handler for CTRL-C. Necessary to avoid `forrtl: error (200): # program aborting...` crash on CTRL-C if we're runnging from Windows # cmd.exe def handler(dw_ctrl_type, hook_sigint=_thread.interrupt_main): """Handler for CTRL-C interrupt """ if dw_ctrl_type == 0: # CTRL-C hook_sigint() return 1 # don't chain to the next handler return 0 # chain to the next handler win32api.SetConsoleCtrlHandler(handler, 1) # Create backend server process _run_server(args)
[docs]def setup_project_folder(args): """Setup a sample project """ copy_project_folder(args.directory)
[docs]def parse_arguments(): """Parse command line arguments Returns ======= :class:`argparse.ArgumentParser` """ parser = ArgumentParser(description='Command line tools for smif') parser.add_argument('-V', '--version', action='version', version="smif " + smif.__version__, help='show the current version of smif') parser.add_argument('-v', '--verbose', action='count', help='show messages: -v to see messages reporting on progress, ' + '-vv to see debug messages.') parent_parser = ArgumentParser(add_help=False) parent_parser.add_argument('-v', '--verbose', action='count', help='show messages: -v to see messages reporting on ' + 'progress, -vv to see debug messages.') parent_parser.add_argument('-i', '--interface', default='local_csv', choices=['local_csv', 'local_binary'], help="Select the data interface (default: %(default)s)") parent_parser.add_argument('-d', '--directory', default='.', help="Path to the project directory") subparsers = parser.add_subparsers(help='available commands') # SETUP parser_setup = subparsers.add_parser( 'setup', help='Setup the project folder', parents=[parent_parser]) parser_setup.set_defaults(func=setup_project_folder) # LIST parser_list = subparsers.add_parser( 'list', help='List available model runs', parents=[parent_parser]) parser_list.set_defaults(func=list_model_runs) # APP parser_app = subparsers.add_parser( 'app', help='Open smif app', parents=[parent_parser]) parser_app.set_defaults(func=run_app) # RUN parser_run = subparsers.add_parser( 'run', help='Run a model', parents=[parent_parser]) parser_run.set_defaults(func=run_model_runs) parser_run.add_argument('-w', '--warm', action='store_true', help="Use intermediate results from the last modelrun \ and continue from where it had left") parser_run.add_argument('-b', '--batchfile', action='store_true', help="Use a batchfile instead of a modelrun name (a \ list of modelrun names)") parser_run.add_argument('modelrun', help="Name of the model run to run") return parser
[docs]def confirm(prompt=None, response=False): """Prompts for a yes or no response from the user Arguments --------- prompt : str, default=None response : bool, default=False Returns ------- bool True for yes and False for no. Notes ----- `response` should be set to the default value assumed by the caller when user simply types ENTER. Examples -------- >>> confirm(prompt='Create Directory?', response=True) Create Directory? [y]|n: True >>> confirm(prompt='Create Directory?', response=False) Create Directory? [n]|y: False >>> confirm(prompt='Create Directory?', response=False) Create Directory? [n]|y: y True """ if prompt is None: prompt = 'Confirm' if response: prompt = '{} [{}]|{}: '.format(prompt, 'y', 'n') else: prompt = '{} [{}]|{}: '.format(prompt, 'n', 'y') while True: ans = input(prompt) if not ans: return response if ans not in ['y', 'Y', 'n', 'N']: print('please enter y or n.') continue if ans in ['y', 'Y']: return True if ans in ['n', 'N']: return False
[docs]def main(arguments=None): """Parse args and run """ parser = parse_arguments() args = parser.parse_args(arguments) def exception_handler(exception_type, exception, traceback, debug_hook=sys.excepthook): if args.verbose: debug_hook(exception_type, exception, traceback) else: print("{}: {}".format(exception_type.__name__, exception)) sys.excepthook = exception_handler if 'func' in args: args.func(args) else: parser.print_help()