From b87acddd575fb557fc107418565da715b8e5683f Mon Sep 17 00:00:00 2001 From: Emanuel Lima Date: Thu, 14 Dec 2023 17:30:00 -0300 Subject: [PATCH 1/6] Substitute CircleCI for GH Actions (#314) * Substitute CircleCI for GH Actions Signed-off-by: Emanuel Lima --- .circleci/config.yml | 67 --------------------------------- .github/workflows/cadcad-ci.yml | 61 ++++++++++++++++++++++++++++++ requirements.txt | 30 ++++++--------- 3 files changed, 72 insertions(+), 86 deletions(-) delete mode 100644 .circleci/config.yml create mode 100644 .github/workflows/cadcad-ci.yml diff --git a/.circleci/config.yml b/.circleci/config.yml deleted file mode 100644 index 8d16cb4f..00000000 --- a/.circleci/config.yml +++ /dev/null @@ -1,67 +0,0 @@ -version: 2.1 - -orbs: - python: circleci/python@0.3.2 -jobs: - test: - executor: python/default - steps: - - checkout - - python/load-cache - - python/install-deps - - python/save-cache - - run: pip install cadCAD==0.4.23 - - run: - command: python testing/tests/a_b_tests/multi_model_row_count_0_4_23.py - name: Multi Model Row Count (ver. 0.4.23) - - run: pip install -r requirements.txt --force-reinstall - - run: python setup.py bdist_wheel - - run: pip install dist/*.whl --force-reinstall - - run: - command: python testing/tests/multi_model_row_count.py - name: Multi Model Row Count - - run: - command: python testing/tests/param_sweep.py - name: Parameter sweep - - run: - command: python testing/tests/policy_aggregation.py - name: Policy Aggregation - - run: - command: python testing/tests/timestep1psub0.py - name: Timestep equals 1 instead of 0 for 1st PSUB - - run: - command: python testing/tests/runs_not_zero.py - name: Value Error thrown when Runs < 1 - - run: - command: python testing/tests/run1psub0.py - name: Run Starts at 1 for PSUB 0 - - run: - command: python testing/tests/append_mod_test.py - name: Auto Append Model ID - - run: - command: python testing/tests/cadCAD_exp.py - name: Package Root Experiment and configs object -# - run: -# command: python -m unittest discover -s testing/tests -p "*_test.py" -# name: Test Suite - jupyterServerTest: - docker: - - image: cimg/python:3.9.5 - steps: - - checkout - - python/load-cache - - run: python --version - - run: pip install --upgrade pip - - run: pip install jupyter - - python/save-cache - - run: python setup.py bdist_wheel - - run: pip install dist/*.whl --force-reinstall - - run: - command: python testing/tests/import_cadCAD_test.py - name: cadCAD importable by Jupyter Server - -workflows: - main: - jobs: - - test - - jupyterServerTest diff --git a/.github/workflows/cadcad-ci.yml b/.github/workflows/cadcad-ci.yml new file mode 100644 index 00000000..9ff30705 --- /dev/null +++ b/.github/workflows/cadcad-ci.yml @@ -0,0 +1,61 @@ +# This workflow will install Python dependencies and run tests with multiple versions of Python + +name: cadCAD CI + +on: + push: + branches: [ "master" ] + pull_request: + branches: [ "master" ] + +permissions: + contents: read + +jobs: + build: + continue-on-error: true + + strategy: + matrix: + python-version: ["3.9", "3.10", "3.11", "3.12"] + os: [ubuntu-latest, macos-latest] + + runs-on: ${{ matrix.os }} + + steps: + - uses: actions/checkout@v4 + + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v5 + with: + python-version: ${{ matrix.python-version }} + cache: 'pip' + + - name: Display Python version + run: python -c "import sys; print(sys.version)" + + - name: Install test and build dependencies + run: | + python -m pip install --upgrade pip + python -m pip install jupyter + pip install -r requirements.txt + + - name: Build cadCAD + run: | + python setup.py bdist_wheel + python -m pip install dist/*.whl --force-reinstall + + - name: Run tests + run: | + python testing/tests/multi_model_row_count.py + python testing/tests/param_sweep.py + python testing/tests/policy_aggregation.py + python testing/tests/timestep1psub0.py + python testing/tests/runs_not_zero.py + python testing/tests/run1psub0.py + python testing/tests/append_mod_test.py + python testing/tests/cadCAD_exp.py + + - name: Run Jupyter test + run: | + python testing/tests/import_cadCAD_test.py diff --git a/requirements.txt b/requirements.txt index 0890a625..a951734a 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,21 +1,13 @@ -i https://pypi.org/simple -matplotlib==3.3.2 -networkx==2.5 -parameterized==0.7.4 -plotly==4.10.0 -pytest==6.0.2 -scikit-learn==0.23.2 -scipy>=1.5.2 -seaborn==0.11.0 -tabulate==0.8.7 -xarray==0.16.0 -wheel==0.38.1 -pandas==1.1.5 -fn==0.4.3 -funcy==1.16 -dill==0.3.4 -pathos==0.2.8 -numpy==1.22.0 -pytz==2021.1 -six>=1.11.0 +parameterized>=0.7.4 +pytest>=6.0.2 +tabulate>=0.8.7 +wheel>=0.38.1 +pandas>=1.1.5 +funcy>=1.16 +dill>=0.3.4 +pathos>=0.2.8 +numpy>=1.22.0 +pytz>=2021.1 +setuptools>=69.0.2 From b9d291873496f5a79a34910dc9e90a48049d2666 Mon Sep 17 00:00:00 2001 From: Danilo Lessa Bernardineli Date: Thu, 14 Dec 2023 17:35:57 -0300 Subject: [PATCH 2/6] fix tests + rm simulations/ folder (#312) --- simulations/__init__.py | 0 simulations/external_data/output.csv | 27 --- simulations/regression_tests/__init__.py | 1 - .../regression_tests/execs/__init__.py | 0 .../regression_tests/execs/config1_test.py | 16 -- .../regression_tests/execs/config2_test.py | 16 -- .../execs/multi_config_test.py | 17 -- .../execs/multi_config_test2.py | 17 -- .../execs/param_sweep_test.py | 16 -- .../regression_tests/execs/policy_agg_test.py | 17 -- simulations/regression_tests/experiments.py | 3 - .../regression_tests/models/__init__.py | 0 .../regression_tests/models/config1.py | 158 ---------------- .../regression_tests/models/config2.py | 148 --------------- .../regression_tests/models/config_multi_1.py | 172 ------------------ .../regression_tests/models/config_multi_2.py | 158 ---------------- .../regression_tests/models/param_sweep.py | 94 ---------- .../models/policy_aggregation.py | 83 --------- .../regression_tests/models/sweep_config.py | 101 ---------- testing/generic_test.py | 3 +- testing/tests/cadCAD_memory_address.json | 1 + testing/tests/dev/compare_results_dev.py | 40 ---- testing/tests/dev/compare_results_old.py | 79 -------- testing/tests/dev/out_check_dev.py | 28 --- testing/tests/dev/utils_dev.py | 5 - testing/tests/import_cadCAD.ipynb | 6 +- 26 files changed, 5 insertions(+), 1201 deletions(-) delete mode 100644 simulations/__init__.py delete mode 100644 simulations/external_data/output.csv delete mode 100644 simulations/regression_tests/__init__.py delete mode 100644 simulations/regression_tests/execs/__init__.py delete mode 100644 simulations/regression_tests/execs/config1_test.py delete mode 100644 simulations/regression_tests/execs/config2_test.py delete mode 100644 simulations/regression_tests/execs/multi_config_test.py delete mode 100644 simulations/regression_tests/execs/multi_config_test2.py delete mode 100644 simulations/regression_tests/execs/param_sweep_test.py delete mode 100644 simulations/regression_tests/execs/policy_agg_test.py delete mode 100644 simulations/regression_tests/experiments.py delete mode 100644 simulations/regression_tests/models/__init__.py delete mode 100644 simulations/regression_tests/models/config1.py delete mode 100644 simulations/regression_tests/models/config2.py delete mode 100644 simulations/regression_tests/models/config_multi_1.py delete mode 100644 simulations/regression_tests/models/config_multi_2.py delete mode 100644 simulations/regression_tests/models/param_sweep.py delete mode 100644 simulations/regression_tests/models/policy_aggregation.py delete mode 100644 simulations/regression_tests/models/sweep_config.py create mode 100644 testing/tests/cadCAD_memory_address.json delete mode 100644 testing/tests/dev/compare_results_dev.py delete mode 100644 testing/tests/dev/compare_results_old.py delete mode 100644 testing/tests/dev/out_check_dev.py delete mode 100644 testing/tests/dev/utils_dev.py diff --git a/simulations/__init__.py b/simulations/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/simulations/external_data/output.csv b/simulations/external_data/output.csv deleted file mode 100644 index c28448f2..00000000 --- a/simulations/external_data/output.csv +++ /dev/null @@ -1,27 +0,0 @@ -ds1,ds2,ds3,run,substep,timestep -0,0,1,1,0,0 -1,40,5,1,1,1 -2,40,5,1,2,1 -3,40,5,1,3,1 -4,40,5,1,1,2 -5,40,5,1,2,2 -6,40,5,1,3,2 -7,40,5,1,1,3 -8,40,5,1,2,3 -9,40,5,1,3,3 -10,40,5,1,1,4 -11,40,5,1,2,4 -12,40,5,1,3,4 -0,0,1,2,0,0 -1,40,5,2,1,1 -2,40,5,2,2,1 -3,40,5,2,3,1 -4,40,5,2,1,2 -5,40,5,2,2,2 -6,40,5,2,3,2 -7,40,5,2,1,3 -8,40,5,2,2,3 -9,40,5,2,3,3 -10,40,5,2,1,4 -11,40,5,2,2,4 -12,40,5,2,3,4 diff --git a/simulations/regression_tests/__init__.py b/simulations/regression_tests/__init__.py deleted file mode 100644 index 666406b4..00000000 --- a/simulations/regression_tests/__init__.py +++ /dev/null @@ -1 +0,0 @@ -from simulations.regression_tests import experiments \ No newline at end of file diff --git a/simulations/regression_tests/execs/__init__.py b/simulations/regression_tests/execs/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/simulations/regression_tests/execs/config1_test.py b/simulations/regression_tests/execs/config1_test.py deleted file mode 100644 index 1e9d5c8c..00000000 --- a/simulations/regression_tests/execs/config1_test.py +++ /dev/null @@ -1,16 +0,0 @@ -from pprint import pprint -import pandas as pd -from tabulate import tabulate - -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from simulations.regression_tests.models import config1 - -exec_mode = ExecutionMode() - -local_proc_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=local_proc_ctx, configs=config1.exp.configs) - -raw_result, tensor_fields, _ = run.execute() -result = pd.DataFrame(raw_result) -print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -print(tabulate(result, headers='keys', tablefmt='psql')) \ No newline at end of file diff --git a/simulations/regression_tests/execs/config2_test.py b/simulations/regression_tests/execs/config2_test.py deleted file mode 100644 index e745c759..00000000 --- a/simulations/regression_tests/execs/config2_test.py +++ /dev/null @@ -1,16 +0,0 @@ -from pprint import pprint -import pandas as pd -from tabulate import tabulate - -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from simulations.regression_tests.models import config2 - -exec_mode = ExecutionMode() - -local_proc_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=local_proc_ctx, configs=config2.exp.configs) - -raw_result, tensor_fields, _ = run.execute() -result = pd.DataFrame(raw_result) -print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -print(tabulate(result, headers='keys', tablefmt='psql')) \ No newline at end of file diff --git a/simulations/regression_tests/execs/multi_config_test.py b/simulations/regression_tests/execs/multi_config_test.py deleted file mode 100644 index d2329404..00000000 --- a/simulations/regression_tests/execs/multi_config_test.py +++ /dev/null @@ -1,17 +0,0 @@ -from tabulate import tabulate -from pprint import pprint -import pandas as pd - -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from simulations.regression_tests.experiments import multi_exp -from simulations.regression_tests.models import config_multi_1, config_multi_2 - -exec_mode = ExecutionMode() - -local_proc_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=local_proc_ctx, configs=multi_exp.configs) - -raw_result, tensor_fields, _ = run.execute() -result = pd.DataFrame(raw_result) -print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -print(tabulate(result, headers='keys', tablefmt='psql')) \ No newline at end of file diff --git a/simulations/regression_tests/execs/multi_config_test2.py b/simulations/regression_tests/execs/multi_config_test2.py deleted file mode 100644 index bdf785cd..00000000 --- a/simulations/regression_tests/execs/multi_config_test2.py +++ /dev/null @@ -1,17 +0,0 @@ -from tabulate import tabulate -from pprint import pprint -import pandas as pd - -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from simulations.regression_tests.experiments import multi_exp -from simulations.regression_tests.models import config_multi_1, sweep_config - -exec_mode = ExecutionMode() - -local_proc_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=local_proc_ctx, configs=multi_exp.configs) - -raw_result, tensor_fields, _ = run.execute() -result = pd.DataFrame(raw_result) -print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -print(tabulate(result, headers='keys', tablefmt='psql')) \ No newline at end of file diff --git a/simulations/regression_tests/execs/param_sweep_test.py b/simulations/regression_tests/execs/param_sweep_test.py deleted file mode 100644 index 6bb5eb2c..00000000 --- a/simulations/regression_tests/execs/param_sweep_test.py +++ /dev/null @@ -1,16 +0,0 @@ -from pprint import pprint - -import pandas as pd -from tabulate import tabulate -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from simulations.regression_tests.models import sweep_config - -exec_mode = ExecutionMode() - -local_proc_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=local_proc_ctx, configs=sweep_config.exp.configs) - -raw_result, tensor_fields, _ = run.execute() -result = pd.DataFrame(raw_result) -print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -print(tabulate(result, headers='keys', tablefmt='psql')) diff --git a/simulations/regression_tests/execs/policy_agg_test.py b/simulations/regression_tests/execs/policy_agg_test.py deleted file mode 100644 index 6a456db7..00000000 --- a/simulations/regression_tests/execs/policy_agg_test.py +++ /dev/null @@ -1,17 +0,0 @@ -from pprint import pprint - -import pandas as pd -from tabulate import tabulate - -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from simulations.regression_tests.models import policy_aggregation - -exec_mode = ExecutionMode() - -local_proc_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=local_proc_ctx, configs=policy_aggregation.exp.configs) - -raw_result, tensor_fields, _ = run.execute() -result = pd.DataFrame(raw_result) -print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -print(tabulate(result, headers='keys', tablefmt='psql')) diff --git a/simulations/regression_tests/experiments.py b/simulations/regression_tests/experiments.py deleted file mode 100644 index 87e7a234..00000000 --- a/simulations/regression_tests/experiments.py +++ /dev/null @@ -1,3 +0,0 @@ -from cadCAD.configuration import Experiment - -multi_exp = Experiment() \ No newline at end of file diff --git a/simulations/regression_tests/models/__init__.py b/simulations/regression_tests/models/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/simulations/regression_tests/models/config1.py b/simulations/regression_tests/models/config1.py deleted file mode 100644 index f27fddaf..00000000 --- a/simulations/regression_tests/models/config1.py +++ /dev/null @@ -1,158 +0,0 @@ -import numpy as np -from datetime import timedelta - -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import bound_norm_random, config_sim, time_step, env_trigger - -seeds = { - 'z': np.random.RandomState(1), - 'a': np.random.RandomState(2), - 'b': np.random.RandomState(3), - 'c': np.random.RandomState(4) -} - - -# Policies per Mechanism -def p1m1(_g, step, sL, s, **kwargs): - return {'param1': 1} -def p2m1(_g, step, sL, s, **kwargs): - return {'param1': 1, 'param2': 4} - -def p1m2(_g, step, sL, s, **kwargs): - return {'param1': 'a', 'param2': 2} -def p2m2(_g, step, sL, s, **kwargs): - return {'param1': 'b', 'param2': 4} - -def p1m3(_g, step, sL, s, **kwargs): - return {'param1': ['c'], 'param2': np.array([10, 100])} -def p2m3(_g, step, sL, s, **kwargs): - return {'param1': ['d'], 'param2': np.array([20, 200])} - - -# Internal States per Mechanism -def s1m1(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = s['s1'] + 1 - return (y, x) -def s2m1(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m2(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = s['s1'] + 1 - return (y, x) -def s2m2(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m3(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = s['s1'] + 1 - return (y, x) -def s2m3(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def policies(_g, step, sL, s, _input, **kwargs): - y = 'policies' - x = _input - return (y, x) - - -# Exogenous States -proc_one_coef_A = 0.7 -proc_one_coef_B = 1.3 - -def es3(_g, step, sL, s, _input, **kwargs): - y = 's3' - x = s['s3'] * bound_norm_random(seeds['a'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def es4(_g, step, sL, s, _input, **kwargs): - y = 's4' - x = s['s4'] * bound_norm_random(seeds['b'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def update_timestamp(_g, step, sL, s, _input, **kwargs): - y = 'timestamp' - return y, time_step(dt_str=s[y], dt_format='%Y-%m-%d %H:%M:%S', _timedelta=timedelta(days=0, minutes=0, seconds=1)) - - -# Genesis States -genesis_states = { - 's1': 0.0, - 's2': 0.0, - 's3': 1.0, - 's4': 1.0, - 'timestamp': '2018-10-01 15:16:24' -} - - -# Environment Process -trigger_timestamps = ['2018-10-01 15:16:25', '2018-10-01 15:16:27', '2018-10-01 15:16:29'] -env_processes = { - "s3": [lambda _g, x: 5], - "s4": env_trigger(3)(trigger_field='timestamp', trigger_vals=trigger_timestamps, funct_list=[lambda _g, x: 10]) -} - - -partial_state_update_block = [ - { - "policies": { - "b1": p1m1, - "b2": p2m1 - }, - "variables": { - "s1": s1m1, - "s2": s2m1, - "s3": es3, - "s4": es4, - "timestamp": update_timestamp - } - }, - { - "policies": { - "b1": p1m2, - "b2": p2m2 - }, - "variables": { - "s1": s1m2, - "s2": s2m2, - # "s3": es3p1, - # "s4": es4p2, - } - }, - { - "policies": { - "b1": p1m3, - "b2": p2m3 - }, - "variables": { - "s1": s1m3, - "s2": s2m3, - # "s3": es3p1, - # "s4": es4p2, - } - } -] - -sim_config_dict = { - "N": 1, - "T": range(5) - } - -sim_config = config_sim(sim_config_dict) - -exp = Experiment() -exp.append_model( - model_id='sys_model_1', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_processes, - partial_state_update_blocks=partial_state_update_block, - policy_ops=[lambda a, b: a + b] -) diff --git a/simulations/regression_tests/models/config2.py b/simulations/regression_tests/models/config2.py deleted file mode 100644 index 88a8aa4e..00000000 --- a/simulations/regression_tests/models/config2.py +++ /dev/null @@ -1,148 +0,0 @@ -import numpy as np -from datetime import timedelta - -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import bound_norm_random, config_sim, env_trigger, time_step - -seeds = { - 'z': np.random.RandomState(1), - 'a': np.random.RandomState(2), - 'b': np.random.RandomState(3), - 'c': np.random.RandomState(3) -} - - -# Policies per Mechanism -def p1m1(_g, step, sL, s, **kwargs): - return {'param1': 1} -def p2m1(_g, step, sL, s, **kwargs): - return {'param2': 4} - -def p1m2(_g, step, sL, s, **kwargs): - return {'param1': 'a', 'param2': 2} -def p2m2(_g, step, sL, s, **kwargs): - return {'param1': 'b', 'param2': 4} - -def p1m3(_g, step, sL, s, **kwargs): - return {'param1': ['c'], 'param2': np.array([10, 100])} -def p2m3(_g, step, sL, s, **kwargs): - return {'param1': ['d'], 'param2': np.array([20, 200])} - - -# Internal States per Mechanism -def s1m1(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = _input['param1'] - return (y, x) -def s2m1(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m2(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = _input['param1'] - return (y, x) -def s2m2(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m3(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = _input['param1'] - return (y, x) -def s2m3(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - - -# Exogenous States -proc_one_coef_A = 0.7 -proc_one_coef_B = 1.3 - -def es3(_g, step, sL, s, _input, **kwargs): - y = 's3' - x = s['s3'] * bound_norm_random(seeds['a'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def es4(_g, step, sL, s, _input, **kwargs): - y = 's4' - x = s['s4'] * bound_norm_random(seeds['b'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def update_timestamp(_g, step, sL, s, _input, **kwargs): - y = 'timestamp' - return y, time_step(dt_str=s[y], dt_format='%Y-%m-%d %H:%M:%S', _timedelta=timedelta(days=0, minutes=0, seconds=1)) - - -# Genesis States -genesis_states = { - 's1': 0, - 's2': 0, - 's3': 1, - 's4': 1, - 'timestamp': '2018-10-01 15:16:24' -} - - -# Environment Process -trigger_timestamps = ['2018-10-01 15:16:25', '2018-10-01 15:16:27', '2018-10-01 15:16:29'] -env_processes = { - "s3": [lambda _g, x: 5], - "s4": env_trigger(3)(trigger_field='timestamp', trigger_vals=trigger_timestamps, funct_list=[lambda _g, x: 10]) -} - -partial_state_update_block = { - "m1": { - "policies": { - "b1": p1m1, - # "b2": p2m1 - }, - "states": { - "s1": s1m1, - # "s2": s2m1 - "s3": es3, - "s4": es4, - "timestep": update_timestamp - } - }, - "m2": { - "policies": { - "b1": p1m2, - # "b2": p2m2 - }, - "states": { - "s1": s1m2, - # "s2": s2m2 - } - }, - "m3": { - "policies": { - "b1": p1m3, - "b2": p2m3 - }, - "states": { - "s1": s1m3, - "s2": s2m3 - } - } -} - -sim_config_dict = { - "N": 3, - "T": range(5), -} - - -sim_config = config_sim(sim_config_dict) - -exp = Experiment() -exp.append_model( - model_id='sys_model_1', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_processes, - partial_state_update_blocks=partial_state_update_block -) diff --git a/simulations/regression_tests/models/config_multi_1.py b/simulations/regression_tests/models/config_multi_1.py deleted file mode 100644 index 03274368..00000000 --- a/simulations/regression_tests/models/config_multi_1.py +++ /dev/null @@ -1,172 +0,0 @@ -from copy import deepcopy -from pprint import pprint - -import numpy as np -from datetime import timedelta - -# from cadCAD import configs -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import bound_norm_random, config_sim, time_step, env_trigger -from simulations.regression_tests.experiments import multi_exp - -seeds = { - 'z': np.random.RandomState(1), - 'a': np.random.RandomState(2), - 'b': np.random.RandomState(3), - 'c': np.random.RandomState(4) -} - - -# Policies per Mechanism -def p1m1(_g, step, sL, s, **kwargs): - return {'param1': 1} -def p2m1(_g, step, sL, s, **kwargs): - return {'param1': 1, 'param2': 4} - -def p1m2(_g, step, sL, s, **kwargs): - return {'param1': 'a', 'param2': 2} -def p2m2(_g, step, sL, s, **kwargs): - return {'param1': 'b', 'param2': 4} - -def p1m3(_g, step, sL, s, **kwargs): - return {'param1': ['c'], 'param2': np.array([10, 100])} -def p2m3(_g, step, sL, s, **kwargs): - return {'param1': ['d'], 'param2': np.array([20, 200])} - - -# Internal States per Mechanism -def s1m1(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = s['s1'] + 1 - return (y, x) -def s2m1(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m2(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = s['s1'] + 1 - return (y, x) -def s2m2(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m3(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = s['s1'] + 1 - return (y, x) -def s2m3(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def policies(_g, step, sL, s, _input, **kwargs): - y = 'policies' - x = _input - return (y, x) - - -# Exogenous States -proc_one_coef_A = 0.7 -proc_one_coef_B = 1.3 - -def es3(_g, step, sL, s, _input, **kwargs): - y = 's3' - x = s['s3'] * bound_norm_random(seeds['a'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def es4(_g, step, sL, s, _input, **kwargs): - y = 's4' - x = s['s4'] * bound_norm_random(seeds['b'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def update_timestamp(_g, step, sL, s, _input, **kwargs): - y = 'timestamp' - return y, time_step(dt_str=s[y], dt_format='%Y-%m-%d %H:%M:%S', _timedelta=timedelta(days=0, minutes=0, seconds=1)) - - -# Genesis States -genesis_states = { - 's1': 0.0, - 's2': 0.0, - 's3': 1.0, - 's4': 1.0, - 'timestamp': '2018-10-01 15:16:24' -} - - -# Environment Process -trigger_timestamps = ['2018-10-01 15:16:25', '2018-10-01 15:16:27', '2018-10-01 15:16:29'] -env_processes = { - "s3": [lambda _g, x: 5], - "s4": env_trigger(3)(trigger_field='timestamp', trigger_vals=trigger_timestamps, funct_list=[lambda _g, x: 10]) -} - - -partial_state_update_block = [ - { - "policies": { - "b1": p1m1, - "b2": p2m1 - }, - "variables": { - "s1": s1m1, - "s2": s2m1, - "s3": es3, - "s4": es4, - "timestamp": update_timestamp - } - }, - { - "policies": { - "b1": p1m2, - "b2": p2m2 - }, - "variables": { - "s1": s1m2, - "s2": s2m2, - # "s3": es3p1, - # "s4": es4p2, - } - }, - { - "policies": { - "b1": p1m3, - "b2": p2m3 - }, - "variables": { - "s1": s1m3, - "s2": s2m3, - # "s3": es3p1, - # "s4": es4p2, - } - } -] - -sim_config_dict = { - "N": 2, - "T": range(5) - } - -sim_config = config_sim(sim_config_dict) - -exp = Experiment() -exp.append_model( - user_id='user_a', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_processes, - partial_state_update_blocks=partial_state_update_block, - policy_ops=[lambda a, b: a + b] -) - -multi_exp.append_model( - model_id='sys_model_1', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_processes, - partial_state_update_blocks=partial_state_update_block, - policy_ops=[lambda a, b: a + b] -) diff --git a/simulations/regression_tests/models/config_multi_2.py b/simulations/regression_tests/models/config_multi_2.py deleted file mode 100644 index e7f15eed..00000000 --- a/simulations/regression_tests/models/config_multi_2.py +++ /dev/null @@ -1,158 +0,0 @@ -import numpy as np -from datetime import timedelta - -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import bound_norm_random, config_sim, env_trigger, time_step -from simulations.regression_tests.experiments import multi_exp - -seeds = { - 'z': np.random.RandomState(1), - 'a': np.random.RandomState(2), - 'b': np.random.RandomState(3), - 'c': np.random.RandomState(3) -} - - -# Policies per Mechanism -def p1m1(_g, step, sL, s, **kwargs): - return {'param1': 1} -def p2m1(_g, step, sL, s, **kwargs): - return {'param2': 4} - -def p1m2(_g, step, sL, s, **kwargs): - return {'param1': 'a', 'param2': 2} -def p2m2(_g, step, sL, s, **kwargs): - return {'param1': 'b', 'param2': 4} - -def p1m3(_g, step, sL, s, **kwargs): - return {'param1': ['c'], 'param2': np.array([10, 100])} -def p2m3(_g, step, sL, s, **kwargs): - return {'param1': ['d'], 'param2': np.array([20, 200])} - - -# Internal States per Mechanism -def s1m1(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = _input['param1'] - return (y, x) -def s2m1(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m2(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = _input['param1'] - return (y, x) -def s2m2(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - -def s1m3(_g, step, sL, s, _input, **kwargs): - y = 's1' - x = _input['param1'] - return (y, x) -def s2m3(_g, step, sL, s, _input, **kwargs): - y = 's2' - x = _input['param2'] - return (y, x) - - -# Exogenous States -proc_one_coef_A = 0.7 -proc_one_coef_B = 1.3 - -def es3(_g, step, sL, s, _input, **kwargs): - y = 's3' - x = s['s3'] * bound_norm_random(seeds['a'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def es4(_g, step, sL, s, _input, **kwargs): - y = 's4' - x = s['s4'] * bound_norm_random(seeds['b'], proc_one_coef_A, proc_one_coef_B) - return (y, x) - -def update_timestamp(_g, step, sL, s, _input, **kwargs): - y = 'timestamp' - return y, time_step(dt_str=s[y], dt_format='%Y-%m-%d %H:%M:%S', _timedelta=timedelta(days=0, minutes=0, seconds=1)) - - -# Genesis States -genesis_states = { - 's1': 0, - 's2': 0, - 's3': 1, - 's4': 1, - 'timestamp': '2018-10-01 15:16:24' -} - - -# Environment Process -trigger_timestamps = ['2018-10-01 15:16:25', '2018-10-01 15:16:27', '2018-10-01 15:16:29'] -env_processes = { - "s3": [lambda _g, x: 5], - "s4": env_trigger(3)(trigger_field='timestamp', trigger_vals=trigger_timestamps, funct_list=[lambda _g, x: 10]) -} - -partial_state_update_block = { - "m1": { - "policies": { - "b1": p1m1, - # "b2": p2m1 - }, - "states": { - "s1": s1m1, - # "s2": s2m1 - "s3": es3, - "s4": es4, - "timestep": update_timestamp - } - }, - "m2": { - "policies": { - "b1": p1m2, - # "b2": p2m2 - }, - "states": { - "s1": s1m2, - # "s2": s2m2 - } - }, - "m3": { - "policies": { - "b1": p1m3, - "b2": p2m3 - }, - "states": { - "s1": s1m3, - "s2": s2m3 - } - } -} - -sim_config_dict = { - "N": 3, - "T": range(5), -} - - -sim_config = config_sim(sim_config_dict) - -exp = Experiment() -exp.append_model( - # config_list=configs, - user_id='user_b', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_processes, - partial_state_update_blocks=partial_state_update_block -) - -multi_exp.append_model( - model_id='sys_model_2', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_processes, - partial_state_update_blocks=partial_state_update_block -) diff --git a/simulations/regression_tests/models/param_sweep.py b/simulations/regression_tests/models/param_sweep.py deleted file mode 100644 index 65e3977d..00000000 --- a/simulations/regression_tests/models/param_sweep.py +++ /dev/null @@ -1,94 +0,0 @@ -import pprint -from typing import Dict, List, Any - -# from cadCAD.configuration import append_configs -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import env_trigger, var_substep_trigger, config_sim, psub_list - -pp = pprint.PrettyPrinter(indent=4) - -def some_function(x): - return x - -# Optional -# dict must contain lists opf 2 distinct lengths -g: Dict[str, List[Any]] = { - 'alpha': [1], - 'beta': [2, some_function], - 'gamma': [3, 4], - 'omega': [7] -} - -psu_steps = ['m1', 'm2', 'm3'] -system_substeps = len(psu_steps) -var_timestep_trigger = var_substep_trigger([0, system_substeps]) -env_timestep_trigger = env_trigger(system_substeps) -env_process = {} - - -# ['s1', 's2', 's3', 's4'] -# Policies per Mechanism -def gamma(_g, step, sL, s, **kwargs): - return {'gamma': _g['gamma']} - - -def omega(_g, step, sL, s, **kwargs): - return {'omega': _g['omega']} - - -# Internal States per Mechanism -def alpha(_g, step, sL, s, _input, **kwargs): - return 'alpha', _g['alpha'] - - -def beta(_g, step, sL, s, _input, **kwargs): - return 'beta', _g['beta'] - - -def policies(_g, step, sL, s, _input, **kwargs): - return 'policies', _input - - -def sweeped(_g, step, sL, s, _input, **kwargs): - return 'sweeped', {'beta': _g['beta'], 'gamma': _g['gamma']} - -psu_block = {k: {"policies": {}, "variables": {}} for k in psu_steps} -for m in psu_steps: - psu_block[m]['policies']['gamma'] = gamma - psu_block[m]['policies']['omega'] = omega - psu_block[m]["variables"]['alpha'] = alpha - psu_block[m]["variables"]['beta'] = beta - psu_block[m]['variables']['policies'] = policies - psu_block[m]["variables"]['sweeped'] = var_timestep_trigger(y='sweeped', f=sweeped) - - -# Genesis States -genesis_states = { - 'alpha': 0, - 'beta': 0, - 'policies': {}, - 'sweeped': {} -} - -# Environment Process -env_process['sweeped'] = env_timestep_trigger(trigger_field='timestep', trigger_vals=[5], funct_list=[lambda _g, x: _g['beta']]) - - -sim_config = config_sim( - { - "N": 2, - "T": range(2), - "M": g, # Optional - } -) - -partial_state_update_blocks = psub_list(psu_block, psu_steps) - -exp = Experiment() -exp.append_model( - model_id='sys_model_1', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_process, - partial_state_update_blocks=partial_state_update_blocks -) diff --git a/simulations/regression_tests/models/policy_aggregation.py b/simulations/regression_tests/models/policy_aggregation.py deleted file mode 100644 index 7621f7da..00000000 --- a/simulations/regression_tests/models/policy_aggregation.py +++ /dev/null @@ -1,83 +0,0 @@ -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import config_sim - -# Policies per Mechanism -def p1m1(_g, step, sL, s, **kwargs): - return {'policy1': 1} -def p2m1(_g, step, sL, s, **kwargs): - return {'policy2': 2} - -def p1m2(_g, step, sL, s, **kwargs): - return {'policy1': 2, 'policy2': 2} -def p2m2(_g, step, sL, s, **kwargs): - return {'policy1': 2, 'policy2': 2} - -def p1m3(_g, step, sL, s, **kwargs): - return {'policy1': 1, 'policy2': 2, 'policy3': 3} -def p2m3(_g, step, sL, s, **kwargs): - return {'policy1': 1, 'policy2': 2, 'policy3': 3} - - -# Internal States per Mechanism -def add(y, x): - return lambda _g, step, sH, s, _input, **kwargs: (y, s[y] + x) - -def policies(_g, step, sH, s, _input, **kwargs): - y = 'policies' - x = _input - return (y, x) - - -# Genesis States -genesis_states = { - 'policies': {}, - 's1': 0 -} - -variables = { - 's1': add('s1', 1), - "policies": policies -} - -partial_state_update_block = { - "m1": { - "policies": { - "p1": p1m1, - "p2": p2m1 - }, - "variables": variables - }, - "m2": { - "policies": { - "p1": p1m2, - "p2": p2m2 - }, - "variables": variables - }, - "m3": { - "policies": { - "p1": p1m3, - "p2": p2m3 - }, - "variables": variables - } -} - - -sim_config = config_sim( - { - "N": 2, - "T": range(3), - } -) - -# Aggregation == Reduce Map / Reduce Map Aggregation -# using env functions (include in reg test using / for env proc) -exp = Experiment() -exp.append_model( - model_id='sys_model_1', - sim_configs=sim_config, - initial_state=genesis_states, - partial_state_update_blocks=partial_state_update_block, - policy_ops=[lambda a, b: a + b, lambda y: y * 2] # Default: lambda a, b: a + b -) diff --git a/simulations/regression_tests/models/sweep_config.py b/simulations/regression_tests/models/sweep_config.py deleted file mode 100644 index 00d7911d..00000000 --- a/simulations/regression_tests/models/sweep_config.py +++ /dev/null @@ -1,101 +0,0 @@ -from pprint import pprint -from typing import Dict, List, Any - -# from cadCAD.configuration import append_configs -from cadCAD.configuration import Experiment -from cadCAD.configuration.utils import env_trigger, var_substep_trigger, config_sim, psub_list -from simulations.regression_tests.experiments import multi_exp - - -def some_function(x): - return x - -# Optional -# dict must contain lists opf 2 distinct lengths -g: Dict[str, List[Any]] = { - 'alpha': [1], - 'beta': [2, some_function], - 'gamma': [3, 4], - 'omega': [7] -} - -psu_steps = ['m1', 'm2', 'm3'] -system_substeps = len(psu_steps) -var_timestep_trigger = var_substep_trigger([0, system_substeps]) -env_timestep_trigger = env_trigger(system_substeps) -env_process = {} - - -# ['s1', 's2', 's3', 's4'] -# Policies per Mechanism -def gamma(_g, step, sL, s, **kwargs): - return {'gamma': _g['gamma']} - - -def omega(_g, step, sL, s, **kwargs): - return {'omega': _g['omega']} - - -# Internal States per Mechanism -def alpha(_g, step, sL, s, _input, **kwargs): - return 'alpha', _g['alpha'] - - -def beta(_g, step, sL, s, _input, **kwargs): - return 'beta', _g['beta'] - - -def policies(_g, step, sL, s, _input, **kwargs): - return 'policies', _input - - -def sweeped(_g, step, sL, s, _input, **kwargs): - return 'sweeped', {'beta': _g['beta'], 'gamma': _g['gamma']} - -psu_block = {k: {"policies": {}, "variables": {}} for k in psu_steps} -for m in psu_steps: - psu_block[m]['policies']['gamma'] = gamma - psu_block[m]['policies']['omega'] = omega - psu_block[m]["variables"]['alpha'] = alpha - psu_block[m]["variables"]['beta'] = beta - psu_block[m]['variables']['policies'] = policies - psu_block[m]["variables"]['sweeped'] = var_timestep_trigger(y='sweeped', f=sweeped) - - -# Genesis States -genesis_states = { - 'alpha': 0, - 'beta': 0, - 'policies': {}, - 'sweeped': {} -} - -# Environment Process -env_process['sweeped'] = env_timestep_trigger(trigger_field='timestep', trigger_vals=[5], funct_list=[lambda _g, x: _g['beta']]) - - -sim_config = config_sim( - { - "N": 2, - "T": range(2), - "M": g, # Optional - } -) - -partial_state_update_blocks = psub_list(psu_block, psu_steps) - -exp = Experiment() -exp.append_model( - model_id='sys_model_1', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_process, - partial_state_update_blocks=partial_state_update_blocks -) -multi_exp.append_model( - model_id='sys_model_2', - sim_configs=sim_config, - initial_state=genesis_states, - env_processes=env_process, - partial_state_update_blocks=partial_state_update_blocks -) \ No newline at end of file diff --git a/testing/generic_test.py b/testing/generic_test.py index 5c35cca6..015312ce 100644 --- a/testing/generic_test.py +++ b/testing/generic_test.py @@ -1,9 +1,8 @@ import unittest from functools import reduce -from tabulate import tabulate +from tabulate import tabulate # type: ignore from parameterized import parameterized - def generate_assertions_df(df, expected_results, target_cols, evaluations): test_names = [] for eval_f in evaluations: diff --git a/testing/tests/cadCAD_memory_address.json b/testing/tests/cadCAD_memory_address.json new file mode 100644 index 00000000..859ec289 --- /dev/null +++ b/testing/tests/cadCAD_memory_address.json @@ -0,0 +1 @@ +{"memory_address": "0x10fbedd50"} \ No newline at end of file diff --git a/testing/tests/dev/compare_results_dev.py b/testing/tests/dev/compare_results_dev.py deleted file mode 100644 index b950ae9d..00000000 --- a/testing/tests/dev/compare_results_dev.py +++ /dev/null @@ -1,40 +0,0 @@ -import unittest -import pandas as pd -from copy import deepcopy - -from testing.models import param_sweep -from testing.results_comparison import dataframe_difference, compare_results -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor - -exec_mode = ExecutionMode() -exec_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=exec_ctx, configs=param_sweep.exp.configs) - -raw_result, tensor_fields, sessions = run.execute() -result = pd.DataFrame(raw_result) -# print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -# pprint(sessions) -# print(tabulate(result, headers='keys', tablefmt='psql')) - - -result_1 = result -result_2 = deepcopy(result) -result_df1 = pd.DataFrame({'a': [1, 2], 'b': [3, 5]}) -result_df2 = pd.DataFrame({'a': ['hi', 2], 'b': [3.0, 4.0]}) - -# print(result_df1.shape) -# exit() - - - -equivalent_result_diff = dataframe_difference(result_1, result_2) -different_result_diff = dataframe_difference(result_df1, result_df2) - -class dfCompareTest(compare_results(different_result_diff)): - pass - -class EquivalentTest(compare_results(equivalent_result_diff)): - pass - -if __name__ == '__main__': - unittest.main() diff --git a/testing/tests/dev/compare_results_old.py b/testing/tests/dev/compare_results_old.py deleted file mode 100644 index 68210670..00000000 --- a/testing/tests/dev/compare_results_old.py +++ /dev/null @@ -1,79 +0,0 @@ -from copy import deepcopy - -import pandas as pd -import numpy as np -# import pandasql -# from tabulate import tabulate -from tabulate import tabulate - -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor -from testing.models import param_sweep - -exec_mode = ExecutionMode() -exec_ctx = ExecutionContext(context=exec_mode.local_mode) -run = Executor(exec_context=exec_ctx, configs=param_sweep.exp.configs) - -raw_result, tensor_fields, sessions = run.execute() -result = pd.DataFrame(raw_result) -# print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -# pprint(sessions) -# print(tabulate(result, headers='keys', tablefmt='psql')) - -# result_1 = result -# result_2 = deepcopy(result) - -# test_df1 = pd.testing.assert_frame_equal(result_1, result_2) -# print(tabulate(test_df1, headers='keys', tablefmt='psql')) - -result_df = pd.DataFrame({'a': [1, 2], 'b': [3, 5]}).reset_index() -df2 = pd.DataFrame({'a': [3.1, 2], 'b': [3.0, 4.0]}).reset_index() - -# test_df2 = pd.testing.assert_frame_equal(df1, df2) -# print(tabulate(test_df2, headers='keys', tablefmt='psql')) - -def dataframe_difference(df1: pd.DataFrame, df2: pd.DataFrame, which=None): - """ - Find rows which are different between two DataFrames. - https://hackersandslackers.com/compare-rows-pandas-dataframes/ - """ - comparison_df = df1.merge( - df2, - indicator=True, - how='outer' - ) - if which is None: - diff_df = comparison_df[comparison_df['_merge'] != 'both'] - else: - diff_df = comparison_df[comparison_df['_merge'] == which] - # diff_df.to_csv('data/diff.csv') - return diff_df - - -merge_df = dataframe_difference(result_df, df2) -cols_no__merge = list(filter(lambda col: '_merge' not in col, merge_df.columns.tolist())) -cols_no_index = list(filter(lambda col: 'index' not in col, cols_no__merge)) -aggregation = dict((k, 'unique') for k in cols_no_index) -diff_df = merge_df[cols_no__merge].groupby('index').agg(aggregation) -# print(tabulate(diff_df, headers='keys', tablefmt='psql')) - - -def discrepancies(row): - return dict([ - (col, list(vals)) for col, vals in row.items() - if type(vals) is np.ndarray and len(vals) > 1 - ]) - - -def val_error(val): - if type(val) is dict: - return False - else: - return True - - -diff_df['discrepancies'] = diff_df.apply(discrepancies, axis=1) -discrepancies_df = diff_df[['discrepancies']] -result_diff = result_df.merge(discrepancies_df, how='left', on='index') -result_diff['val_error'] = result_diff['discrepancies'].apply(val_error) -print(tabulate(result_diff, headers='keys', tablefmt='psql')) - diff --git a/testing/tests/dev/out_check_dev.py b/testing/tests/dev/out_check_dev.py deleted file mode 100644 index 99d5fedb..00000000 --- a/testing/tests/dev/out_check_dev.py +++ /dev/null @@ -1,28 +0,0 @@ -from pprint import pprint - -import pandas as pd -from tabulate import tabulate -from cadCAD.engine import ExecutionMode, ExecutionContext, Executor - -from testing.models import param_sweep -from cadCAD import configs - -exec_mode = ExecutionMode() - -exec_ctx = ExecutionContext(context=exec_mode.local_mode) -# exec_ctx = ExecutionContext(context=exec_mode.multi_proc) -run = Executor(exec_context=exec_ctx, configs=configs) - -raw_result, tensor_fields, sessions = run.execute() -result = pd.DataFrame(raw_result) -# print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -# pprint(sessions) -print(tabulate(result, headers='keys', tablefmt='psql')) - -print() - -raw_result, tensor_fields, sessions = run.execute() -result = pd.DataFrame(raw_result) -# print(tabulate(tensor_fields[0], headers='keys', tablefmt='psql')) -# pprint(sessions) -print(tabulate(result, headers='keys', tablefmt='psql')) \ No newline at end of file diff --git a/testing/tests/dev/utils_dev.py b/testing/tests/dev/utils_dev.py deleted file mode 100644 index 63885f81..00000000 --- a/testing/tests/dev/utils_dev.py +++ /dev/null @@ -1,5 +0,0 @@ -def gen_metric_row(row, cols): - return ((row['run'], row['timestep'], row['substep']), {col: row[col] for col in cols}) - -def gen_metric_dict(df, cols): - return dict([gen_metric_row(row, cols) for index, row in df.iterrows()]) diff --git a/testing/tests/import_cadCAD.ipynb b/testing/tests/import_cadCAD.ipynb index 01b8083d..325b2c60 100644 --- a/testing/tests/import_cadCAD.ipynb +++ b/testing/tests/import_cadCAD.ipynb @@ -2,7 +2,7 @@ "cells": [ { "cell_type": "code", - "execution_count": 16, + "execution_count": 2, "id": "15b9b09a", "metadata": {}, "outputs": [], @@ -12,7 +12,7 @@ }, { "cell_type": "code", - "execution_count": 25, + "execution_count": 3, "id": "1b54b99a", "metadata": {}, "outputs": [], @@ -42,7 +42,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.6.13" + "version": "3.11.0" } }, "nbformat": 4, From 0a291e8cafc51eeb46c4ee2481a2a2bc02d91c62 Mon Sep 17 00:00:00 2001 From: Danilo Lessa Bernardineli Date: Thu, 14 Dec 2023 17:36:30 -0300 Subject: [PATCH 3/6] Add Type Hints for cadCAD objects (#313) * add types.py --- cadCAD/types.py | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) create mode 100644 cadCAD/types.py diff --git a/cadCAD/types.py b/cadCAD/types.py new file mode 100644 index 00000000..5b47964e --- /dev/null +++ b/cadCAD/types.py @@ -0,0 +1,18 @@ +from typing import TypedDict, Callable, Union, Dict, List, Tuple + +State = Dict[str, object] +Parameters = Dict[str, object] +Substep = int +StateHistory = List[List[State]] +PolicyOutput = Dict[str, object] +StateVariable = object + +PolicyFunction = Callable[[Parameters, Substep, StateHistory, State], PolicyOutput] +StateUpdateFunction = Callable[[Parameters, Substep, StateHistory, State, PolicyOutput], Tuple[str, StateVariable]] + +class StateUpdateBlock(TypedDict): + policies: Dict[str, PolicyFunction] + variables: Dict[str, StateUpdateFunction] + + +StateUpdateBlocks = List[StateUpdateBlock] From 1b3241e6db03ce8f6991a04ce82661e1f41cf57c Mon Sep 17 00:00:00 2001 From: Danilo Lessa Bernardineli Date: Thu, 14 Dec 2023 20:32:55 -0300 Subject: [PATCH 4/6] Fix single_proc + add support for single_proc for multi-runs & sweeps (#315) * fix tests + rm simulations/ folder * add types.py * single run / multi mc is ok * fix for single run / single param * add support for single proc runs --------- Co-authored-by: Emanuel Lima --- cadCAD/configuration/utils/__init__.py | 52 +++--- cadCAD/engine/__init__.py | 77 ++++++--- cadCAD/engine/execution.py | 91 ++++++----- cadCAD/types.py | 31 +++- cadCAD/utils/__init__.py | 7 +- testing/test_runs.py | 210 +++++++++++++++++++++++++ 6 files changed, 377 insertions(+), 91 deletions(-) create mode 100644 testing/test_runs.py diff --git a/cadCAD/configuration/utils/__init__.py b/cadCAD/configuration/utils/__init__.py index 5063a007..fce7bcb7 100644 --- a/cadCAD/configuration/utils/__init__.py +++ b/cadCAD/configuration/utils/__init__.py @@ -1,10 +1,11 @@ -import pandas as pd -import numpy as np +import pandas as pd # type: ignore from datetime import datetime, timedelta from collections import Counter from copy import deepcopy from functools import reduce -from funcy import curry +from funcy import curry # type: ignore +from cadCAD.types import * +from typing import Union, Dict, List from cadCAD.configuration.utils.depreciationHandler import sanitize_partial_state_updates from cadCAD.utils import dict_filter, contains_type, flatten_tabulated_dict, tabulate_dict @@ -161,27 +162,40 @@ def env_update(state_dict, sweep_dict, target_value): curry(trigger)(end_substep)(trigger_field)(trigger_vals)(funct_list) -def config_sim(d): - def process_variables(d): - return flatten_tabulated_dict(tabulate_dict(d)) +def config_sim(config_dict: ConfigurationDict): - if "N" in d: - if d["N"] <= 0: + if "N" in config_dict: + if config_dict["N"] <= 0: raise ValueError("'N' must be > 0") + else: + pass else: - raise KeyError("The 'sim_configs' dictionary must contain the key 'N'") - - if "T" not in d: - raise KeyError("The 'sim_configs' dictionary must contain the key 'T'") + raise KeyError("The 'sim_configs' dictionary must contain the key 'N' (# of Monte Carlo Runs)") - if "M" in d: - M_lengths = len(list(set({key: len(value) for key, value in d["M"].items()}.values()))) - if M_lengths > 2: - raise Exception('`M` values require up to a maximum of 2 distinct lengths') - return [{"N": d["N"], "T": d["T"], "M": M} for M in process_variables(d["M"])] + if "T" not in config_dict: + raise KeyError("The 'sim_configs' dictionary must contain the key 'T' (Timestep Iterator)") else: - d["M"] = [{}] - return d + if "M" in config_dict: + params = config_dict['M'] + + param_values_length = {key: len(value) if type(value) == list else 0 + for key, value in params.items()} + param_values_length_set = set(param_values_length.values()) + distinct_param_value_lengths = len(param_values_length_set) + + if distinct_param_value_lengths > 2: + raise Exception('When sweeping, `M` list lengths should either be 1 and/or equal. More than two distinct lengths are not allowed') + elif (distinct_param_value_lengths == 1) and (0 in param_values_length_set): + return config_dict + elif (1 in param_values_length_set): + return [{**config_dict, "M": M} + for M in flatten_tabulated_dict(tabulate_dict(params))] + else: + raise Exception('When sweeping, `M` list lengths should either be 1 and/or equal. ') + + else: + config_dict["M"] = [{}] + return config_dict def psub_list(psu_block, psu_steps): diff --git a/cadCAD/engine/__init__.py b/cadCAD/engine/__init__.py index 57d84bc7..c154e72f 100644 --- a/cadCAD/engine/__init__.py +++ b/cadCAD/engine/__init__.py @@ -7,6 +7,7 @@ from cadCAD.configuration.utils import TensorFieldReport, configs_as_objs, configs_as_dicts from cadCAD.engine.simulation import Executor as SimExecutor from cadCAD.engine.execution import single_proc_exec, parallelize_simulations, local_simulations +from cadCAD.types import * VarDictType = Dict[str, List[Any]] StatesListsType = List[Dict[str, Any]] @@ -24,6 +25,17 @@ class ExecutionMode: multi_proc = 'multi_proc' +def auto_mode_switcher(config_amt: int): + try: + if config_amt == 1: + return ExecutionMode.single_mode, single_proc_exec + elif (config_amt > 1): + return ExecutionMode.multi_mode, parallelize_simulations + except AttributeError: + if config_amt < 1: + raise ValueError('N must be >= 1!') + + class ExecutionContext: def __init__(self, context=ExecutionMode.local_mode, method=None, additional_objs=None) -> None: self.name = context @@ -39,7 +51,7 @@ def distroduce_proc( ExpIDs, SubsetIDs, SubsetWindows, - configured_n, # exec_method, + configured_n, # exec_method, sc, additional_objs=additional_objs ): return method( @@ -47,7 +59,7 @@ def distroduce_proc( ExpIDs, SubsetIDs, SubsetWindows, - configured_n, # exec_method, + configured_n, # exec_method, sc, additional_objs ) @@ -56,8 +68,8 @@ def distroduce_proc( class Executor: def __init__(self, - exec_context: ExecutionContext, configs: List[Configuration], sc=None, empty_return=False - ) -> None: + exec_context: ExecutionContext, configs: List[Configuration], sc=None, empty_return=False + ) -> None: self.sc = sc self.SimExecutor = SimExecutor self.exec_method = exec_context.method @@ -70,7 +82,8 @@ def execute(self) -> Tuple[Any, Any, Dict[str, Any]]: return [], [], [] config_proc = Processor() - create_tensor_field = TensorFieldReport(config_proc).create_tensor_field + create_tensor_field = TensorFieldReport( + config_proc).create_tensor_field sessions = [] var_dict_list, states_lists = [], [] @@ -105,18 +118,30 @@ def execute(self) -> Tuple[Any, Any, Dict[str, Any]]: var_dict_list.append(x.sim_config['M']) states_lists.append([x.initial_state]) eps.append(list(x.exogenous_states.values())) - configs_structs.append(config_proc.generate_config(x.initial_state, x.partial_state_update_blocks, eps[config_idx])) + configs_structs.append(config_proc.generate_config( + x.initial_state, x.partial_state_update_blocks, eps[config_idx])) env_processes_list.append(x.env_processes) partial_state_updates.append(x.partial_state_update_blocks) sim_executors.append(SimExecutor(x.policy_ops).simulation) config_idx += 1 - def get_final_dist_results(simulations, psus, eps, sessions): - tensor_fields = [create_tensor_field(psu, ep) for psu, ep in list(zip(psus, eps))] + remote_threshold = 100 + config_amt = len(self.configs) + + def get_final_dist_results(simulations: List[StateHistory], + psus: List[StateUpdateBlocks], + eps, + sessions: List[SessionDict]): + tensor_fields = [create_tensor_field( + psu, ep) for psu, ep in list(zip(psus, eps))] return simulations, tensor_fields, sessions - def get_final_results(simulations, psus, eps, sessions, remote_threshold): + def get_final_results(simulations: List[StateHistory], + psus: List[StateUpdateBlocks], + eps, + sessions: List[SessionDict], + remote_threshold: int): flat_timesteps, tensor_fields = [], [] for sim_result, psu, ep in list(zip(simulations, psus, eps)): flat_timesteps.append(flatten(sim_result)) @@ -128,25 +153,23 @@ def get_final_results(simulations, psus, eps, sessions, remote_threshold): elif config_amt > 1: return flat_simulations, tensor_fields, sessions - remote_threshold = 100 - config_amt = len(self.configs) - - def auto_mode_switcher(config_amt): - try: - if config_amt == 1: - return ExecutionMode.single_mode, single_proc_exec - elif (config_amt > 1): - return ExecutionMode.multi_mode, parallelize_simulations - except AttributeError: - if config_amt < 1: - raise ValueError('N must be >= 1!') - final_result = None original_N = len(configs_as_dicts(self.configs)) if self.exec_context != ExecutionMode.distributed: # Consider Legacy Support - if self.exec_context != ExecutionMode.local_mode: - self.exec_context, self.exec_method = auto_mode_switcher(config_amt) + if self.exec_context == ExecutionMode.local_mode: + self.exec_context, self.exec_method = auto_mode_switcher( + config_amt) + elif self.exec_context == ExecutionMode.single_mode or self.exec_context == ExecutionMode.single_proc: + self.exec_context, self.exec_method = ExecutionMode.single_mode, single_proc_exec + elif self.exec_context == ExecutionMode.multi_mode or self.exec_context == ExecutionMode.multi_proc: + if config_amt == 1: + raise ValueError("Multi mode must have at least 2 configs") + else: + self.exec_context, self.exec_method = ExecutionMode.multi_mode, parallelize_simulations + else: + raise ValueError("Invalid execution mode specified") + print("Execution Method: " + self.exec_method.__name__) simulations_results = self.exec_method( @@ -154,14 +177,16 @@ def auto_mode_switcher(config_amt): ExpIDs, SubsetIDs, SubsetWindows, original_N ) - final_result = get_final_results(simulations_results, partial_state_updates, eps, sessions, remote_threshold) + final_result = get_final_results( + simulations_results, partial_state_updates, eps, sessions, remote_threshold) elif self.exec_context == ExecutionMode.distributed: print("Execution Method: " + self.exec_method.__name__) simulations_results = self.exec_method( sim_executors, var_dict_list, states_lists, configs_structs, env_processes_list, Ts, SimIDs, RunIDs, ExpIDs, SubsetIDs, SubsetWindows, original_N, self.sc ) - final_result = get_final_dist_results(simulations_results, partial_state_updates, eps, sessions) + final_result = get_final_dist_results( + simulations_results, partial_state_updates, eps, sessions) t2 = time() print(f"Total execution time: {t2 - t1 :.2f}s") diff --git a/cadCAD/engine/execution.py b/cadCAD/engine/execution.py index 41e86e87..fdf0452c 100644 --- a/cadCAD/engine/execution.py +++ b/cadCAD/engine/execution.py @@ -1,7 +1,7 @@ from typing import Callable, Dict, List, Any, Tuple -from pathos.multiprocessing import ProcessPool as PPool +from pathos.multiprocessing import ProcessPool as PPool # type: ignore from collections import Counter - +from cadCAD.types import * from cadCAD.utils import flatten VarDictType = Dict[str, List[Any]] @@ -11,26 +11,31 @@ def single_proc_exec( - simulation_execs: List[Callable], - var_dict_list: List[VarDictType], - states_lists: List[StatesListsType], - configs_structs: List[ConfigsType], - env_processes_list: List[EnvProcessesType], - Ts: List[range], - SimIDs, - Ns: List[int], + simulation_execs: List[ExecutorFunction], + var_dict_list: List[Parameters], + states_lists: List[StateHistory], + configs_structs: List[StateUpdateBlocks], + env_processes_list: List[EnvProcesses], + Ts: List[TimeSeq], + SimIDs: List[SimulationID], + Ns: List[Run], ExpIDs: List[int], - SubsetIDs, - SubsetWindows, - configured_n + SubsetIDs: List[SubsetID], + SubsetWindows: List[SubsetWindow], + configured_n: List[N_Runs] ): + + # HACK for making it run with N_Runs=1 + if type(var_dict_list) == list: + var_dict_list = var_dict_list[0] + print(f'Execution Mode: single_threaded') - params = [ + raw_params: List[List] = [ simulation_execs, states_lists, configs_structs, env_processes_list, Ts, SimIDs, Ns, SubsetIDs, SubsetWindows ] simulation_exec, states_list, config, env_processes, T, sim_id, N, subset_id, subset_window = list( - map(lambda x: x.pop(), params) + map(lambda x: x.pop(), raw_params) ) result = simulation_exec( var_dict_list, states_list, config, env_processes, T, sim_id, N, subset_id, subset_window, configured_n @@ -38,19 +43,22 @@ def single_proc_exec( return flatten(result) + + + def parallelize_simulations( - simulation_execs: List[Callable], - var_dict_list: List[VarDictType], - states_lists: List[StatesListsType], - configs_structs: List[ConfigsType], - env_processes_list: List[EnvProcessesType], - Ts: List[range], - SimIDs, - Ns: List[int], + simulation_execs: List[ExecutorFunction], + var_dict_list: List[Parameters], + states_lists: List[StateHistory], + configs_structs: List[StateUpdateBlocks], + env_processes_list: List[EnvProcesses], + Ts: List[TimeSeq], + SimIDs: List[SimulationID], + Ns: List[Run], ExpIDs: List[int], - SubsetIDs, - SubsetWindows, - configured_n + SubsetIDs: List[SubsetID], + SubsetWindows: List[SubsetWindow], + configured_n: List[N_Runs] ): print(f'Execution Mode: parallelized') @@ -104,32 +112,29 @@ def process_executor(params): def local_simulations( - simulation_execs: List[Callable], - var_dict_list: List[VarDictType], - states_lists: List[StatesListsType], - configs_structs: List[ConfigsType], - env_processes_list: List[EnvProcessesType], - Ts: List[range], - SimIDs, - Ns: List[int], - ExpIDs: List[int], - SubsetIDs, - SubsetWindows, - configured_n + simulation_execs: List[ExecutorFunction], + var_dict_list: List[Parameters], + states_lists: List[StateHistory], + configs_structs: List[StateUpdateBlocks], + env_processes_list: List[EnvProcesses], + Ts: List[TimeSeq], + SimIDs: List[SimulationID], + Ns: List[Run], + ExpIDs: List[int], + SubsetIDs: List[SubsetID], + SubsetWindows: List[SubsetWindow], + configured_n: List[N_Runs] ): config_amt = len(configs_structs) - _params = None if config_amt == 1: # and configured_n != 1 - _params = var_dict_list[0] return single_proc_exec( - simulation_execs, _params, states_lists, configs_structs, env_processes_list, + simulation_execs, var_dict_list, states_lists, configs_structs, env_processes_list, Ts, SimIDs, Ns, ExpIDs, SubsetIDs, SubsetWindows, configured_n ) elif config_amt > 1: # and configured_n != 1 - _params = var_dict_list return parallelize_simulations( - simulation_execs, _params, states_lists, configs_structs, env_processes_list, + simulation_execs, var_dict_list, states_lists, configs_structs, env_processes_list, Ts, SimIDs, Ns, ExpIDs, SubsetIDs, SubsetWindows, configured_n ) # elif config_amt > 1 and configured_n == 1: diff --git a/cadCAD/types.py b/cadCAD/types.py index 5b47964e..5e8eb274 100644 --- a/cadCAD/types.py +++ b/cadCAD/types.py @@ -1,7 +1,9 @@ -from typing import TypedDict, Callable, Union, Dict, List, Tuple +from typing import TypedDict, Callable, Union, Dict, List, Tuple, Iterator +from collections import deque State = Dict[str, object] Parameters = Dict[str, object] +SweepableParameters = Dict[str, list[object]] Substep = int StateHistory = List[List[State]] PolicyOutput = Dict[str, object] @@ -16,3 +18,30 @@ class StateUpdateBlock(TypedDict): StateUpdateBlocks = List[StateUpdateBlock] + +class ConfigurationDict(TypedDict): + T: Iterator # Generator for the timestep variable + N: int # Number of MC Runs + M: Union[Parameters, SweepableParameters] # Parameters / List of Parameter to Sweep + + +EnvProcesses = object +TimeSeq = Iterator +SimulationID = int +Run = int +SubsetID = int +SubsetWindow = Iterator +N_Runs = int + +ExecutorFunction = Callable[[Parameters, StateHistory, StateUpdateBlocks, EnvProcesses, TimeSeq, SimulationID, Run, SubsetID, SubsetWindow, N_Runs], object] +ExecutionParameter = Tuple[ExecutorFunction, Parameters, StateHistory, StateUpdateBlocks, EnvProcesses, TimeSeq, SimulationID, Run, SubsetID, SubsetWindow, N_Runs] + + +class SessionDict(TypedDict): + user_id: str + experiment_id: int + session_id: str + simulation_id: int + run_id: int + subset_id: int + subset_window: deque diff --git a/cadCAD/utils/__init__.py b/cadCAD/utils/__init__.py index 2c75a57b..cb7977a0 100644 --- a/cadCAD/utils/__init__.py +++ b/cadCAD/utils/__init__.py @@ -3,11 +3,14 @@ from collections import defaultdict from itertools import product import warnings +from typing import Union +from cadCAD.types import * +from typing import List, Dict, Union import functools import operator -from pandas import DataFrame +from pandas import DataFrame # type: ignore class SilentDF(DataFrame): @@ -99,7 +102,7 @@ def tabulate_dict(d: Dict[str, List[int]]) -> Dict[str, List[int]]: def flatten_tabulated_dict(d: Dict[str, List[int]]) -> List[Dict[str, int]]: max_len = get_max_dict_val_len(d) - dl = [{} for i in range(max_len)] + dl: list[dict] = [{} for i in range(max_len)] for k, vl in d.items(): for v, i in zip(vl, list(range(len(vl)))): diff --git a/testing/test_runs.py b/testing/test_runs.py new file mode 100644 index 00000000..2c33c2b4 --- /dev/null +++ b/testing/test_runs.py @@ -0,0 +1,210 @@ +from typing import Dict, List +from cadCAD.engine import Executor, ExecutionContext, ExecutionMode +from cadCAD.configuration import Experiment +from cadCAD.configuration.utils import env_trigger, var_substep_trigger, config_sim, psub_list +from cadCAD.types import * +import pandas as pd # type: ignore +import types +import inspect +import pytest + +def describe_or_return(v: object) -> object: + """ + Thanks @LinuxIsCool! + """ + if isinstance(v, types.FunctionType): + return f'function: {v.__name__}' + elif isinstance(v, types.LambdaType) and v.__name__ == '': + return f'lambda: {inspect.signature(v)}' + else: + return v + + +def select_M_dict(M_dict: Dict[str, object], keys: set) -> Dict[str, object]: + """ + Thanks @LinuxIsCool! + """ + return {k: describe_or_return(v) for k, v in M_dict.items() if k in keys} + + +def select_config_M_dict(configs: list, i: int, keys: set) -> Dict[str, object]: + return select_M_dict(configs[i].sim_config['M'], keys) + + +def drop_substeps(_df): + first_ind = (_df.substep == 0) & (_df.timestep == 0) + last_ind = _df.substep == max(_df.substep) + inds_to_drop = first_ind | last_ind + return _df.copy().loc[inds_to_drop].drop(columns=['substep']) + + +def assign_params(_df: pd.DataFrame, configs) -> pd.DataFrame: + """ + Based on `cadCAD-tools` package codebase, by @danlessa + """ + M_dict = configs[0].sim_config['M'] + params_set = set(M_dict.keys()) + selected_params = params_set + + # Attribute parameters to each row + # 1. Assign the parameter set from the first row first, so that + # columns are created + first_param_dict = select_config_M_dict(configs, 0, selected_params) + + # 2. Attribute parameter on an (simulation, subset, run) basis + df = _df.assign(**first_param_dict).copy() + for i, (_, subset_df) in enumerate(df.groupby(['simulation', 'subset', 'run'])): + df.loc[subset_df.index] = subset_df.assign(**select_config_M_dict(configs, + i, + selected_params)) + return df + + + + +SWEEP_PARAMS: Dict[str, List] = { + 'alpha': [1], + 'beta': [lambda x: 2 * x, lambda x: x], + 'gamma': [3, 4], + 'omega': [7] + } + +SINGLE_PARAMS: Dict[str, object] = { + 'alpha': 1, + 'beta': lambda x: x, + 'gamma': 3, + 'omega': 5 + } + + +def create_experiment(N_RUNS=2, N_TIMESTEPS=3, params: dict=SWEEP_PARAMS): + psu_steps = ['m1', 'm2', 'm3'] + system_substeps = len(psu_steps) + var_timestep_trigger = var_substep_trigger([0, system_substeps]) + env_timestep_trigger = env_trigger(system_substeps) + env_process = {} + + + # ['s1', 's2', 's3', 's4'] + # Policies per Mechanism + def gamma(params: Parameters, substep: Substep, history: StateHistory, state: State, **kwargs): + return {'gamma': params['gamma']} + + + def omega(params: Parameters, substep: Substep, history: StateHistory, state: State, **kwarg): + return {'omega': params['omega']} + + + # Internal States per Mechanism + def alpha(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'alpha_var', params['alpha'] + + + def beta(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'beta_var', params['beta'] + + def gamma_var(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'gamma_var', params['gamma'] + + def omega_var(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'omega_var', params['omega'] + + + def policies(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'policies', _input + + + def sweeped(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'sweeped', {'beta': params['beta'], 'gamma': params['gamma']} + + psu_block: dict = {k: {"policies": {}, "states": {}} for k in psu_steps} + for m in psu_steps: + psu_block[m]['policies']['gamma'] = gamma + psu_block[m]['policies']['omega'] = omega + psu_block[m]["states"]['alpha_var'] = alpha + psu_block[m]["states"]['beta_var'] = beta + psu_block[m]["states"]['gamma_var'] = gamma_var + psu_block[m]["states"]['omega_var'] = omega_var + psu_block[m]['states']['policies'] = policies + psu_block[m]["states"]['sweeped'] = var_timestep_trigger(y='sweeped', f=sweeped) + + + # Genesis States + genesis_states = { + 'alpha_var': 0, + 'beta_var': 0, + 'gamma_var': 0, + 'omega_var': 0, + 'policies': {}, + 'sweeped': {} + } + + # Environment Process + env_process['sweeped'] = env_timestep_trigger(trigger_field='timestep', trigger_vals=[5], funct_list=[lambda _g, x: _g['beta']]) + + sim_config = config_sim( + { + "N": N_RUNS, + "T": range(N_TIMESTEPS), + "M": params, # Optional + } + ) + + # New Convention + partial_state_update_blocks = psub_list(psu_block, psu_steps) + + exp = Experiment() + exp.append_model( + sim_configs=sim_config, + initial_state=genesis_states, + env_processes=env_process, + partial_state_update_blocks=partial_state_update_blocks + ) + return exp + + +def test_mc_sweep_experiment(): + experiment_assertions(create_experiment(N_RUNS=2, N_TIMESTEPS=2, params=SWEEP_PARAMS), ExecutionMode.local_mode) + experiment_assertions(create_experiment(N_RUNS=2, N_TIMESTEPS=2, params=SWEEP_PARAMS), ExecutionMode.single_mode) + experiment_assertions(create_experiment(N_RUNS=2, N_TIMESTEPS=2, params=SWEEP_PARAMS), ExecutionMode.multi_mode) + +def test_unique_sweep_experiment(): + experiment_assertions(create_experiment(N_RUNS=1, N_TIMESTEPS=2, params=SWEEP_PARAMS), ExecutionMode.local_mode) + experiment_assertions(create_experiment(N_RUNS=1, N_TIMESTEPS=2, params=SWEEP_PARAMS), ExecutionMode.single_mode) + experiment_assertions(create_experiment(N_RUNS=1, N_TIMESTEPS=2, params=SWEEP_PARAMS), ExecutionMode.multi_mode) + +def test_mc_single_experiment(): + experiment_assertions(create_experiment(N_RUNS=2, N_TIMESTEPS=2, params=SINGLE_PARAMS), ExecutionMode.local_mode) + experiment_assertions(create_experiment(N_RUNS=2, N_TIMESTEPS=2, params=SINGLE_PARAMS), ExecutionMode.single_mode) + experiment_assertions(create_experiment(N_RUNS=2, N_TIMESTEPS=2, params=SINGLE_PARAMS), ExecutionMode.multi_mode) + +def test_unique_single_experiment(): + experiment_assertions(create_experiment(N_RUNS=1, N_TIMESTEPS=2, params=SINGLE_PARAMS), ExecutionMode.local_mode) + experiment_assertions(create_experiment(N_RUNS=1, N_TIMESTEPS=2, params=SINGLE_PARAMS), ExecutionMode.single_mode) + with pytest.raises(ValueError) as e_info: + experiment_assertions(create_experiment(N_RUNS=1, N_TIMESTEPS=2, params=SINGLE_PARAMS), ExecutionMode.multi_mode) + + + +def experiment_assertions(exp, mode=None): + if mode == None: + mode = ExecutionMode().local_mode + exec_context = ExecutionContext(mode) + executor = Executor(exec_context=exec_context, configs=exp.configs) + (records, tensor_field, _) = executor.execute() + df = drop_substeps(assign_params(pd.DataFrame(records), exp.configs)) + + # XXX: parameters should always be of the same type. Else, the test will fail + first_sim_config = exp.configs[0].sim_config['M'] + + + for (i, row) in df.iterrows(): + if row.timestep > 0: + + assert row['alpha_var'] == row['alpha'] + assert type(row['alpha_var']) == type(first_sim_config['alpha']) + assert row['gamma_var'] == row['gamma'] + assert type(row['gamma_var']) == type(first_sim_config['gamma']) + assert row['omega_var'] == row['omega'] + assert type(row['omega_var']) == type(first_sim_config['omega']) + From 0aa86be59831a739145fd4578f1a6a71a199c2cd Mon Sep 17 00:00:00 2001 From: Danilo Lessa Bernardineli Date: Thu, 14 Dec 2023 20:38:50 -0300 Subject: [PATCH 5/6] Add switch for toggling deepcopy off (#316) * fix tests + rm simulations/ folder * add types.py * single run / multi mc is ok * fix for single run / single param * add support for single proc runs * add switch for using deepcopy + fix bug on additional_objs * bug fix * bug fix --------- Co-authored-by: Emanuel Lima --- cadCAD/engine/__init__.py | 6 +- cadCAD/engine/execution.py | 17 +-- cadCAD/engine/simulation.py | 50 +++++---- cadCAD/types.py | 4 +- testing/test_additional_objs.py | 187 ++++++++++++++++++++++++++++++++ testing/test_arg_count.py | 51 +++++++++ 6 files changed, 286 insertions(+), 29 deletions(-) create mode 100644 testing/test_additional_objs.py create mode 100644 testing/test_arg_count.py diff --git a/cadCAD/engine/__init__.py b/cadCAD/engine/__init__.py index c154e72f..59e49d6b 100644 --- a/cadCAD/engine/__init__.py +++ b/cadCAD/engine/__init__.py @@ -1,5 +1,5 @@ from time import time -from typing import Callable, Dict, List, Any, Tuple +from typing import Callable, Dict, List, Any, Tuple, Union from cadCAD.utils import flatten from cadCAD.utils.execution import print_exec_info @@ -39,6 +39,7 @@ def auto_mode_switcher(config_amt: int): class ExecutionContext: def __init__(self, context=ExecutionMode.local_mode, method=None, additional_objs=None) -> None: self.name = context + self.additional_objs = additional_objs if context == 'local_proc': self.method = local_simulations elif context == 'single_proc': @@ -74,6 +75,7 @@ def __init__(self, self.SimExecutor = SimExecutor self.exec_method = exec_context.method self.exec_context = exec_context.name + self.additional_objs = exec_context.additional_objs self.configs = configs self.empty_return = empty_return @@ -174,7 +176,7 @@ def get_final_results(simulations: List[StateHistory], print("Execution Method: " + self.exec_method.__name__) simulations_results = self.exec_method( sim_executors, var_dict_list, states_lists, configs_structs, env_processes_list, Ts, SimIDs, RunIDs, - ExpIDs, SubsetIDs, SubsetWindows, original_N + ExpIDs, SubsetIDs, SubsetWindows, original_N, self.additional_objs ) final_result = get_final_results( diff --git a/cadCAD/engine/execution.py b/cadCAD/engine/execution.py index fdf0452c..febafffe 100644 --- a/cadCAD/engine/execution.py +++ b/cadCAD/engine/execution.py @@ -22,7 +22,8 @@ def single_proc_exec( ExpIDs: List[int], SubsetIDs: List[SubsetID], SubsetWindows: List[SubsetWindow], - configured_n: List[N_Runs] + configured_n: List[N_Runs], + additional_objs=None ): # HACK for making it run with N_Runs=1 @@ -38,7 +39,7 @@ def single_proc_exec( map(lambda x: x.pop(), raw_params) ) result = simulation_exec( - var_dict_list, states_list, config, env_processes, T, sim_id, N, subset_id, subset_window, configured_n + var_dict_list, states_list, config, env_processes, T, sim_id, N, subset_id, subset_window, configured_n, additional_objs ) return flatten(result) @@ -58,7 +59,8 @@ def parallelize_simulations( ExpIDs: List[int], SubsetIDs: List[SubsetID], SubsetWindows: List[SubsetWindow], - configured_n: List[N_Runs] + configured_n: List[N_Runs], + additional_objs=None ): print(f'Execution Mode: parallelized') @@ -96,7 +98,7 @@ def process_executor(params): if len_configs_structs > 1: pp = PPool(processes=len_configs_structs) results = pp.map( - lambda t: t[0](t[1], t[2], t[3], t[4], t[5], t[6], t[7], t[8], t[9], configured_n), params + lambda t: t[0](t[1], t[2], t[3], t[4], t[5], t[6], t[7], t[8], t[9], configured_n, additional_objs), params ) pp.close() pp.join() @@ -123,18 +125,19 @@ def local_simulations( ExpIDs: List[int], SubsetIDs: List[SubsetID], SubsetWindows: List[SubsetWindow], - configured_n: List[N_Runs] + configured_n: List[N_Runs], + additional_objs=None ): config_amt = len(configs_structs) if config_amt == 1: # and configured_n != 1 return single_proc_exec( simulation_execs, var_dict_list, states_lists, configs_structs, env_processes_list, - Ts, SimIDs, Ns, ExpIDs, SubsetIDs, SubsetWindows, configured_n + Ts, SimIDs, Ns, ExpIDs, SubsetIDs, SubsetWindows, configured_n, additional_objs ) elif config_amt > 1: # and configured_n != 1 return parallelize_simulations( simulation_execs, var_dict_list, states_lists, configs_structs, env_processes_list, - Ts, SimIDs, Ns, ExpIDs, SubsetIDs, SubsetWindows, configured_n + Ts, SimIDs, Ns, ExpIDs, SubsetIDs, SubsetWindows, configured_n, additional_objs ) # elif config_amt > 1 and configured_n == 1: diff --git a/cadCAD/engine/simulation.py b/cadCAD/engine/simulation.py index cb492706..1e7829e5 100644 --- a/cadCAD/engine/simulation.py +++ b/cadCAD/engine/simulation.py @@ -1,10 +1,12 @@ from typing import Any, Callable, Dict, List, Tuple from copy import deepcopy +from types import MappingProxyType from functools import reduce -from funcy import curry +from funcy import curry # type: ignore from cadCAD.utils import flatten from cadCAD.engine.utils import engine_exception +from cadCAD.types import * id_exception: Callable = curry(engine_exception)(KeyError)(KeyError)(None) @@ -102,20 +104,30 @@ def env_composition(target_field, state_dict, target_value): # mech_step def partial_state_update( self, - sweep_dict: Dict[str, List[Any]], - sub_step: int, - sL, - sH, - state_funcs: List[Callable], - policy_funcs: List[Callable], - env_processes: Dict[str, Callable], + sweep_dict: Parameters, + sub_step: Substep, + sL: list[State], + sH: StateHistory, + state_funcs: List[StateUpdateFunction], + policy_funcs: List[PolicyFunction], + env_processes: EnvProcesses, time_step: int, run: int, additional_objs ) -> List[Dict[str, Any]]: - # last_in_obj: Dict[str, Any] = MappingProxyType(sL[-1]) - last_in_obj: Dict[str, Any] = deepcopy(sL[-1]) + if type(additional_objs) == dict: + if additional_objs.get('deepcopy_off', False) == True: + last_in_obj = MappingProxyType(sL[-1]) + if len(additional_objs) == 1: + additional_objs = None + # XXX: drop the additional objects if only used for deepcopy + # toggling. + else: + last_in_obj = deepcopy(sL[-1]) + else: + last_in_obj = deepcopy(sL[-1]) + _input: Dict[str, Any] = self.policy_update_exception( self.get_policy_input(sweep_dict, sub_step, sH, last_in_obj, policy_funcs, additional_objs) ) @@ -206,18 +218,18 @@ def run_pipeline( def simulation( self, - sweep_dict: Dict[str, List[Any]], - states_list: List[Dict[str, Any]], + sweep_dict: SweepableParameters, + states_list: StateHistory, configs, - env_processes: Dict[str, Callable], - time_seq: range, - simulation_id: int, + env_processes: EnvProcesses, + time_seq: TimeSeq, + simulation_id: SimulationID, run: int, - subset_id, - subset_window, - configured_N, + subset_id: SubsetID, + subset_window: SubsetWindow, + configured_N: int, # remote_ind - additional_objs=None + additional_objs: Union[None, Dict]=None ): run += 1 subset_window.appendleft(subset_id) diff --git a/cadCAD/types.py b/cadCAD/types.py index 5e8eb274..4f5de596 100644 --- a/cadCAD/types.py +++ b/cadCAD/types.py @@ -25,7 +25,9 @@ class ConfigurationDict(TypedDict): M: Union[Parameters, SweepableParameters] # Parameters / List of Parameter to Sweep -EnvProcesses = object +TargetValue = object +EnvProcess: Callable[[State, SweepableParameters, TargetValue], TargetValue] +EnvProcesses = dict[str, Callable] TimeSeq = Iterator SimulationID = int Run = int diff --git a/testing/test_additional_objs.py b/testing/test_additional_objs.py new file mode 100644 index 00000000..89cd2995 --- /dev/null +++ b/testing/test_additional_objs.py @@ -0,0 +1,187 @@ +from typing import Dict, List +from cadCAD.engine import Executor, ExecutionContext, ExecutionMode +from cadCAD.configuration import Experiment +from cadCAD.configuration.utils import env_trigger, var_substep_trigger, config_sim, psub_list +from cadCAD.types import * +import pandas as pd # type: ignore +import types +import inspect +import pytest + +def describe_or_return(v: object) -> object: + """ + Thanks @LinuxIsCool! + """ + if isinstance(v, types.FunctionType): + return f'function: {v.__name__}' + elif isinstance(v, types.LambdaType) and v.__name__ == '': + return f'lambda: {inspect.signature(v)}' + else: + return v + + +def select_M_dict(M_dict: Dict[str, object], keys: set) -> Dict[str, object]: + """ + Thanks @LinuxIsCool! + """ + return {k: describe_or_return(v) for k, v in M_dict.items() if k in keys} + + +def select_config_M_dict(configs: list, i: int, keys: set) -> Dict[str, object]: + return select_M_dict(configs[i].sim_config['M'], keys) + + +def drop_substeps(_df): + first_ind = (_df.substep == 0) & (_df.timestep == 0) + last_ind = _df.substep == max(_df.substep) + inds_to_drop = first_ind | last_ind + return _df.copy().loc[inds_to_drop].drop(columns=['substep']) + + +def assign_params(_df: pd.DataFrame, configs) -> pd.DataFrame: + """ + Based on `cadCAD-tools` package codebase, by @danlessa + """ + M_dict = configs[0].sim_config['M'] + params_set = set(M_dict.keys()) + selected_params = params_set + + # Attribute parameters to each row + # 1. Assign the parameter set from the first row first, so that + # columns are created + first_param_dict = select_config_M_dict(configs, 0, selected_params) + + # 2. Attribute parameter on an (simulation, subset, run) basis + df = _df.assign(**first_param_dict).copy() + for i, (_, subset_df) in enumerate(df.groupby(['simulation', 'subset', 'run'])): + df.loc[subset_df.index] = subset_df.assign(**select_config_M_dict(configs, + i, + selected_params)) + return df + + + + +SWEEP_PARAMS: Dict[str, List] = { + 'alpha': [1], + 'beta': [lambda x: 2 * x, lambda x: x], + 'gamma': [3, 4], + 'omega': [7] + } + +SINGLE_PARAMS: Dict[str, object] = { + 'alpha': 1, + 'beta': lambda x: x, + 'gamma': 3, + 'omega': 5 + } + + +def create_experiment(N_RUNS=2, N_TIMESTEPS=3, params: dict=SWEEP_PARAMS): + psu_steps = ['m1', 'm2', 'm3'] + system_substeps = len(psu_steps) + var_timestep_trigger = var_substep_trigger([0, system_substeps]) + env_timestep_trigger = env_trigger(system_substeps) + env_process = {} + + + # ['s1', 's2', 's3', 's4'] + # Policies per Mechanism + def gamma(params: Parameters, substep: Substep, history: StateHistory, state: State, **kwargs): + return {'gamma': params['gamma']} + + + def omega(params: Parameters, substep: Substep, history: StateHistory, state: State, **kwarg): + return {'omega': params['omega']} + + + # Internal States per Mechanism + def alpha(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'alpha_var', params['alpha'] + + + def beta(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'beta_var', params['beta'] + + def gamma_var(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'gamma_var', params['gamma'] + + def omega_var(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'omega_var', params['omega'] + + + def policies(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'policies', _input + + + def sweeped(params: Parameters, substep: Substep, history: StateHistory, state: State, _input: PolicyOutput, **kwargs): + return 'sweeped', {'beta': params['beta'], 'gamma': params['gamma']} + + psu_block: dict = {k: {"policies": {}, "states": {}} for k in psu_steps} + for m in psu_steps: + psu_block[m]['policies']['gamma'] = gamma + psu_block[m]['policies']['omega'] = omega + psu_block[m]["states"]['alpha_var'] = alpha + psu_block[m]["states"]['beta_var'] = beta + psu_block[m]["states"]['gamma_var'] = gamma_var + psu_block[m]["states"]['omega_var'] = omega_var + psu_block[m]['states']['policies'] = policies + psu_block[m]["states"]['sweeped'] = var_timestep_trigger(y='sweeped', f=sweeped) + + + # Genesis States + genesis_states = { + 'alpha_var': 0, + 'beta_var': 0, + 'gamma_var': 0, + 'omega_var': 0, + 'policies': {}, + 'sweeped': {} + } + + # Environment Process + env_process['sweeped'] = env_timestep_trigger(trigger_field='timestep', trigger_vals=[5], funct_list=[lambda _g, x: _g['beta']]) + + sim_config = config_sim( + { + "N": N_RUNS, + "T": range(N_TIMESTEPS), + "M": params, # Optional + } + ) + + # New Convention + partial_state_update_blocks = psub_list(psu_block, psu_steps) + + exp = Experiment() + exp.append_model( + sim_configs=sim_config, + initial_state=genesis_states, + env_processes=env_process, + partial_state_update_blocks=partial_state_update_blocks + ) + return exp + + +def test_deepcopy_off(): + exp = create_experiment() + mode = ExecutionMode().local_mode + exec_context = ExecutionContext(mode, additional_objs={'deepcopy_off': True}) + executor = Executor(exec_context=exec_context, configs=exp.configs) + (records, tensor_field, _) = executor.execute() + df = drop_substeps(assign_params(pd.DataFrame(records), exp.configs)) + + # XXX: parameters should always be of the same type. Else, the test will fail + first_sim_config = exp.configs[0].sim_config['M'] + + + for (i, row) in df.iterrows(): + if row.timestep > 0: + + assert row['alpha_var'] == row['alpha'] + assert type(row['alpha_var']) == type(first_sim_config['alpha']) + assert row['gamma_var'] == row['gamma'] + assert type(row['gamma_var']) == type(first_sim_config['gamma']) + assert row['omega_var'] == row['omega'] + assert type(row['omega_var']) == type(first_sim_config['omega']) + diff --git a/testing/test_arg_count.py b/testing/test_arg_count.py new file mode 100644 index 00000000..e0ac71e6 --- /dev/null +++ b/testing/test_arg_count.py @@ -0,0 +1,51 @@ +from typing import Dict, List +from cadCAD.engine import Executor, ExecutionContext, ExecutionMode +from cadCAD.configuration import Experiment +from cadCAD.configuration.utils import env_trigger, var_substep_trigger, config_sim, psub_list +from cadCAD.types import * +import pandas as pd # type: ignore +import types +import inspect +import pytest + +def test_sufs(): + + psubs = [ + { + 'policies': { + 'p_A': lambda _1, _2, _3, _4: {} + }, + 'variables': { + 'v_A': lambda _1, _2, _3, _4, _5: ('v_a', None) + } + } + ] + + initial_state = { + 'v_A': None + } + + params = {'p_A': [1]} + + N_t = 5 + N_r = 1 + + sim_config = config_sim( + { + "N": N_r, + "T": range(N_t), + "M": params, # Optional + } + ) + + exp = Experiment() + exp.append_model( + sim_configs=sim_config, + initial_state=initial_state, + partial_state_update_blocks=psubs + ) + + mode = ExecutionMode().local_mode + exec_context = ExecutionContext(mode, additional_objs={'deepcopy_off': True}) + executor = Executor(exec_context=exec_context, configs=exp.configs) + (records, tensor_field, _) = executor.execute() From 546881a22d2ce6158f902e275de06ce928c9596d Mon Sep 17 00:00:00 2001 From: Danilo Lessa Bernardineli Date: Thu, 14 Dec 2023 20:46:57 -0300 Subject: [PATCH 6/6] Add recent contributors (#317) * fix tests + rm simulations/ folder * add types.py * single run / multi mc is ok * fix for single run / single param * add support for single proc runs * change contributor list to reflect the present state --------- Co-authored-by: Emanuel Lima --- AUTHORS.txt | 45 ++++++++++++++++++++++++++++----------------- cadCAD/types.py | 1 - setup.py | 4 ++-- 3 files changed, 30 insertions(+), 20 deletions(-) diff --git a/AUTHORS.txt b/AUTHORS.txt index 351daeaa..933b38b6 100644 --- a/AUTHORS.txt +++ b/AUTHORS.txt @@ -1,30 +1,41 @@ Authors ======= -cadCAD was implemented by Joshua E. Jodesty and designed by Michael Zargham, Markus B. Koch, and -Matthew V. Barlin from 2018 to 2020. +cadCAD was originally implemented by Joshua E. Jodesty and designed by Michael Zargham, Markus B. Koch, and +Matthew V. Barlin in 2018. Since then, upgrades and improvements were further committed by Danilo L. Bernardineli, Tyler D. Mace +and Emanuel Lima. + -Project Maintainers: -- Joshua E. Jodesty -- Markus B. Koch -- Michael Zargham +Current Project Maintainers: +- Emanuel Lima @emanuellima1 +- Michael Zargham @mzargham -Original Contributors: -- Joshua E. Jodesty -- Markus B. Koch -- Matthew V. Barlin -- Michael Zargham -- Zixuan Zhang -- Charles Rice +Active and Past Contributors: +- Joshua E. Jodesty @JEJodesty +- Markus B. Koch @markusbkoch +- Matthew V. Barlin @matttyb80 +- Michael Zargham @mzargham +- Danilo L. Bernardineli @danlessa +- Tyler D. Mace @tylerdmace +- Emanuel Lima @emanuellima1 +- Zixuan Zhang @zixuanzh +- Charles Rice @charles-rice +- Benjamin Scholz @BenSchZA We’d also like to thank: -- Andrew Clark -- Nick Hirannet -- Jonathan Gabler -- Harry Goodnight +- Andrew Clark @aclarkData +- Nick Hirannet @nick-phl-7 +- Jonathan Gabler +- Harry Goodnight @ - Charlie Hoppes - Nikhil Jamdade - Chris Frazier +- Griff Green @GriffGreen +- Isaac @eenti +- Jonny Dubowsky @jonnydubowsky +- Sem Brestels @sembrestels +- @fjribi +- Andrea Maria Piana @cammellos diff --git a/cadCAD/types.py b/cadCAD/types.py index 4f5de596..d2658e48 100644 --- a/cadCAD/types.py +++ b/cadCAD/types.py @@ -24,7 +24,6 @@ class ConfigurationDict(TypedDict): N: int # Number of MC Runs M: Union[Parameters, SweepableParameters] # Parameters / List of Parameter to Sweep - TargetValue = object EnvProcess: Callable[[State, SweepableParameters, TargetValue], TargetValue] EnvProcesses = dict[str, Callable] diff --git a/setup.py b/setup.py index fc8ae488..dfd786c4 100644 --- a/setup.py +++ b/setup.py @@ -28,8 +28,8 @@ description=short_description, long_description=long_description, url='https://github.com/cadCAD-org/cadCAD', - author='Joshua E. Jodesty', - author_email='joshua@block.science', + author='cadCAD-org Developers', + author_email='info@block.science', license='LICENSE.txt', packages=find_packages(), install_requires=[