From 850bcf1e6bffb794e9d1f74daafa5b850c907012 Mon Sep 17 00:00:00 2001 From: zhaojing Date: Tue, 24 Oct 2023 17:44:32 +0800 Subject: [PATCH] Add the implementation for the model selection example --- CMakeLists.txt | 6 +- LICENSE | 56 ++ .../Dockerfile | 58 ++ .../README.md | 148 ++++ .../documents/dev_guide.md | 236 ++++++ .../documents/image-20231020174425377.png | Bin 0 -> 302750 bytes .../documents/image-20231020174945226.png | Bin 0 -> 26176 bytes .../internal/cache-service/cache_service.py | 167 +++++ .../cache-service/trigger_cache_svc.py | 19 + .../internal/ml/model_selection/README.md | 271 +++++++ .../internal/ml/model_selection/config.ini | 86 +++ .../ml/model_selection/eva_service.py | 78 ++ .../model_selection/exp_result_singa/.gitkeep | 0 .../ml/model_selection/exps/README.md | 21 + .../ml/model_selection/exps/draw_img_lib.py | 706 ++++++++++++++++++ .../ml/model_selection/exps/draw_tab_lib.py | 197 +++++ .../nas_bench_tabular/0.train_one_model.py | 78 ++ .../2.seq_train_dist_online.py | 145 ++++ .../nas_bench_tabular/2.seq_train_online.py | 100 +++ .../nas_bench_tabular/4.seq_score_online.py | 116 +++ .../exps/nas_bench_tabular/measure_ecdf.py | 119 +++ .../nas_bench_tabular/measure_param_auc.py | 126 ++++ .../internal/ml/model_selection/init_env | 12 + .../internal/ml/model_selection/main.py | 60 ++ .../ml/model_selection/pg_interface.py | 617 +++++++++++++++ .../ml/model_selection/requirement.txt | 54 ++ .../scripts/anytime_img_w_baseline.sh | 42 ++ .../ml/model_selection/scripts/anytime_tab.sh | 126 ++++ .../scripts/baseline_system_img.sh | 45 ++ .../scripts/baseline_system_tab.sh | 67 ++ .../scripts/benchmark_weight_sharing.sh | 25 + .../scripts/database/load_data_to_db.sh | 67 ++ .../scripts/latency_embedding_cache.sh | 122 +++ .../latency_embedding_cache_concurrent.sh | 139 ++++ .../scripts/latency_phase1_cpu_gpu.sh | 211 ++++++ .../scripts/latency_phase1_in_db.sh | 61 ++ .../model_selection/scripts/latency_phase2.sh | 61 ++ .../scripts/micro_budget_aware_alg.sh | 44 ++ .../scripts/micro_nku_tradeoff.sh | 163 ++++ .../scripts/micro_score_metrics_relation.sh | 38 + .../scripts/micro_search_strategy.sh | 54 ++ .../nas-bench-img/convert_api_2_json.sh | 12 + .../nas-bench-img/explore_all_models.sh | 60 ++ .../scripts/nas-bench-img/score_all_models.sh | 60 ++ .../score_all_modesl_criteo.sh | 27 + .../score_all_modesl_frappe.sh | 27 + .../nas-bench-tabular/score_all_modesl_uci.sh | 28 + .../train_all_models_criteo.sh | 46 ++ .../train_all_models_criteo_distirbuted.sh | 48 ++ .../train_all_models_diabetes.sh | 47 ++ .../train_all_models_frappe.sh | 45 ++ .../nas-bench-tabular/train_one_model_dev.sh | 25 + .../train_params_tune_criteo.sh | 144 ++++ .../train_params_tune_diabetes.sh | 70 ++ .../pre_processing/pre_processing_data.sh | 47 ++ .../ml/model_selection/shared_config.py | 94 +++ .../ml/model_selection/src/__init__.py | 4 + .../ml/model_selection/src/common/constant.py | 64 ++ .../model_selection/src/common/structure.py | 89 +++ .../src/controller/__init__.py | 17 + .../src/controller/controler.py | 173 +++++ .../src/controller/core/__init__.py | 0 .../src/controller/core/metrics.py | 28 + .../src/controller/core/sample.py | 28 + .../src/controller/sampler_EA/__init__.py | 4 + .../src/controller/sampler_RL/__init__.py | 4 + .../src/controller/sampler_all/__init__.py | 4 + .../src/controller/sampler_all/seq_sampler.py | 32 + .../src/controller/sampler_bohb/bohb_or.py | 285 +++++++ .../controller/sampler_ea/regularized_ea.py | 130 ++++ .../src/controller/sampler_rand/__init__.py | 4 + .../controller/sampler_rand/random_sample.py | 22 + .../sampler_rl/reinforcement_learning.py | 48 ++ .../src/dataset_utils/__init__.py | 2 + .../download_critero_and_avazu.py | 42 ++ .../src/dataset_utils/sequence_dataloader.py | 78 ++ .../src/dataset_utils/stream_dataloader.py | 78 ++ .../src/eva_engine/__init__.py | 10 + .../src/eva_engine/coordinator.py | 98 +++ .../src/eva_engine/phase1/__init__.py | 4 + .../src/eva_engine/phase1/algo/__init__.py | 4 + .../eva_engine/phase1/algo/prune_synflow.py | 407 ++++++++++ .../eva_engine/phase1/algo/singa_ms/README.md | 2 + .../phase1/algo/singa_ms/cnn_ms/README.md | 46 ++ .../cnn_ms/autograd/cifar10_multiprocess.py | 43 ++ .../singa_ms/cnn_ms/autograd/mnist_cnn.py | 304 ++++++++ .../singa_ms/cnn_ms/autograd/mnist_dist.py | 25 + .../cnn_ms/autograd/mnist_multiprocess.py | 39 + .../cnn_ms/autograd/resnet_cifar10.py | 292 ++++++++ .../singa_ms/cnn_ms/autograd/resnet_dist.py | 87 +++ .../cnn_ms/autograd/sparsification_mnist.py | 45 ++ .../singa_ms/cnn_ms/autograd/xceptionnet.py | 303 ++++++++ .../phase1/algo/singa_ms/cnn_ms/benchmark.py | 121 +++ .../algo/singa_ms/cnn_ms/data/cifar10.py | 89 +++ .../algo/singa_ms/cnn_ms/data/cifar100.py | 81 ++ .../singa_ms/cnn_ms/data/download_cifar10.py | 49 ++ .../singa_ms/cnn_ms/data/download_cifar100.py | 26 + .../singa_ms/cnn_ms/data/download_mnist.py | 49 ++ .../phase1/algo/singa_ms/cnn_ms/data/mnist.py | 91 +++ .../algo/singa_ms/cnn_ms/model/alexnet.py | 119 +++ .../phase1/algo/singa_ms/cnn_ms/model/cnn.py | 90 +++ .../algo/singa_ms/cnn_ms/model/resnet.py | 300 ++++++++ .../algo/singa_ms/cnn_ms/model/xceptionnet.py | 311 ++++++++ .../singa_ms/cnn_ms/pkg_model_code/model.py | 357 +++++++++ .../phase1/algo/singa_ms/cnn_ms/run.sh | 38 + .../phase1/algo/singa_ms/cnn_ms/train_cnn.py | 564 ++++++++++++++ .../phase1/algo/singa_ms/cnn_ms/train_mpi.py | 91 +++ .../algo/singa_ms/cnn_ms/train_ms_model.py | 592 +++++++++++++++ .../singa_ms/cnn_ms/train_multiprocess.py | 111 +++ .../algo/singa_ms/ms_model_mlp/model.py | 226 ++++++ .../algo/singa_ms/ms_model_mlp/native.py | 137 ++++ .../phase1/algo/singa_ms/msmlp/model.py | 217 ++++++ .../phase1/algo/singa_ms/msmlp/native.py | 137 ++++ .../src/eva_engine/phase1/vote.py | 115 +++ .../src/eva_engine/phase2/__init__.py | 4 + .../src/eva_engine/phase2/algo/__init__.py | 4 + .../src/eva_engine/phase2/algo/trainer.py | 535 +++++++++++++ .../src/eva_engine/phase2/run_sr.py | 126 ++++ .../src/eva_engine/phase2/run_uniform.py | 78 ++ .../ml/model_selection/src/logger/__init__.py | 23 + .../model_selection/src/query_api/README.md | 2 + .../model_selection/src/query_api/__init__.py | 4 + .../src/query_api/img_explore_ea.py | 78 ++ .../src/query_api/img_train_baseline.py | 113 +++ .../src/query_api/interface.py | 123 +++ .../src/query_api/query_api_img.py | 277 +++++++ .../src/query_api/query_api_mlp.py | 145 ++++ .../src/query_api/singleton.py | 13 + .../src/search_space/__init__.py | 0 .../src/search_space/core/__init__.py | 4 + .../src/search_space/core/model_params.py | 23 + .../src/search_space/init_search_space.py | 37 + .../src/search_space/mlp_api/__init__.py | 4 + .../src/search_space/mlp_api/model_params.py | 16 + .../src/search_space/mlp_api/rl_policy.py | 18 + .../src/search_space/mlp_api/space.py | 625 ++++++++++++++++ .../src/search_space/utils/__init__.py | 4 + .../search_space/utils/weight_initializers.py | 78 ++ .../model_selection/src/third_pkg/__init__.py | 4 + .../src/third_pkg/darts_lib/__init__.py | 4 + .../src/third_pkg/darts_lib/genotypes.py | 18 + .../src/third_pkg/darts_lib/model.py | 291 ++++++++ .../src/third_pkg/darts_lib/util_convert.py | 109 +++ .../third_pkg/models/cell_infers/__init__.py | 5 + .../third_pkg/models/cell_searchs/__init__.py | 33 + .../models/cell_searchs/genotypes.py | 280 +++++++ .../third_pkg/models/shape_infers/__init__.py | 9 + .../models/shape_infers/shared_utils.py | 5 + .../models/shape_searchs/__init__.py | 9 + .../src/third_pkg/sp101_lib/__init__.py | 4 + .../src/third_pkg/sp101_lib/graph_util.py | 168 +++++ .../src/third_pkg/sp101_lib/model_spec.py | 325 ++++++++ .../src/third_pkg/sp101_lib/nb101_api.py | 465 ++++++++++++ .../src/third_pkg/sp201_lib/__init__.py | 42 ++ .../src/third_pkg/sp201_lib/nasbench2.py | 117 +++ .../src/third_pkg/sp201_lib/nasbench2_ops.py | 160 ++++ .../ml/model_selection/src/tools/__init__.py | 2 + .../ml/model_selection/src/tools/compute.py | 121 +++ .../model_selection/src/tools/correlation.py | 115 +++ .../ml/model_selection/src/tools/io_tools.py | 42 ++ .../model_selection/src/tools/res_measure.py | 84 +++ .../ml/model_selection/src/tools/utils.py | 454 +++++++++++ .../internal/ml/model_slicing/.gitkeep | 0 .../internal/pg_extension/.cargo/config.toml | 3 + .../internal/pg_extension/.gitignore | 6 + .../internal/pg_extension/Cargo.toml | 39 + .../pg_extension/pg_extension.control | 5 + .../pg_extension/sql/filter_phase.sql | 35 + .../pg_extension/sql/model_selection_cpu.sql | 31 + .../sql/model_selection_cpu_workloads.sql | 32 + .../pg_extension/sql/model_selection_dev.sql | 80 ++ .../sql/model_selection_trails.sql | 31 + .../sql/model_selection_trails_workloads.sql | 32 + .../pg_extension/sql/pg_extension--0.1.0.sql | 139 ++++ .../pg_extension/src/bindings/ml_register.rs | 55 ++ .../internal/pg_extension/src/bindings/mod.rs | 7 + .../pg_extension/src/bindings/model.rs | 19 + .../internal/pg_extension/src/bindings/ms.rs | 207 +++++ .../internal/pg_extension/src/lib.rs | 171 +++++ .../internal/pg_extension/test/lib.rs | 0 .../requirement.txt | 32 + setup.py | 2 +- tool/conda/singa/meta.yaml | 2 +- .../centos6/cuda10/Dockerfile.manylinux2014 | 4 +- 184 files changed, 18402 insertions(+), 7 deletions(-) create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/Dockerfile create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/README.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/dev_guide.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/image-20231020174425377.png create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/image-20231020174945226.png create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/cache_service.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/trigger_cache_svc.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/README.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/config.ini create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/eva_service.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exp_result_singa/.gitkeep create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/README.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_img_lib.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_tab_lib.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_dist_online.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_ecdf.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_param_auc.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/init_env create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/main.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/pg_interface.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/requirement.txt create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_img_w_baseline.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_tab.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_img.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_tab.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/benchmark_weight_sharing.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/database/load_data_to_db.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache_concurrent.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_cpu_gpu.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_in_db.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase2.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_budget_aware_alg.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_nku_tradeoff.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_score_metrics_relation.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_search_strategy.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/convert_api_2_json.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/explore_all_models.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/score_all_models.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_criteo.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_frappe.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_uci.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo_distirbuted.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_diabetes.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_frappe.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_one_model_dev.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_criteo.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_diabetes.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/pre_processing/pre_processing_data.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/shared_config.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/constant.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/structure.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/controler.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/metrics.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/sample.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_EA/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_RL/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/seq_sampler.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_bohb/bohb_or.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_ea/regularized_ea.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/random_sample.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rl/reinforcement_learning.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/download_critero_and_avazu.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/sequence_dataloader.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/stream_dataloader.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/coordinator.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/prune_synflow.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/README.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/README.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/cifar10_multiprocess.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_cnn.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_dist.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_multiprocess.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_cifar10.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_dist.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/sparsification_mnist.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/xceptionnet.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/benchmark.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar10.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar100.py create mode 100755 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar10.py create mode 100755 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar100.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_mnist.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/mnist.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/alexnet.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/cnn.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/resnet.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/xceptionnet.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/pkg_model_code/model.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/run.sh create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_cnn.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_mpi.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_ms_model.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_multiprocess.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/model.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/native.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/model.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/native.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/vote.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/trainer.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_sr.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_uniform.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/logger/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/README.md create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_explore_ea.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_train_baseline.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/interface.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_img.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_mlp.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/singleton.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/model_params.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/init_search_space.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/model_params.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/rl_policy.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/space.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/weight_initializers.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/genotypes.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/model.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/util_convert.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_infers/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/genotypes.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/shared_utils.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_searchs/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/graph_util.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/model_spec.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/nb101_api.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2_ops.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/__init__.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/compute.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/correlation.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/io_tools.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/res_measure.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/utils.py create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_slicing/.gitkeep create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.cargo/config.toml create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.gitignore create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/Cargo.toml create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/pg_extension.control create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/filter_phase.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu_workloads.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_dev.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails_workloads.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/pg_extension--0.1.0.sql create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ml_register.rs create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/mod.rs create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/model.rs create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ms.rs create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/lib.rs create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/test/lib.rs create mode 100644 examples/model_selection/TRAILS-Database-Native-Model-Selection/requirement.txt diff --git a/CMakeLists.txt b/CMakeLists.txt index 9f49efa692..50e67f970f 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -29,10 +29,10 @@ LIST(APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake/Thirdparty) #string(REGEX REPLACE "^[0-9]+\\.[0-9]+\\.([0-9]+).*" "\\1" VERSION_PATCH "${VERSION}") -SET(PACKAGE_VERSION 4.0.0) # ${VERSION}) -SET(VERSION 4.0.0) +SET(PACKAGE_VERSION 4.1.0) # ${VERSION}) +SET(VERSION 4.1.0) SET(SINGA_MAJOR_VERSION 4) -SET(SINGA_MINOR_VERSION 0) +SET(SINGA_MINOR_VERSION 1) SET(SINGA_PATCH_VERSION 0) #SET(SINGA_MAJOR_VERSION ${VERSION_MAJOR}) # 0 - #SET(SINGA_MINOR_VERSION ${VERSION_MINOR}) # 0 - 9 diff --git a/LICENSE b/LICENSE index f448e8b2b8..9c7ffb4751 100644 --- a/LICENSE +++ b/LICENSE @@ -559,3 +559,59 @@ PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +=============================================================================== +SINGA bundles the following under MIT License: +examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/* + +MIT License + +Portions Copyright 2019-2021 ZomboDB, LLC. +Portions Copyright 2021-2023 Technology Concepts & Design, Inc. +Portions Copyright 2023 PgCentral Foundation, Inc. + +All rights reserved. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +=============================================================================== +SINGA bundles the following under The PostgreSQL License: +examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/* + +The PostgreSQL License + +Portions Copyright (c) 1996-2023, The PostgreSQL Global Development Group + +Portions Copyright (c) 1994, The Regents of the University of California + +Permission to use, copy, modify, and distribute this software and its documentation for any +purpose, without fee, and without a written agreement is hereby granted, provided that the above +copyright notice and this paragraph and the following two paragraphs appear in all copies. + +IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, +SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING +OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF +THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, +BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, +AND THE UNIVERSITY OF CALIFORNIA HAS NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, +UPDATES, ENHANCEMENTS, OR MODIFICATIONS. + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/Dockerfile b/examples/model_selection/TRAILS-Database-Native-Model-Selection/Dockerfile new file mode 100644 index 0000000000..cd622e2d7f --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/Dockerfile @@ -0,0 +1,58 @@ +FROM ubuntu:20.04 + +ENV DEBIAN_FRONTEND=noninteractive + +# Install Python, Vim, and necessary libraries +RUN apt-get update && \ + apt-get install -y software-properties-common wget gnupg2 lsb-release git && \ + add-apt-repository ppa:deadsnakes/ppa && \ + apt-get install -y python3.6 python3-pip vim && \ + apt-get clean && \ + rm -rf /var/lib/apt/lists/* + +# Install necessary dependencies for PostgreSQL and Rust +RUN apt-get update && \ + apt-get install -y pkg-config libssl-dev libpq-dev libclang-dev curl && \ + apt-get clean && \ + rm -rf /var/lib/apt/lists/* + +# Install necessary dependencies for pgrx +RUN apt-get update && \ + apt-get install -y bison flex libreadline-dev && \ + apt-get clean && \ + rm -rf /var/lib/apt/lists/* + +# Create the postgres user +USER root +RUN adduser --disabled-password --gecos "" postgres && \ + mkdir /project && \ + adduser postgres sudo && \ + chown -R postgres:postgres /project + +# Switch to the postgres user andInstall Rust and init the cargo +USER postgres +RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y && \ + echo 'source $HOME/.cargo/env' >> $HOME/.bashrc && \ + /bin/bash -c "source $HOME/.cargo/env && cargo install cargo-pgrx --version '0.9.7' --locked" && \ + /bin/bash -c "source $HOME/.cargo/env && cargo pgrx init" + +# Set environment variables for Rust and Python +ENV PATH="/root/.cargo/bin:${PATH}" +ENV PYTHONPATH="${PYTHONPATH}:/project/TRAILS/internal/ml/model_selection" + +WORKDIR /project +COPY ./internal/ml/model_selection/requirement.txt ./requirement.txt +RUN pip install -r requirement.txt + +RUN pip install https://www.comp.nus.edu.sg/~zhaojing/files/singa-3.1.0-cp38-cp38-manylinux2014_x86_64.whl + +# appendix +USER root +RUN apt-get update && apt-get install -y \ + postgresql-client && \ + apt-get clean && \ + rm -rf /var/lib/apt/lists/* + +USER postgres + +CMD ["tail", "-f", "/dev/null"] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/README.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/README.md new file mode 100644 index 0000000000..cbc9749cf9 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/README.md @@ -0,0 +1,148 @@ +# Database-Native Model Selection + +​ -- based on Singa + + + +![image-20231020174425377](documents/image-20231020174425377.png) + +## Build Docker Image + +```bash +git clone https://github.com/apache/singa.git +cd singa/examples/model_selection/TRAILS-Database-Native-Model-Selection/ +docker build -t trails-singa . +``` + +## Run Docker Image +Download exp_data.zip from https://www.dropbox.com/scl/fi/xz4teosklwmfc5j4x2ug6/exp_data.zip?rlkey=5fk2ttib0zt49suyppcjhsrn2&dl=0 +and unzip the exp_data/ folder to a specific directory (path_to_exp_data_folder) +```bash +docker run -d --name trails-singa \ + --network="host" \ + -v path_to_exp_data_folder:/project/exp_data \ + trails-singa +``` + +## Start PostgreSQL Instance + +```bash +# 1. Run docker container +docker exec -it trails-singa bash +# 2. Clone the code +cd ~ +git clone https://github.com/apache/singa.git +cd singa/examples/model_selection/TRAILS-Database-Native-Model-Selection/ +# 3. Export PYTHONPATH +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +# 4. Start the RDBMS and then exit +cd internal/pg_extension +cargo pgrx run +exit +cd ../.. +# 5. Load data into RDBMS +bash internal/ml/model_selection/scripts/database/load_data_to_db.sh /project/exp_data/data/structure_data/frappe frappe +# 6. Run database server +cd internal/pg_extension +cargo pgrx run + +``` + + +## Register Stored Procedure + +```sql +CREATE OR REPLACE +PROCEDURE model_selection_sp( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + N INTEGER, --number of models to evaluate + batch_size INTEGER, --batch size, for profiling, filtering + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + + -- 4. Run filtering phase to get top K models. + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + LIMIT %s OFFSET 0 + ) + SELECT filtering_phase( + json_agg(row_to_json(t))::text, %s, %s, %L + ) + FROM batch_rows AS t', column_list, dataset, batch_size, N, 1, config_file) INTO result_status; + RAISE NOTICE '4. run filtering phase, k models = %', result_status; + +END; $$; +``` + +# Compile the UDF + +```bash +# Try compile the UDF +DROP EXTENSION IF EXISTS pg_extension; +CREATE EXTENSION pg_extension; +``` + +If the above fails, open another terminal and go into the docker via docker exec -it trails-singa bash +Then run the following +```bash +rm /home/postgres/.pgrx/14.9/pgrx-install/share/extension/pg_extension--0.1.0.sql +vi /home/postgres/.pgrx/14.9/pgrx-install/share/extension/pg_extension--0.1.0.sql +# Copy the following to the /home/postgres/.pgrx/14.9/pgrx-install/share/extension/pg_extension--0.1.0.sql +-- src/lib.rs:66 +-- pg_extension::filtering_phase +CREATE FUNCTION "filtering_phase"( + "mini_batch" TEXT, /* alloc::string::String */ + "n" INT, /* i32 */ + "k" INT, /* i32 */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'filtering_phase_wrapper'; +``` + +Go back to the first terminal and run the following in the database server again +```bash +# Try compile the UDF +DROP EXTENSION IF EXISTS pg_extension; +CREATE EXTENSION pg_extension; +``` + +## Run Model Selection + +```sql +-- Template for calling 'model_selection_sp' stored procedure +CALL model_selection_sp( + , -- The name of the table or dataset from which data should be retrieved. + , -- An array of column names to be considered in the model selection process. + , -- Number of models to explore + , -- Batch size + -- The file path to a configuration file needed for the process. +); + + +# For example +CALL model_selection_sp( + 'frappe_train', + ARRAY['col1', 'col2', 'col3', 'label'], + 10, + 32, + '/home/postgres/singa/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/config.ini'); +``` + +# Example Result + +![image-20231020174945226](documents/image-20231020174945226.png) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/dev_guide.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/dev_guide.md new file mode 100644 index 0000000000..abb283768b --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/dev_guide.md @@ -0,0 +1,236 @@ + + + + +# Change the permission + +```bash +chmod -R 777 internal/pg_extension +chmod -R 777 TRAILS +``` + +# PSQL CMD + +```sql +psql -h localhost -p 28814 -U postgres +\c frappe +\dt +\d frappe_train +DROP TABLE frappe_train; +SELECT * FROM frappe_train LIMIT 10; +SELECT * FROM frappe_test LIMIT 10; +SELECT * FROM frappe_valid LIMIT 10; +DROP DATABASE frappe; +psql -U postgres +``` + +# Build and run the container + +```bash +docker build -t trails . + +docker run -d --name trails \ + --network="host" \ + -v $(pwd)/TRAILS:/project/TRAILS \ + -v /hdd1/xingnaili/exp_data/:/project/exp_data \ + trails + +docker exec -it trails bash +``` + +# This is in docker image already + +```bash +# if those are already on docker, skip them. +cargo install --locked cargo-pgrx +# run after package update +cargo pgrx init +cargo pgrx new my_extension +# just run this after code updates. +cargo pgrx run +``` + +# Develop + +## Load data into database. + +```bash +bash /project/TRAILS/internal/ml/model_selection/scripts/database/load_data_to_db.sh /project/exp_data/data/structure_data/frappe frappe +bash /project/TRAILS/internal/ml/model_selection/scripts/database/load_data_to_db.sh /project/exp_data/data/structure_data/uci_diabetes uci_diabetes +bash /project/TRAILS/internal/ml/model_selection/scripts/database/load_data_to_db.sh /project/exp_data/data/structure_data/criteo_full criteo +``` + +## 1. Compile + +In shell + +```bash +cd ./internal/pg_extension/ +cargo clean +rm -r /home/postgres/.pgrx/14.9/pgrx-install/lib/pg_extension.so +cargo pgrx run +rm /home/postgres/.pgrx/14.9/pgrx-install/share/extension/pg_extension--0.1.0.sql +vi /home/postgres/.pgrx/14.9/pgrx-install/share/extension/pg_extension--0.1.0.sql +paste the latest sqls +# generate schema +cargo pgrx schema >> /home/postgres/.pgrx/14.9/pgrx-install/share/extension/pg_extension--0.1.0.sql +``` + +In SQL + +```sql +DROP EXTENSION IF EXISTS pg_extension; +CREATE EXTENSION pg_extension; +``` + +## 2. Edit the config file + +Update the `nfield` in the `config.ini` file, it is == number of columns used. E.g, `ARRAY['col1', 'col2', 'col3', 'label']` => `nfield` = 3 + +## 3. Run it + +```sql +CREATE EXTENSION pg_extension; + +# Test if the UDF is there or not +SELECT * FROM pg_proc WHERE proname = 'model_selection_workloads'; + +# micro +select benchmark_filtering_phase_latency(4, '/project/TRAILS/internal/ml/model_selection/config.ini'); + +select benchmark_filtering_latency_in_db(5000, 'frappe', '/project/TRAILS/internal/ml/model_selection/config.ini'); + +select benchmark_filtering_latency_in_db(5000, 'uci_diabetes', '/project/TRAILS/internal/ml/model_selection/config.ini'); + +select benchmark_filtering_latency_in_db(4, 'criteo', '/project/TRAILS/internal/ml/model_selection/config.ini'); + +# Test coordinator +SELECT coordinator('0.08244', '168.830156', '800', false, '/project/TRAILS/internal/ml/model_selection/config.ini'); + +# this is database name, columns used, time budget, batch size, and config file +CALL model_selection_sp('dummy', ARRAY['col1', 'col2', 'col3', 'label'], '30', 32, '/project/TRAILS/internal/ml/model_selection/config.ini'); + +# end2end model selection +CALL model_selection_end2end('dummy', ARRAY['col1', 'col2', 'col3', 'label'], '15', '/project/TRAILS/internal/ml/model_selection/config.ini'); + +# filtering & refinement with workloads +CALL model_selection_workloads('dummy', ARRAY['col1', 'col2', 'col3', 'label'], 300, 3, '/project/TRAILS/internal/ml/model_selection/config.ini'); + +response = requests.post(args.refinement_url, json=data).json() + +``` + +# Test the pg-extension works using pipython + +```sql +# switch to a postgres +su postgres + +CREATE EXTENSION plpython3u; + +CREATE FUNCTION py_version() RETURNS text AS $$ +import sys +return sys.version +$$ LANGUAGE plpython3u; + +SELECT py_version(); + +CREATE OR REPLACE FUNCTION test_numpy() + RETURNS text +LANGUAGE plpython3u +AS $$ +import numpy +import torch +import sklearn +import torchvision +import tqdm +print("asdf") +return str(numpy.__version__) + " torch: " + str(torch.__version__) +$$; + +SELECT test_numpy(); + +CREATE EXTENSION my_extension; +SELECT hello_my_extension(); +``` + +# Container log + +Each line in your output represents a different process that is currently running on your PostgreSQL server. Here's what each one is doing: + +1. `/bin/sh -c service postgresql start && tail -F /var/log/postgresql/postgresq` : This is the command that was used to start your PostgreSQL server. It also includes a command to continuously display new entries from the PostgreSQL log file. + + +2. `/usr/lib/postgresql/14/bin/postgres -D /var/lib/postgresql/14/main -c config` : This is the main PostgreSQL process. All other PostgreSQL processes are children of this process. + + +3. `postgres: 14/main: checkpointer` : The checkpointer process is responsible for making sure data changes get saved to disk regularly. This is important for database recovery in case of a crash. + + +4. `postgres: 14/main: background writer` : The background writer process is responsible for writing buffers to disk when they become dirty. This reduces the amount of work that needs to be done when a buffer is reused. + + +5. `postgres: 14/main: walwriter` : The walwriter process writes transaction logs (Write-Ahead Logs or WAL) to disk. This is also important for database recovery and replication. + + +6. `postgres: 14/main: autovacuum launcher` : The autovacuum launcher process starts autovacuum worker processes as needed. These processes automatically clean up and optimize the database. + + +7. `postgres: 14/main: stats collector` : The stats collector process collects statistics about the server's activity. This information can be viewed using the `pg_stat` family of system views. + + +8. `postgres: 14/main: logical replication launcher` : The logical replication launcher manages the worker processes that perform logical replication, copying data changes to other databases. + + +9. `tail -F /var/log/postgresql/postgresql-14-main.log` : This process is displaying the end of the PostgreSQL log file and updating as more entries are added. + + +10. `bash` : These are shell sessions, likely interactive ones you've started. + + +11. `/usr/lib/postgresql/14/bin/psql -h localhost -p 28814 pg_extension` : These are instances of the psql command line interface, connected to your database. + + +12. `postgres: postgres pg_extension 127.0.0.1(52236) CALL` : This is your currently running stored procedure. + + +13. `ps aux` : This is the command you ran to display the list of processes. + +Each process is part of the PostgreSQL database system and helps it to run efficiently and robustly. + +# MAC locally + +```bash +conda activate firmest38 +export PYTHON_SYS_EXECUTABLE=/Users/kevin/opt/anaconda3/envs/firmest38/bin/python +export DYLD_LIBRARY_PATH=/Users/kevin/opt/anaconda3/envs/firmest38/lib/:$DYLD_LIBRARY_PATH +cargo run --features python +``` + +# What cargo run do? + +Before: + +``` +postgres 1 0.1 0.0 2612 588 ? Ss 14:30 0:00 /bin/sh -c service postgresql start && tail -F /var/log/postgresql/postgresql-14-main.log +postgres 20 0.1 0.0 214688 29332 ? Ss 14:30 0:00 /usr/lib/postgresql/14/bin/postgres -D /var/lib/postgresql/14/main -c config_file=/etc/postgresql/14/main/postgresql.conf +postgres 22 0.0 0.0 214688 6120 ? Ss 14:30 0:00 postgres: 14/main: checkpointer +postgres 23 0.0 0.0 214688 6084 ? Ss 14:30 0:00 postgres: 14/main: background writer +postgres 24 0.0 0.0 214688 10352 ? Ss 14:30 0:00 postgres: 14/main: walwriter +postgres 25 0.0 0.0 215224 8864 ? Ss 14:30 0:00 postgres: 14/main: autovacuum launcher +postgres 26 0.0 0.0 69280 5184 ? Ss 14:30 0:00 postgres: 14/main: stats collector +postgres 27 0.0 0.0 215236 6972 ? Ss 14:30 0:00 postgres: 14/main: logical replication launcher +postgres 38 0.0 0.0 2548 512 ? S 14:30 0:00 tail -F /var/log/postgresql/postgresql-14-main.log +postgres 39 0.1 0.0 4112 3424 pts/0 Ss+ 14:30 0:00 bash +postgres 48 0.1 0.0 4112 3424 pts/1 Ss 14:30 0:00 bash +postgres 59 0.0 0.0 5896 2860 pts/1 R+ 14:30 0:00 ps aux +``` + +After: + + + + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/image-20231020174425377.png b/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/image-20231020174425377.png new file mode 100644 index 0000000000000000000000000000000000000000..9e73b270d9e74c304b4b493ced7a0e39ac235824 GIT binary patch literal 302750 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%0D?k{6Mb|ad$ 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
ww^=KUTW(&iU6qaM#?2c?49)0vPkH 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�iCQ0$q#4>@bRn+4BpP>qH;zIH7pO{%cVyc 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
    9oVGzPTzfD} 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 literal 0 HcmV?d00001 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/image-20231020174945226.png b/examples/model_selection/TRAILS-Database-Native-Model-Selection/documents/image-20231020174945226.png new file mode 100644 index 0000000000000000000000000000000000000000..d7b686d2ba71e563f2007feb6e80b2b99e91fcf8 GIT binary patch literal 26176 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 literal 0 HcmV?d00001 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/cache_service.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/cache_service.py new file mode 100644 index 0000000000..7c6ce826e8 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/cache_service.py @@ -0,0 +1,167 @@ +import time +import threading +import queue +import psycopg2 +from typing import Any, List, Dict, Tuple +from sanic import Sanic +from sanic.response import json +import calendar +import os +import logging + +log_logger_folder_name = "log_cache_service" +if not os.path.exists(f"./{log_logger_folder_name}"): + os.makedirs(f"./{log_logger_folder_name}") +logger = logging.getLogger(__name__) +logging.basicConfig(level=logging.DEBUG, + format='%(asctime)s %(levelname)-8s %(message)s', + datefmt='%d %b %Y %H:%M:%S', + filename=f"./{log_logger_folder_name}/log_{str(calendar.timegm(time.gmtime()))}", filemode='w') + +USER = "postgres" +HOST = "127.0.0.1" +PORT = "28814" +DB_NAME = "pg_extension" +CACHE_SIZE = 10 + + +class CacheService: + def __init__(self, name_space: str, database: str, table: str, columns: List, batch_size: int, max_size: int = CACHE_SIZE): + """ + name_space: train, valid, test + database: database to use + table: which table + columns: selected cols + max_size: max batches to cache + """ + self.name_space = name_space + self.batch_size = batch_size + self.last_id = -1 + self.database = database + self.table = table + self.columns = columns + self.queue = queue.Queue(maxsize=max_size) + self.thread = threading.Thread(target=self.fetch_data, daemon=True) + self.thread.start() + + def decode_libsvm(self, columns): + map_func = lambda pair: (int(pair[0]), float(pair[1])) + # 0 is id, 1 is label + id, value = zip(*map(lambda col: map_func(col.split(':')), columns[2:])) + sample = {'id': list(id), + 'value': list(value), + 'y': int(columns[1])} + return sample + + def pre_processing(self, mini_batch_data: List[Tuple]): + """ + mini_batch_data: [('0', '0', '123:123', '123:123', '123:123',) + """ + sample_lines = len(mini_batch_data) + feat_id = [] + feat_value = [] + y = [] + + for i in range(sample_lines): + row_value = mini_batch_data[i] + sample = self.decode_libsvm(row_value) + feat_id.append(sample['id']) + feat_value.append(sample['value']) + y.append(sample['y']) + return {'id': feat_id, 'value': feat_value, 'y': y} + + def fetch_data(self): + with psycopg2.connect(database=self.database, user=USER, host=HOST, port=PORT) as conn: + while True: + try: + # fetch and preprocess data from PostgreSQL + batch, time_usg = self.fetch_and_preprocess(conn) + self.queue.put(batch) + print(f"Data is fetched, {self.name_space} queue_size={self.queue.qsize()}, time_usg={time_usg}") + logger.info(f"Data is fetched, queue_size={self.queue.qsize()}, time_usg={time_usg}") + # block until a free slot is available + time.sleep(0.1) + except psycopg2.OperationalError: + logger.exception("Lost connection to the database, trying to reconnect...") + time.sleep(5) # wait before trying to establish a new connection + conn = psycopg2.connect(database=self.database, user=USER, host=HOST, port=PORT) + + def fetch_and_preprocess(self, conn): + begin_time = time.time() + cur = conn.cursor() + # Assuming you want to get the latest 100 rows + columns_str = ', '.join(self.columns) + # Select rows greater than last_id + cur.execute(f"SELECT id, {columns_str} FROM {self.table} " + f"WHERE id > {self.last_id} ORDER BY id ASC LIMIT {self.batch_size}") + rows = cur.fetchall() + + if rows: + # Update last_id with max id of fetched rows + self.last_id = max(row[0] for row in rows) # assuming 'id' is at index 0 + else: + # If no more new rows, reset last_id to start over scan and return 'end_position' + self.last_id = -1 + return "end_position", time.time() - begin_time + + batch = self.pre_processing(rows) + return batch, time.time() - begin_time + + def get(self): + return self.queue.get() + + def is_empty(self): + return self.queue.empty() + + +app = Sanic("CacheServiceApp") + + +# start the server, this is from pg_ingerface +@app.route("/", methods=["POST"]) +async def start_service(request): + try: + columns = request.json.get('columns') + # can only be train or valid + name_space = request.json.get('name_space') + table_name = request.json.get('table_name') + batch_size = request.json.get('batch_size') + + if columns is None: + return json({"error": "No columns specified"}, status=400) + if name_space not in ["train", "valid", "test"]: + return json({"error": name_space + " is not correct"}, status=400) + + print(f"columns are {columns}, name_space = {name_space}") + + if not hasattr(app.ctx, f'{table_name}_{name_space}_cache'): + setattr(app.ctx, f'{table_name}_{name_space}_cache', + CacheService(name_space, DB_NAME, table_name, columns, batch_size, CACHE_SIZE)) + + return json("OK") + except Exception as e: + return json({"error": str(e)}, status=500) + + +# serve the data retrieve request from eva_service.py +@app.route("/", methods=["GET"]) +async def serve_get_request(request): + name_space = request.args.get('name_space') + table_name = request.args.get('table_name') + + # check if exist + if not hasattr(app.ctx, f'{table_name}_{name_space}_cache'): + return json({"error": f"{table_name}_{name_space}_cache not start yet"}, status=404) + + # get data + data = getattr(app.ctx, f'{table_name}_{name_space}_cache').get() + + # return + if data is None: + return json({"error": "No data available"}, status=404) + else: + return json(data) + + +if __name__ == "__main__": + app.run(host="0.0.0.0", port=8093) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/trigger_cache_svc.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/trigger_cache_svc.py new file mode 100644 index 0000000000..45cecdf730 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/cache-service/trigger_cache_svc.py @@ -0,0 +1,19 @@ + +import requests + +url = 'http://localhost:8093/' +columns = ['label', 'col1', 'col2', 'col3', 'col4', 'col5', 'col6', 'col7', 'col8', 'col9', 'col10'] +response = requests.post( + url, json={'columns': columns, + 'name_space': "train", + 'table_name': "frappe_train", + "batch_size": 32}) +print(response.json()) + +response = requests.post( + url, json={'columns': columns, + 'name_space': "valid", + 'table_name': "frappe_valid", + "batch_size": 1024}) +print(response.json()) + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/README.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/README.md new file mode 100644 index 0000000000..9b9b7b5635 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/README.md @@ -0,0 +1,271 @@ +# TRAILS: A Database Native Model Selection System + +![image-20230702035806963](documents/imgs/image-20230702035806963.png) + +[TOC] + +# Config Environments + +```bash +# Create virtual env +conda config --set ssl_verify false +conda create -n "trails" python=3.8.10 +conda activate trails +pip install -r requirement.txt + +cd TRAILS + +# make a dir to store all results. +mkdir ../exp_data +``` + +# Reproduce the results + +## NAS-Bench-Tabular + + NAS-Bench-Tabular can be either **download** or build from scratch. + +### Download NAS-Bench-Tabular + +1. **Download** the dataset using the following link, and extract them to `exp_data` + +```bash +https://drive.google.com/file/d/1TGii9ymbmX81c9-GKWXbe_4Z64R8Btz1/view?usp=sharing +``` + +### Build NAS-Bench-Tabular + +2. Build the **NAS-Bench-Tabular** from scratch + +```python +# Construct NAS-Bench-Tabular: +## 1. Training all models. +bash internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_frappe.sh +bash internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_diabetes.sh +bash internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo.sh + +## 2. Scoring all models using all TFMEMs. +bash internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_frappe.sh +bash internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_uci.sh +bash internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_criteo.sh +``` + +3. Build the **NAS-Bench-Img** from scratch + + To facilitate the experiments and query speed (NASBENCH API is slow) + + 1. We retrieve all results from NASBENCH API and store them as a json file. + 2. We score all models in NB201 and 28K models in NB101. + 3. We search with EA + Score and record the searching process in terms of + `run_id, current_explored_model, top_400 highest scored model, time_usage` + to SQLLite. + +```python +# 1. Record NASBENCH API data into json file +## This requires to install nats_bench: pip install nats_bench +bash ./internal/ml/model_selection/scripts/nas-bench-img/convert_api_2_json.sh + +# 2. Scoring all models using all TFMEMs. +nohup bash ./internal/ml/model_selection/scripts/nas-bench-img/score_all_models.sh & + +# 3. Explore with EA ans score result and store exploring process into SQLLite +bash ./internal/ml/model_selection/scripts/nas-bench-img/explore_all_models.sh + +# 4. Generate the baseline. +bash ./internal/ml/model_selection/scripts/baseline_system_img.sh +``` + +The following experiment could then query filtering phase results based on `run_id`. + +## SLO-Aware 2Phase-MS + +With the above **NAS-Bench-Tabular**, we could run various experiments. + +```bash +# 1. Generate the results for drawing the figure +## tabular data: training-base-ms +bash internal/ml/model_selection/scripts/baseline_system_tab.sh +## tabular data: training-free-ms, 2phase-ms +nohup bash internal/ml/model_selection/scripts/anytime_tab.sh & +## image data: training-base-ms, training-free-ms, 2phase-ms +nohup bash internal/ml/model_selection/scripts/anytime_img_w_baseline.sh & + +# 2. Draw figure +python internal/ml/model_selection/exps/macro/anytime_tab_draw.py +python internal/ml/model_selection/exps/macro/anytime_img_draw.py +``` + +![image-20230702035554579](documents/imgs/image-20230702035554579.png) + +## Micro: Benchmark TFMEMs + +```bash +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails +python ./internal/ml/model_selection/exps/micro/benchmark_correlation.py +``` + +![image-20230421214835152](./documents/imgs/image-20230421214835152.png) + +## Micro: Benchmark Budge-Aware Algorithm + +```bash +bash internal/ml/model_selection/scripts/micro_budget_aware_alg.sh +``` + +![image-20230724111659545](./documents/imgs/image-20230724111659545.png) + +## Micro: Benchmark N, K, U + +With ranking the models by ther TFMEM score in the filtering phase, we aim to determine + +1. Further examinng more models (**K**) with each going through less training epoch (**U**) is more easier to find good model? + or examine less but each training more epochs? +2. How many models to explore (**N**) and how many to keep (**K**) ? + +```bash +bash internal/ml/model_selection/scripts/micro_nku_tradeoff.sh +``` + +This is the experimental result conducted at the UCI Diabetes datasets. +Clearly, expore more models in refinement phase (large **K** ) is more helpful to find the a better model. +Although increasing **U** can find a better model accurately, it runs more training epochs leading to higher training cost. + +![image-20230722202555763](./documents/imgs/image-20230722202555763.png) + +Then we fix **U=1** for cost efficiency and determine N/K for higher searching effectiveness. +Clearly, K/N reaches 100 yields better scheduling result in both image and tabular dataset, thus, we set **N/K=100** in coordinator. + +![image-20230724111325368](./documents/imgs/image-20230724111325368.png) + +![image-20230722205244718](./documents/imgs/image-20230722205244718.png) + +## Micro: Device Placement & Embedding Cache + +1. To measure the time usage for filtering phase on vairous hardware, run the following + + ```bash + # Without embedding cache at the filtering phase + nohup bash internal/ml/model_selection/scripts/latency_phase1_cpu_gpu.sh & + # With embedding cache at the filtering phase (faster) + nohup bash internal/ml/model_selection/scripts/latency_embedding_cache.sh & + # Draw graph + python ./internal/ml/model_selection/exps/micro/draw_filtering_latency.py + python ./internal/ml/model_selection/exps/micro/draw_filtering_memory_bar.py + python ./internal/ml/model_selection/exps/micro/draw_filtering_memory_line.py + python ./internal/ml/model_selection/exps/micro/draw_filtering_memory_cache_CPU.py + ``` + +2. Further we measure the end-2-end latency under two CPU, GPU, and Hybrid. + + ```bash + nohup bash internal/ml/model_selection/scripts/latency_phase1_cpu_gpu.sh & + ``` + +## Micro: In-DB vs Out-DB filtering phase + +```bash +# run out-of db, read data via psycopg2 +bash ./internal/ml/model_selection/scripts/latency_phase1_in_db.sh + +# run in-db query, read data via SPI +select benchmark_filtering_latency_in_db(5000, 'frappe', '/project/TRAILS/internal/ml/model_selection/config.ini'); + +select benchmark_filtering_latency_in_db(5000, 'uci_diabetes', '/project/TRAILS/internal/ml/model_selection/config.ini'); + +select benchmark_filtering_latency_in_db(5000, 'criteo', '/project/TRAILS/internal/ml/model_selection/config.ini'); +``` + +## Micro: On-the-Fly Data transmission, Refinement + +```bash +# start cache service +python ./internal/cache-service/cache_service.py +python ./internal/cache-service/trigger_cache_svc.py +# consume from the cache-svc + + +``` + +## Reproduce Figure7 + +```bash +python exps/main_v2/analysis/2.\ cost_draw.py +python exps/main_v2/analysis/3.\ cost_train_based.py +``` + +![image-20230702035622198](documents/imgs/image-20230702035622198.png) + +## Reproduce Figure8 + +```bash +# draw figure 8(a) +python exps/main_v2/analysis/5.draw_IDMS_var_workloads.py +# draw figure 8(b) +python exps/main_v2/analysis/6.draw_IDMS_dataloading.py +``` + +![image-20230702035639502](documents/imgs/image-20230702035639502.png) +# Baselines + +We compare with Training-Based MS, TabNAS, and training-free MS etc. + +For image data, it already generated at the NAS-Bench-Img part, see above. + +# Appendix + +Here all experiments is on the Frappe dataset. + +1. Computational Costs + + ```bash + bash ./internal/ml/model_selection/exps/micro/resp/benchmark_cost.sh + ``` + +2. Search Cost, multiple training-free or training-based combinations (warm-up / movel proposal) + + ```bash + # get RL, RE, RS + training-based model evaluation + bash ./internal/ml/model_selection/scripts/micro_search_strategy.sh + # this will read previous file, and run warm-up/move proposal, and draw all together + bash ./internal/ml/model_selection/exps/micro/resp/benchmark_search_cost.sh + ``` + +3. How des the K influence the result? + + ```bash + python ./internal/ml/model_selection/exps/micro/resp/benchmark_k_fix_time.py + ``` + +4. Nosy in selecting top K models + + ```bash + python ./internal/ml/model_selection/exps/micro/resp/benchmark_noisy_influence.py + ``` + +5. Weight-sharing result + + ```bash + nohup bash internal/ml/model_selection/scripts/benchmark_weight_sharing.sh & + ``` + + + + + + + +# Run end2end model selection + +download the dataset and put it in the `exp_data/data/structure_data` + +``` +python main.py --budget=100 --dataset=frappe +``` + +Check the log at the `logs_default` + +![image-20230421220338391](./documents/imgs/image-20230421220338391.png) + +![image-20230421220443231](./documents/imgs/image-20230421220443231.png) + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/config.ini b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/config.ini new file mode 100644 index 0000000000..61bdf6cfc5 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/config.ini @@ -0,0 +1,86 @@ +[DEFAULT] +log_name = in_db_ms +budget = 100 +device = cpu +log_folder = ./internal/debug_singa_logger +;log_folder = /project/TRAILS/log_score_time_frappe +result_dir = ./internal/ml/model_selection/exp_result_singa/ +;result_dir = /project/TRAILS/internal/ml/model_selection/exp_result_sever_cache_sql_indb/ +num_points = 12 +max_load = -1 + +[SAMPLER] +search_space = mlp_sp +population_size = 10 +sample_size = 3 +simple_score_sum = True + +[NB101] +api_loc = nasbench_only108.pkl +init_channels = 16 +bn = 1 +num_stacks = 3 +num_modules_per_stack = 3 + +[NB201] +init_w_type = none +init_b_type = none +arch_size = 1 + +[MLP] +num_layers = 4 +hidden_choice_len = 20 + +[MLP_TRAINER] +epoch = 20 +batch_size = 32 +lr = 0.002 +patience = 1 +iter_per_epoch = 200 +nfeat = 5500 +nfield = 10 +nemb = 10 +report_freq = 30 +workers = 0 + +[DATASET] +;base_dir = ../exp_data/ +base_dir = /hdd1/xingnaili/exp_data/ +dataset = frappe +num_labels = 2 + +[SEQ_TRAIN] +worker_id = 0 +total_workers = 120 +total_models_per_worker = -1 +pre_partitioned_file = ./internal/ml/model_selection/exps/sampled_data/sampled_models_all.json + +[DIS_TRAIN] +worker_each_gpu = 6 +gpu_num = 8 + +[TUNE_INTERVAL] +kn_rate = -1 + +[ANYTIME] +only_phase1 = False +is_simulate = False + + +[SERVER] +refinement_url = http://localhost:8095/ +cache_svc_url = http://localhost:8093/ + +[DB_CONFIG] +db_name = pg_extension +db_user = postgres +db_host = 127.0.0.1 +db_port = 28814 + + +[SYS_PERFORMANCE] +models_explore = -1 +# tfmem = express_flow +tfmem = synflow +embedding_cache_filtering = True +concurrency = 1 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/eva_service.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/eva_service.py new file mode 100644 index 0000000000..691e739ede --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/eva_service.py @@ -0,0 +1,78 @@ +import calendar +import os +import time +import argparse +import configparser +from sanic import Sanic +from sanic.exceptions import InvalidUsage +from sanic.response import json + +ts = calendar.timegm(time.gmtime()) +os.environ.setdefault("log_logger_folder_name", "log_eval_service") +os.environ.setdefault("log_file_name", "eval_service_" + str(ts) + ".log") +from src.logger import logger +from src.eva_engine.run_ms import RunModelSelection +from src.dataset_utils.stream_dataloader import StreamingDataLoader +from shared_config import parse_config_arguments +from typing import Any, List, Dict, Tuple + + +def refinement_phase(u: int, k_models: List, dataset_name: str, config_file: str): + """ + U: training-epoches + K-Models: k models to train + config_file: config file path + """ + args = parse_config_arguments(config_file) + args.device = "cuda:7" + train_dataloader = StreamingDataLoader( + cache_svc_url=args.cache_svc_url, table_name=f"{dataset_name}_train", name_space="train") + eval_dataloader = StreamingDataLoader( + cache_svc_url=args.cache_svc_url, table_name=f"{dataset_name}_valid", name_space="valid") + + try: + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + best_arch, best_arch_performance, _ = rms.refinement_phase( + U=u, + k_models=k_models, + train_loader=train_dataloader, + valid_loader=eval_dataloader) + finally: + train_dataloader.stop() + eval_dataloader.stop() + return {"best_arch": best_arch, "best_arch_performance": best_arch_performance} + + +app = Sanic("evaApp") + + +@app.route("/", methods=["POST"]) +async def start_refinement_phase(request): + # Check if request is JSON + if not request.json: + logger.info("Expecting JSON payload") + raise InvalidUsage("Expecting JSON payload") + + u = request.json.get('u') + k_models = request.json.get('k_models') + dataset_name = request.json.get('dataset_name') + config_file = request.json.get('config_file') + + if u is None or k_models is None or config_file is None: + logger.info(f"Missing 'u' or 'k_models' in JSON payload, {request.json}") + raise InvalidUsage("Missing 'u' or 'k_models' in JSON payload") + + result = refinement_phase(u, k_models, dataset_name, config_file) + + return json(result) + + +if __name__ == "__main__": + + result = refinement_phase( + u=1, + k_models=["8-8-8-8", "16-16-16-16"], + dataset_name="frappe", + config_file="/home/xingnaili/firmest_docker/TRAILS/internal/ml/model_selection/config.ini") + + # app.run(host="0.0.0.0", port=8095) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exp_result_singa/.gitkeep b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exp_result_singa/.gitkeep new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/README.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/README.md new file mode 100644 index 0000000000..523566c42d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/README.md @@ -0,0 +1,21 @@ +# Folder description + +## baseline + +We store the baseline algorithm here + +## benchmark_tfmem + +We benchmarking TFMEM here + +## macro/micro + +We benchmark the system from both macro and analysis component in micro + +## nas_bench_tabular + +We build a nas-bench-tabular dataset here + +## system + +We run the experiment here \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_img_lib.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_img_lib.py new file mode 100644 index 0000000000..2e3e55431d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_img_lib.py @@ -0,0 +1,706 @@ +import os + +from matplotlib import pyplot as plt +import seaborn as sns +import numpy as np +import palettable +from matplotlib.ticker import MaxNLocator +import numpy +from src.common.constant import Config +import matplotlib + +# lines' mark size +set_marker_size = 15 +# points' mark size +set_marker_point = 14 +# points' mark size +set_font_size = 40 +set_lgend_size = 15 +set_tick_size = 20 + +frontinsidebox = 23 + +# update tick size +matplotlib.rc('xtick', labelsize=set_tick_size) +matplotlib.rc('ytick', labelsize=set_tick_size) + +plt.rcParams['axes.labelsize'] = set_tick_size + +mark_list = ["o", "*", "<", "^", "s", "d", "D", ">", "h"] +mark_size_list = [set_marker_size, set_marker_size + 1, set_marker_size + 1, set_marker_size, + set_marker_size, set_marker_size, set_marker_size, set_marker_size + 1, set_marker_size + 2] +line_shape_list = ['-.', '--', '-', ':'] + + +# this is for draw figure3 only +def get_plot_compare_with_base_line_cfg(search_space, dataset, if_with_phase1=False): + if search_space == Config.NB201: + run_range_ = range(0, 100, 1) + if if_with_phase1: + draw_graph = draw_anytime_result_with_p1 + else: + draw_graph = draw_anytime_result + # min, this is for plot only + if dataset == Config.c10: + # C10 array + budget_array = [0.017, 0.083] + list(range(1, 350, 4)) + sub_graph_y1 = [91, 94.5] + sub_graph_y2 = [53.5, 55] + sub_graph_split = 60 + elif dataset == Config.c100: + # C10 array + budget_array = [0.017, 0.083] + list(range(1, 350, 4)) + + sub_graph_y1 = [64, 73.5] + sub_graph_y2 = [15, 16] + sub_graph_split = 20 + else: + # ImgNet X array + budget_array = [0.017, 0.083] + list(range(1, 350, 4)) + sub_graph_y1 = [33, 48] + sub_graph_y2 = [15.5, 17] + sub_graph_split = 34 + else: + # this is NB101 + C10, because only 101 has 20 run. others have 100 run. + run_range_ = range(0, 20, 1) + if if_with_phase1: + draw_graph = draw_anytime_result_one_graph_with_p1 + # budget_array = list(range(1, 16, 1)) + budget_array = numpy.arange(0.02, 15, 0.02).tolist() + else: + draw_graph = draw_anytime_result_one_graph + budget_array = [0.017, 0.083] + list(range(1, 2000, 8)) + + if dataset == Config.c10: + # C10 array + # budget_array = list(range(0, 2000, 1)) + sub_graph_y1 = [90, 94.5] + sub_graph_y2 = [52, 55] + sub_graph_split = 60 + else: + raise Exception + + return run_range_, budget_array, sub_graph_y1, sub_graph_y2, sub_graph_split, draw_graph + + +def draw_anytime_result(result_dir, y_acc_list_arr, x_T_list, + x_acc_train, y_acc_train_l, y_acc_train_m, y_acc_train_h, + annotations, lv, + name_img, dataset, + x1_lim=[], x2_lim=[], + ): + fig, (ax1, ax2) = plt.subplots(2, 1, sharex=True, dpi=100, gridspec_kw={'height_ratios': [4, 1]}) + exp = np.array(y_acc_list_arr) + sys_acc_h = np.quantile(exp, .75, axis=0) + sys_acc_m = np.quantile(exp, .5, axis=0) + sys_acc_l = np.quantile(exp, .25, axis=0) + + # plot simulate result of system + ax1.fill_between(x_T_list, sys_acc_l, sys_acc_h, alpha=0.1) + ax1.plot(x_T_list, sys_acc_m, mark_list[-1], label="TRAILS") + ax2.fill_between(x_T_list, sys_acc_l, sys_acc_h, alpha=0.1) + + # plot simulate result of train-based line + ax1.fill_between(x_acc_train, y_acc_train_l, y_acc_train_h, alpha=0.3) + ax1.plot(x_acc_train, y_acc_train_m, mark_list[-2], label="Training-based MS") + ax2.fill_between(x_acc_train, y_acc_train_l, y_acc_train_h, alpha=0.3) + + for i in range(len(annotations)): + ele = annotations[i] + if ele[1] < lv: + # convert to mins + ax2.plot(ele[2] / 60, ele[1], mark_list[i], label=ele[0], fontsize=set_marker_size) + else: + ax1.plot(ele[2] / 60, ele[1], mark_list[i], label=ele[0], fontsize=set_marker_size) + # ax2.scatter(ele[2]/60, ele[1]* 0.01, s=100, color="red") + # ax2.annotate(ele[0], (ele[2]/60, ele[1] * 0.01)) + + if len(x1_lim) > 0 and len(x2_lim) > 0: + ax1.set_ylim(x1_lim[0], x1_lim[1]) # 子图1设置y轴范围,只显示部分图 + ax2.set_ylim(x2_lim[0], x2_lim[1]) # 子图2设置y轴范围,只显示部分图 + + ax1.spines['bottom'].set_visible(False) # 关闭子图1中底部脊 + ax2.spines['top'].set_visible(False) ##关闭子图2中顶部脊 + ax2.set_xticks(range(0, 31, 1)) + + d = .85 # 设置倾斜度 + # 绘制断裂处的标记 + kwargs = dict(marker=[(-1, -d), (1, d)], markersize=set_marker_size, + linestyle='none', color='r', mec='r', mew=1, clip_on=False) + ax1.plot([0, 1], [0, 0], transform=ax1.transAxes, **kwargs) + ax2.plot([0, 1], [1, 1], transform=ax2.transAxes, **kwargs) + + plt.tight_layout() + plt.xscale("symlog") + ax1.grid() + ax2.grid() + plt.xlabel("Time Budget given by user (min)", fontsize=set_font_size) + ax1.set_ylabel(f"Test accuracy on {dataset}", fontsize=set_font_size) + ax1.legend(ncol=1, fontsize=set_lgend_size) + ax2.legend(fontsize=set_lgend_size) + # plt.show() + plt.savefig(f"{result_dir}/any_time_{name_img}.pdf", bbox_inches='tight') + + +def draw_anytime_result_one_graph(y_acc_list_arr, x_T_list, + x_acc_train, y_acc_train_l, y_acc_train_m, y_acc_train_h, + annotations, lv, + name_img, dataset, + x1_lim=[], x2_lim=[], + ): + # fig, (ax1, ax2) = plt.subplots(2, 1, sharex=True, dpi=100, gridspec_kw={'height_ratios': [5, 1]}) + exp = np.array(y_acc_list_arr) * 100 + sys_acc_h = np.quantile(exp, .75, axis=0) + sys_acc_m = np.quantile(exp, .5, axis=0) + sys_acc_l = np.quantile(exp, .25, axis=0) + + # exp_time = np.array(real_time_used_arr) + # time_mean = np.quantile(exp_time, .5, axis=0) + time_mean = x_T_list + + # plot simulate result of system + plt.fill_between(time_mean, sys_acc_l, sys_acc_h, alpha=0.1) + plt.plot(time_mean, sys_acc_m, "o-", label="TRAILS") + # plt.plot(time_mean, sys_acc_m, label="TRAILS") + + # plot simulate result of train-based line + plt.fill_between(x_acc_train, y_acc_train_l, y_acc_train_h, alpha=0.3) + plt.plot(x_acc_train, y_acc_train_m, "o-", label="Training-based MS") + # plt.plot(x_acc_train, y_acc_train_m, label="Training-based MS") + + if len(x1_lim) > 0: + plt.ylim(x1_lim[0], x1_lim[1]) # 子图1设置y轴范围,只显示部分图 + + d = .85 # 设置倾斜度 + # 绘制断裂处的标记 + kwargs = dict(marker=[(-1, -d), (1, d)], markersize=set_marker_size, + linestyle='none', color='r', mec='r', mew=1, clip_on=False) + # plt.plot([0, 1], [0, 0], transform=ax1.transAxes, **kwargs) + # plt.plot([0, 1], [1, 1], transform=ax2.transAxes, **kwargs) + + plt.tight_layout() + # plt.xscale("symlog") + plt.grid() + plt.xlabel("Time Budget given by user (min)", fontsize=set_font_size) + plt.ylabel(f"Test accuracy on {dataset}", fontsize=set_font_size) + plt.legend(ncol=1, fontsize=set_lgend_size) + plt.show() + # plt.savefig(f"amy_time_{name_img}.pdf", bbox_inches='tight') + + +# those two function will plot phase 1 and phase 2 +def draw_anytime_result_with_p1(result_dir, y_acc_list_arr, x_T_list, y_acc_list_arr_p1, x_T_list_p1, + x_acc_train, y_acc_train_l, y_acc_train_m, y_acc_train_h, + annotations, lv, + name_img, dataset, max_value, + x1_lim=[], x2_lim=[], + ): + fig, (ax1, ax2) = plt.subplots( + 2, 1, + sharex=True, + dpi=100, + gridspec_kw={'height_ratios': [6, 1]}) + + shade_degree = 0.2 + + # plot simulate result of train-based line + ax1.plot(x_acc_train, y_acc_train_m, mark_list[-3] + line_shape_list[0], label="Training-Based MS", + markersize=mark_size_list[-3]) + ax1.fill_between(x_acc_train, y_acc_train_l, y_acc_train_h, alpha=shade_degree) + ax2.fill_between(x_acc_train, y_acc_train_l, y_acc_train_h, alpha=shade_degree) + + # plot simulate result of system + exp = np.array(y_acc_list_arr_p1) + sys_acc_p1_h = np.quantile(exp, .75, axis=0) + sys_acc_p1_m = np.quantile(exp, .5, axis=0) + sys_acc_p1_l = np.quantile(exp, .25, axis=0) + ax1.plot(x_T_list_p1, sys_acc_p1_m, mark_list[-2] + line_shape_list[1], label="Training-Free MS", + markersize=mark_size_list[-2]) + ax1.fill_between(x_T_list_p1, sys_acc_p1_l, sys_acc_p1_h, alpha=shade_degree) + ax2.fill_between(x_T_list_p1, sys_acc_p1_l, sys_acc_p1_h, alpha=shade_degree) + + # plot simulate result of system + exp = np.array(y_acc_list_arr) + sys_acc_h = np.quantile(exp, .75, axis=0) + sys_acc_m = np.quantile(exp, .5, axis=0) + sys_acc_l = np.quantile(exp, .25, axis=0) + ax1.plot(x_T_list, sys_acc_m, mark_list[-1] + line_shape_list[2], label="2Phase-MS", markersize=mark_size_list[-1]) + ax1.fill_between(x_T_list, sys_acc_l, sys_acc_h, alpha=shade_degree) + ax2.fill_between(x_T_list, sys_acc_l, sys_acc_h, alpha=shade_degree) + + print(f"speed-up on {dataset} = {x_acc_train[-1] / x_T_list[-2]}, " + f"t_train = {x_acc_train[-1]}, t_f = {x_T_list[-2]}") + + for i in range(len(annotations)): + ele = annotations[i] + if ele[1] < lv: + # convert to mins + ax2.plot(ele[2] / 60, ele[1], mark_list[i], label=ele[0], markersize=set_marker_point) + else: + ax1.plot(ele[2] / 60, ele[1], mark_list[i], label=ele[0], markersize=set_marker_point) + # ax2.scatter(ele[2]/60, ele[1]* 0.01, s=100, color="red") + # ax2.annotate(ele[0], (ele[2]/60, ele[1] * 0.01)) + + if len(x1_lim) > 0 and len(x2_lim) > 0: + ax1.set_ylim(x1_lim[0], x1_lim[1]) # 子图1设置y轴范围,只显示部分图 + ax2.set_ylim(x2_lim[0], x2_lim[1]) # 子图2设置y轴范围,只显示部分图 + + ax1.spines['bottom'].set_visible(False) # 关闭子图1中底部脊 + ax2.spines['top'].set_visible(False) ##关闭子图2中顶部脊 + ax2.set_xticks(range(0, 31, 1)) + + d = .85 # 设置倾斜度 + # 绘制断裂处的标记 + kwargs = dict(marker=[(-1, -d), (1, d)], markersize=set_marker_size, + linestyle='none', color='r', mec='r', mew=1, clip_on=False) + ax1.plot([0, 1], [0, 0], transform=ax1.transAxes, **kwargs) + ax2.plot([0, 1], [1, 1], transform=ax2.transAxes, **kwargs) + + plt.xscale("log") + ax1.grid() + ax2.grid() + plt.xlabel(r"Response Time Threshold $T_{max}$ (min)", fontsize=set_font_size) + ax1.set_ylabel(f"Test Acc on {'In-16'}", fontsize=set_font_size) + # ax1.legend(ncol=1, fontsize=set_lgend_size) + # ax2.legend(fontsize=set_lgend_size) + + ax1.xaxis.label.set_size(set_tick_size) + ax1.yaxis.label.set_size(set_tick_size) + # ax1.set_xticks([]) + + ax2.xaxis.label.set_size(set_tick_size) + ax2.yaxis.label.set_size(set_tick_size) + + ax1.yaxis.set_major_locator(MaxNLocator(nbins=4, integer=True)) + + ax1.axhline(max_value, color='r', linestyle='-', label='Global Best Accuracy') + + tick_values = [0.01, 0.1, 1, 10, 100, 1000] + ax2.set_xticks(tick_values) + ax2.set_xticklabels([f'$10^{{{int(np.log10(val))}}}$' for val in tick_values]) + + # this is for unique hash + export_legend( + fig, + colnum=3, + unique_labels=['TE-NAS (Training-Free)', 'ENAS (Weight sharing)', + 'KNAS (Training-Free)', 'DARTS-V1 (Weight sharing)', 'DARTS-V2 (Weight sharing)', + 'Training-Based MS', 'Training-Free MS', '2Phase-MS', 'Global Best Accuracy']) + plt.tight_layout() + fig.savefig(f"{result_dir}/any_time_{name_img}_p1_from_0.1_sec.pdf", bbox_inches='tight') + + +def export_legend(ori_fig, filename="any_time_legend", colnum=9, unique_labels=[]): + fig2 = plt.figure(figsize=(5, 0.3)) + lines_labels = [ax.get_legend_handles_labels() for ax in ori_fig.axes] + lines, labels = [sum(lol, []) for lol in zip(*lines_labels)] + # grab unique labels + if len(unique_labels) == 0: + unique_labels = set(labels) + # assign labels and legends in dict + legend_dict = dict(zip(labels, lines)) + # query dict based on unique labels + unique_lines = [legend_dict[x] for x in unique_labels] + fig2.legend(unique_lines, unique_labels, loc='center', + ncol=colnum, + fancybox=True, + shadow=True, scatterpoints=1, fontsize=set_lgend_size) + fig2.tight_layout() + fig2.savefig(f"{filename}.pdf", bbox_inches='tight') + + +def draw_anytime_result_one_graph_with_p1(y_acc_list_arr, x_T_list, y_acc_list_arr_p1, x_T_list_p1, + x_acc_train, y_acc_train_l, y_acc_train_m, y_acc_train_h, + annotations, lv, + name_img, dataset, + x1_lim=[], x2_lim=[], + ): + # fig, (ax1, ax2) = plt.subplots(2, 1, sharex=True, dpi=100, gridspec_kw={'height_ratios': [5, 1]}) + + # plot simulate result of system + exp = np.array(y_acc_list_arr_p1) * 100 + sys_acc_p1_h = np.quantile(exp, .75, axis=0) + sys_acc_p1_m = np.quantile(exp, .5, axis=0) + sys_acc_p1_l = np.quantile(exp, .25, axis=0) + + plt.fill_between(x_T_list_p1, sys_acc_p1_l, sys_acc_p1_h, alpha=0.1) + plt.plot(x_T_list_p1, sys_acc_p1_m, "o-", label="TRAILS-P1") + # plt.fill_between(x_T_list_p1, sys_acc_p1_l, sys_acc_p1_h, alpha=0.1) + + exp = np.array(y_acc_list_arr) * 100 + sys_acc_h = np.quantile(exp, .75, axis=0) + sys_acc_m = np.quantile(exp, .5, axis=0) + sys_acc_l = np.quantile(exp, .25, axis=0) + + # exp_time = np.array(real_time_used_arr) + # time_mean = np.quantile(exp_time, .5, axis=0) + time_mean = x_T_list + + # plot simulate result of system + plt.fill_between(time_mean, sys_acc_l, sys_acc_h, alpha=0.1) + plt.plot(time_mean, sys_acc_m, "o-", label="TRAILS") + # plt.plot(time_mean, sys_acc_m, label="TRAILS") + + # plot simulate result of train-based line + plt.fill_between(x_acc_train, y_acc_train_l, y_acc_train_h, alpha=0.3) + plt.plot(x_acc_train, y_acc_train_m, "o-", label="Training-based MS") + # plt.plot(x_acc_train, y_acc_train_m, label="Training-based MS") + + if len(x1_lim) > 0: + plt.ylim(x1_lim[0], x1_lim[1]) # 子图1设置y轴范围,只显示部分图 + + d = .85 # 设置倾斜度 + # 绘制断裂处的标记 + kwargs = dict(marker=[(-1, -d), (1, d)], markersize=set_marker_size, + linestyle='none', color='r', mec='r', mew=1, clip_on=False) + # plt.plot([0, 1], [0, 0], transform=ax1.transAxes, **kwargs) + # plt.plot([0, 1], [1, 1], transform=ax2.transAxes, **kwargs) + + plt.tight_layout() + plt.xscale("symlog") + plt.grid() + plt.xlabel("Time Budget given by user (min)", fontsize=set_font_size) + plt.ylabel(f"Test accuracy on {dataset}", fontsize=set_font_size) + plt.legend(ncol=1, fontsize=set_lgend_size) + # plt.show() + plt.savefig(f"amy_time_{name_img}.pdf", bbox_inches='tight') + + +# for K, U N trade-off +def draw_grid_graph_with_budget( + acc, bt, b1, b2, + img_name: str, y_array: list, x_array: list): + """ + :param acc: Two array list + :param bt: Two array list + :param img_name: img name string + :return: + """ + + acc_new = np.array(acc) + acc = acc_new.tolist() + + mask = np.array(acc) + mask[mask > 0] = 0 + mask[mask < 0] = 1 + + bt = np.round(np.array(bt), 2).tolist() + mask2 = np.array(bt) + mask2[mask2 > 0] = 0 + mask2[mask2 < 0] = 1 + + mask3 = np.array(b1) + mask3[mask3 > 0] = 0 + mask3[mask3 < 0] = 1 + + mask4 = np.array(b2) + mask4[mask4 > 0] = 0 + mask4[mask4 < 0] = 1 + + fig, ax = plt.subplots(2, 2, figsize=(15, 14)) + + linewidths = 0.5 + sns.set(font_scale=3) + sns.heatmap( + data=acc, + vmax=99, + vmin=93, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'bottom'}, + mask=mask, + square=True, linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .5}, + ax=ax[0, 0] + ) + + sns.heatmap( + data=bt, + # vmax=, + vmin=-9, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'top'}, + mask=mask2, + square=True, linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .5}, + ax=ax[0, 1] + ) + + sns.heatmap( + data=b1, + vmax=17000, + vmin=15000, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".0f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'top'}, + mask=mask4, + square=True, linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .5}, + ax=ax[1, 0] + ) + + sns.heatmap( + data=b2, + # vmax=, + # vmin=-9, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".0f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'top'}, + mask=mask4, + square=True, linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .5}, + ax=ax[1, 1] + ) + + plt.tight_layout() + plt.xlabel("U (epoch)", fontsize=set_font_size) + plt.ylabel("K (# models)", fontsize=set_font_size) + + for i in [0, 1]: + for j in [0, 1]: + ax[i, j].set_xticklabels(x_array, fontsize=set_font_size) + ax[i, j].set_yticklabels(y_array, fontsize=set_font_size) + ax[i, j].set_xlabel("U (# epoch)", fontsize=set_font_size) + ax[i, j].set_ylabel("K (# models)", fontsize=set_font_size) + + ax[0, 0].set_title('Test Accuracy (%)', fontsize=set_font_size) + ax[0, 1].set_title(r'Time Budget $T$ (min)', fontsize=set_font_size) + ax[1, 0].set_title(r'$N$', fontsize=set_font_size) + ax[1, 1].set_title(r"$K \cdot U \cdot \log_{\eta}K$", fontsize=set_font_size) + + plt.tight_layout() + fig.subplots_adjust(wspace=0.001, hspace=0.3) + + # plt.show() + base_dr = os.getcwd() + path_gra = os.path.join(base_dr, f"{img_name}.pdf") + fig.savefig(path_gra, bbox_inches='tight') + + +def draw_grid_graph_with_budget_only_Acc_and_T( + acc, bt, b1, b2, + img_name: str, y_array: list, x_array: list): + """ + :param acc: Two array list + :param bt: Two array list + :param img_name: img name string + :return: + """ + + acc_new = np.array(acc) + acc = acc_new.tolist() + + mask = np.array(acc) + mask[mask > 0] = 0 + mask[mask < 0] = 1 + + bt = np.round(np.array(bt), 2).tolist() + mask2 = np.array(bt) + mask2[mask2 > 0] = 0 + mask2[mask2 < 0] = 1 + + mask3 = np.array(b1) + mask3[mask3 > 0] = 0 + mask3[mask3 < 0] = 1 + + mask4 = np.array(b2) + mask4[mask4 > 0] = 0 + mask4[mask4 < 0] = 1 + + fig, ax = plt.subplots(1, 2, figsize=(15, 14)) + + linewidths = 0.5 + sns.set(font_scale=2) + sns.heatmap( + data=acc, + vmax=99, + vmin=93, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'bottom'}, + mask=mask, + square=True, + linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .4}, + ax=ax[0] + ) + + sns.heatmap( + data=bt, + vmax=600, + # vmin=-9, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'top'}, + mask=mask2, + square=True, + linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .4}, + ax=ax[1] + ) + + plt.tight_layout() + plt.xlabel("U (epoch)", fontsize=set_font_size) + plt.ylabel("K (# models)", fontsize=set_font_size) + + for j in [0, 1]: + ax[j].set_xticklabels(x_array, fontsize=set_font_size) + ax[j].set_yticklabels(y_array, fontsize=set_font_size) + ax[j].set_xlabel("U (# epoch)", fontsize=set_font_size) + ax[j].set_ylabel("K (# models)", fontsize=set_font_size) + + ax[0].set_title('Test Accuracy (%)', fontsize=set_font_size) + ax[1].set_title(r'Time Budget $T$ (min)', fontsize=set_font_size) + + plt.tight_layout() + fig.subplots_adjust(wspace=0.3, hspace=0.3) + + # plt.show() + base_dr = os.getcwd() + path_gra = os.path.join(base_dr, f"{img_name}.pdf") + fig.savefig(path_gra, bbox_inches='tight') + + +def draw_grid_graph_with_budget_only_Acc( + acc, bt, b1, b2, + img_name: str, y_array: list, x_array: list): + """ + :param acc: Two array list + :param bt: Two array list + :param img_name: img name string + :return: + """ + + acc_new = np.array(acc) + acc = acc_new.tolist() + + mask = np.array(acc) + mask[mask > 0] = 0 + mask[mask < 0] = 1 + + fig = plt.figure(figsize=(7, 14)) + + linewidths = 0.5 + sns.set(font_scale=2) + sns.heatmap( + data=acc, + vmax=99, + vmin=93, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'bottom'}, + mask=mask, + square=True, + linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .4}, + ax=fig + ) + + plt.tight_layout() + plt.xlabel("U (epoch)", fontsize=set_font_size) + plt.ylabel("K (# models)", fontsize=set_font_size) + + plt.xticks(x_array, fontsize=set_font_size) + plt.yticks(y_array, fontsize=set_font_size) + + plt.title('Test Accuracy (%)', fontsize=set_font_size) + plt.tight_layout() + # fig.subplots_adjust(wspace=0.3, hspace=0.3) + # plt.show() + base_dr = os.getcwd() + path_gra = os.path.join(base_dr, f"{img_name}.pdf") + fig.savefig(path_gra, bbox_inches='tight') + + +def draw_grid_graph_with_budget_only_T( + acc, bt, b1, b2, + img_name: str, y_array: list, x_array: list): + """ + :param acc: Two array list + :param bt: Two array list + :param img_name: img name string + :return: + """ + + acc_new = np.array(acc) + acc = acc_new.tolist() + + mask = np.array(acc) + mask[mask > 0] = 0 + mask[mask < 0] = 1 + + bt = np.round(np.array(bt), 2).tolist() + mask2 = np.array(bt) + mask2[mask2 > 0] = 0 + mask2[mask2 < 0] = 1 + + mask3 = np.array(b1) + mask3[mask3 > 0] = 0 + mask3[mask3 < 0] = 1 + + mask4 = np.array(b2) + mask4[mask4 > 0] = 0 + mask4[mask4 < 0] = 1 + + fig, ax = plt.subplots(1, 2, figsize=(15, 14)) + + linewidths = 0.5 + sns.set(font_scale=2) + sns.heatmap( + data=acc, + vmax=99, + vmin=93, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'bottom'}, + mask=mask, + square=True, + linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .4}, + ax=ax[0] + ) + + sns.heatmap( + data=bt, + vmax=600, + # vmin=-9, + cmap=palettable.cmocean.diverging.Curl_10.mpl_colors, + annot=True, + fmt=".2f", + annot_kws={'size': frontinsidebox, 'weight': 'normal', 'color': 'w', 'va': 'top'}, + mask=mask2, + square=True, + linewidths=linewidths, # 每个方格外框显示,外框宽度设置 + cbar_kws={"shrink": .4}, + ax=ax[1] + ) + + plt.tight_layout() + plt.xlabel("U (epoch)", fontsize=set_font_size) + plt.ylabel("K (# models)", fontsize=set_font_size) + + for j in [0, 1]: + ax[j].set_xticklabels(x_array, fontsize=set_font_size) + ax[j].set_yticklabels(y_array, fontsize=set_font_size) + ax[j].set_xlabel("U (# epoch)", fontsize=set_font_size) + ax[j].set_ylabel("K (# models)", fontsize=set_font_size) + + ax[0].set_title('Test Accuracy (%)', fontsize=set_font_size) + ax[1].set_title(r'Time Budget $T$ (min)', fontsize=set_font_size) + + plt.tight_layout() + fig.subplots_adjust(wspace=0.3, hspace=0.3) + + # plt.show() + base_dr = os.getcwd() + path_gra = os.path.join(base_dr, f"{img_name}.pdf") + fig.savefig(path_gra, bbox_inches='tight') diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_tab_lib.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_tab_lib.py new file mode 100644 index 0000000000..146fd36ba9 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/draw_tab_lib.py @@ -0,0 +1,197 @@ +from typing import List + +import numpy as np +from matplotlib import pyplot as plt +from matplotlib.ticker import MaxNLocator +import warnings +import matplotlib.cbook + +warnings.filterwarnings("ignore", category=matplotlib.cbook.mplDeprecation) + +# lines' mark size +set_marker_size = 1 +# points' mark size +set_marker_point = 14 +# points' mark size +set_font_size = 20 +set_lgend_size = 15 +set_tick_size = 20 + +frontinsidebox = 23 + +# update tick size +matplotlib.rc('xtick', labelsize=set_tick_size) +matplotlib.rc('ytick', labelsize=set_tick_size) + +plt.rcParams['axes.labelsize'] = set_tick_size + +mark_list = ["o", "*", "<", "^", "s", "d", "D", ">", "h"] +mark_size_list = [set_marker_size, set_marker_size + 1, set_marker_size + 1, set_marker_size, + set_marker_size, set_marker_size, set_marker_size, set_marker_size + 1, set_marker_size + 2] +line_shape_list = ['-.', '--', '-', ':'] +shade_degree = 0.2 + + +def Add_one_line(x_time_array: list, y_twod_budget: List[List], namespace: str, index, ax): + # training-based + x_ = x_time_array + y_ = y_twod_budget + + if all(isinstance(item, list) for item in x_): + expx = np.array(x_) + x_m = np.quantile(expx, .5, axis=0) + else: + x_m = x_ + + exp = np.array(y_) + exp = np.where(exp > 10, exp, exp * 100) + + y_h = np.quantile(exp, .75, axis=0) + y_m = np.quantile(exp, .5, axis=0) + y_l = np.quantile(exp, .25, axis=0) + + ax.plot(x_m, y_m, + mark_list[int(index % len(mark_list))] + line_shape_list[int(index % len(line_shape_list))], + label=namespace, + markersize=mark_size_list[int(index % len(mark_list))], + linewidth=3 + ) + + ax.fill_between(x_m, y_l, y_h, alpha=shade_degree) + return x_m + + +def draw_structure_data_anytime( + all_lines: List, + dataset: str, name_img: str, max_value, + figure_size=(6.4, 4.5), + annotations=[], + x_ticks=None, y_ticks=None, unique_labels=None): + fig, ax = plt.subplots(figsize=figure_size) + + # draw all lines + time_usage = [] + for i, each_line_info in enumerate(all_lines): + _x_array = each_line_info[0] + _y_2d_array = each_line_info[1] + _name_space = each_line_info[2] + time_arr = Add_one_line(_x_array, _y_2d_array, _name_space, i, ax) + time_usage.append(time_arr) + + # print(f"speed-up on {dataset} = {time_usage[0][-1] / time_usage[2][-2]}, " + # f"t_train = {time_usage[0][-1]}, t_f = {time_usage[2][-2]}") + + # plt.xscale("log") + # plt.grid() + # plt.xlabel(r"Time Budget $T$ (min)", fontsize=set_font_size) + # plt.ylabel(f"AUC on {dataset.upper()}", fontsize=set_font_size) + + plt.xscale("log") + ax.grid() + ax.set_xlabel(r"Response Time Threshold $T_{max}$ (min)", fontsize=set_font_size) + ax.set_ylabel(f"AUC on {dataset.upper()}", fontsize=set_font_size) + # ax.set_xscale("log") + # ax.set_xlim(0.001, 10e4) + # ax.set_ylim(x1_lim[0], x1_lim[1]) + + if y_ticks is not None: + if y_ticks[0] is not None: + ax.set_ylim(bottom=y_ticks[0]) + if y_ticks[1] is not None: + ax.set_ylim(top=y_ticks[1]) + # ax.set_ylim(y_ticks[0], y_ticks[1]) + # ax.set_yticks(y_ticks) + # ax.set_yticklabels(y_ticks) + if x_ticks is not None: + if x_ticks[0] is not None: + ax.set_xlim(left=x_ticks[0]) + if x_ticks[1] is not None: + ax.set_xlim(right=x_ticks[1]) + + ax.yaxis.set_major_locator(MaxNLocator(nbins=6, integer=False)) + + if max_value > 0: + plt.axhline(max_value, color='r', linestyle='-', label='Global Best AUC') + + for i in range(len(annotations)): + ele = annotations[i] + ax.plot(ele[2], ele[1], mark_list[i], label=ele[0], markersize=set_marker_point) + + # export_legend(fig, filename="any_time_legend", unique_labels=["Training-Based MS", "Training-Free MS", "2Phase-MS", 'Global Best AUC']) + export_legend(ori_fig=fig, colnum=5, unique_labels=unique_labels) + plt.tight_layout() + + fig.savefig(f"{name_img}.pdf", bbox_inches='tight') + + +def export_legend(ori_fig, filename="any_time_legend", colnum=9, unique_labels=None): + if unique_labels is None: + unique_labels = [] + fig2 = plt.figure(figsize=(5, 0.3)) + lines_labels = [ax.get_legend_handles_labels() for ax in ori_fig.axes] + lines, labels = [sum(lol, []) for lol in zip(*lines_labels)] + # grab unique labels + if len(unique_labels) == 0: + unique_labels = set(labels) + # assign labels and legends in dict + legend_dict = dict(zip(labels, lines)) + # query dict based on unique labels + unique_lines = [legend_dict[x] for x in unique_labels] + fig2.legend(unique_lines, unique_labels, loc='center', + ncol=colnum, + fancybox=True, + shadow=True, scatterpoints=1, fontsize=set_lgend_size) + fig2.tight_layout() + fig2.savefig(f"{filename}.pdf", bbox_inches='tight') + + +import seaborn as sns +import matplotlib.pyplot as plt + + +def plot_heatmap(data: List, fontsize: int, + x_array_name: str, y_array_name: str, + title: str, output_file: str, + decimal_places: int, + u_ticks, k_ticks, + ): + labelsize = fontsize + # Convert the data to a NumPy array + data_array = np.array(data) + + # Custom annotation function + def custom_annot(val): + return "{:.{}f}".format(val, decimal_places) if val > 0 else "" + + # Convert the custom annotations to a 2D array + annot_array = np.vectorize(custom_annot)(data_array) + + # Create a masked array to hide the cells with values less than or equal to 0 + masked_data = np.ma.masked_array(data_array, data_array <= 0) + + # Set the figure size (width, height) in inches + fig, ax = plt.subplots(figsize=(8, 4)) + + # Use the "viridis" colormap + cmap = "viridis" + + # Create a heatmap + sns.heatmap(masked_data, annot=annot_array, fmt='', cmap=cmap, mask=masked_data.mask, ax=ax, + annot_kws={"size": fontsize, "ha": "center", "va": "center"}, + xticklabels=u_ticks, yticklabels=k_ticks) + + # Set axis labels + ax.set_xlabel(x_array_name, fontsize=fontsize) + ax.set_ylabel(y_array_name, fontsize=fontsize) + + # Set x/y-axis tick size + ax.tick_params(axis='both', which='major', labelsize=labelsize) + + # Set the title + # ax.set_title(title, fontsize=fontsize) + + # Set tight layout + plt.tight_layout() + + # Save the plot to a PDF file + plt.savefig(output_file) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py new file mode 100644 index 0000000000..eb0528b1b5 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py @@ -0,0 +1,78 @@ +import calendar +import json +import os +import time +import traceback +from singa import device as singa_device +import numpy as np + +from exps.shared_args import parse_arguments + +if __name__ == "__main__": + + args = parse_arguments() + + # set the log name + gmt = time.gmtime() + ts = calendar.timegm(gmt) + + os.environ.setdefault("log_logger_folder_name", f"{args.log_folder}") + os.environ.setdefault("log_file_name", f"{args.log_name}_{args.dataset}_ep{args.epoch}_{ts}.log") + os.environ.setdefault("base_dir", args.base_dir) + + from src.logger import logger + from src.eva_engine.phase2.algo.trainer import ModelTrainer + from src.search_space.init_search_space import init_search_space + from src.dataset_utils.structure_data_loader import libsvm_dataloader + + search_space_ins = init_search_space(args) + search_space_ins.load() + + try: + # read the checkpoint + checkpoint_file_name = f"{args.result_dir}/train_config_tune_{args.dataset}_epo_{args.epoch}.json" + + # 1. data loader + train_loader, val_loader, test_loader = libsvm_dataloader( + args=args, + data_dir=os.path.join(args.base_dir, "data", "structure_data", args.dataset), + nfield=args.nfield, + batch_size=args.batch_size) + + # arch_id = "256-256-256-256" + arch_id = "128-128-128-128" + print(f"begin to train the {arch_id}") + + model = search_space_ins.new_architecture(arch_id) + # model.init_embedding(requires_grad=True) + if args.device == 'cpu': + dev = singa_device.get_default_device() + else: # GPU + dev = singa_device.create_cuda_gpu_on(args.local_rank) # need to change to CPU device for CPU-only machines + dev.SetRandSeed(0) + np.random.seed(0) + # model.to(args.device) + + valid_auc, total_run_time, train_log = ModelTrainer.fully_train_arch( + model=model, + use_test_acc=False, + epoch_num=args.epoch, + train_loader=train_loader, + val_loader=val_loader, + test_loader=test_loader, + args=args) + + logger.info(f' ----- model id: {arch_id}, Val_AUC : {valid_auc} Total running time: ' + f'{total_run_time}-----') + print(f' ----- model id: {arch_id}, Val_AUC : {valid_auc} Total running time: ' + f'{total_run_time}-----') + + # update the shared model eval res + logger.info(f" ---- info: {json.dumps({arch_id: train_log})}") + + print(f" ---- info: {json.dumps({arch_id: train_log})}") + + logger.info(f" Saving result to: {checkpoint_file_name}") + except: + print(traceback.format_exc()) + logger.info(traceback.format_exc()) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_dist_online.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_dist_online.py new file mode 100644 index 0000000000..be66279a48 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_dist_online.py @@ -0,0 +1,145 @@ +import argparse +import calendar +import json +import logging +import os +import time + +from exps.shared_args import parse_arguments + + +def partition_list_by_worker_id(lst, num_workers=15): + partitions = [] + for i in range(num_workers): + partitions.append([]) + for idx, item in enumerate(lst): + worker_id = idx % num_workers + partitions[worker_id].append(item) + return partitions + + +def start_one_worker(queue, args, worker_id, my_partition, search_space_ins, res): + from src.tools.io_tools import write_json, read_json + gmt = time.gmtime() + ts = calendar.timegm(gmt) + + os.environ.setdefault("log_file_name", f"{args.log_name}_{args.dataset}_wkid_{worker_id}_{ts}.log") + # import logging + logger = logging.getLogger(f"{args.dataset}_wkid_{worker_id}_{ts}") + if not os.path.exists(f"./{args.log_folder}"): + os.makedirs(f"./{args.log_folder}") + handler = logging.FileHandler(f"./{args.log_folder}/{args.log_name}_{args.dataset}_wkid_{worker_id}_{ts}.log") + formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(message)s') + handler.setFormatter(formatter) + logger.addHandler(handler) + from src.eva_engine.phase2.algo.trainer import ModelTrainer + + if args.total_models_per_worker is None: + logger.info( + f" ---- begin exploring, current worker have " + f"{len(my_partition)} models. explore all those models ") + else: + logger.info(f" ---- begin exploring, current worker have " + f"{len(my_partition)} models. but explore {args.total_models_per_worker} models ") + + train_loader, val_loader, test_loader = queue.get() + + checkpoint_file_name = f"./base_line_res_{args.dataset}/train_baseline_{args.dataset}_wkid_{worker_id}.json" + visited = read_json(checkpoint_file_name) + if visited == {}: + visited = {args.dataset: {}} + logger.info(f" ---- initialize checkpointing with {visited} . ") + else: + logger.info(f" ---- recovery from checkpointing with {len(visited[args.dataset])} model. ") + + explored_arch_num = 0 + for arch_index in my_partition: + print(f"begin to train the {arch_index}") + model = search_space_ins.new_architecture(res[arch_index]).to(args.device) + valid_auc, total_run_time, train_log = ModelTrainer.fully_train_arch( + model=model, + use_test_acc=False, + epoch_num=args.epoch, + train_loader=train_loader, + val_loader=val_loader, + test_loader=test_loader, + args=args, logger=logger) + + logger.info(f' ----- model id: {res[arch_index]}, Val_AUC : {valid_auc} Total running time: ' + f'{total_run_time}-----') + + # update the shared model eval res + logger.info(f" ---- exploring {explored_arch_num} model. ") + logger.info(f" ---- info: {json.dumps({res[arch_index]: train_log})}") + visited[args.dataset][res[arch_index]] = train_log + explored_arch_num += 1 + + if args.total_models_per_worker is not None and explored_arch_num > args.total_models_per_worker: + break + + logger.info(f" Saving result to: {checkpoint_file_name}") + write_json(checkpoint_file_name, visited) + + +if __name__ == "__main__": + mp.set_start_method('spawn', force=True) + args = parse_arguments() + + # set the log name + gmt = time.gmtime() + ts = calendar.timegm(gmt) + + os.environ.setdefault("log_file_name", f"{args.log_name}_{args.dataset}_main_{ts}.log") + os.environ.setdefault("base_dir", args.base_dir) + + from src.search_space.init_search_space import init_search_space + from src.dataset_utils.structure_data_loader import libsvm_dataloader + from src.tools.io_tools import write_json, read_json + import torch.multiprocessing as mp + + search_space_ins = init_search_space(args) + search_space_ins.load() + + # 1. main process partition data and group results, + res = read_json(args.pre_partitioned_file) + + total_workers = args.worker_each_gpu * args.gpu_num + all_partition = partition_list_by_worker_id(list(res.keys()), total_workers) + + train_loader, val_loader, test_loader = libsvm_dataloader( + args=args, + data_dir=os.path.join(args.base_dir, "data", "structure_data", args.dataset), + nfield=args.nfield, + batch_size=args.batch_size) + + # 2. put the shared dataloader into the queue, + queue = mp.Queue() + + # 3. Create a list of processes to train the models + processes = [] + worker_id = 0 + for gpu_id in range(args.gpu_num): + for _ in range(args.worker_each_gpu): + if args.device != "cpu": + args.device = f"cuda:{gpu_id}" + print(f"running process {[args.device, worker_id, len(all_partition[worker_id])]}") + p = mp.Process( + target=start_one_worker, + args=(queue, args, worker_id, all_partition[worker_id], search_space_ins, res, + ) + ) + p.start() + processes.append(p) + worker_id += 1 + + # 4. send to the queue + for gpu_id in range(args.gpu_num): + for _ in range(args.worker_each_gpu): + print("putting to queue ....") + queue.put((train_loader, val_loader, test_loader)) + + print("All processing are running, waiting all to finish....") + for p in processes: + p.join() + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py new file mode 100644 index 0000000000..c36803a8df --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py @@ -0,0 +1,100 @@ +import calendar +import json +import os +import time + +from exps.shared_args import parse_arguments + + +def partition_list_by_worker_id(lst, num_workers=15): + partitions = [] + for i in range(num_workers): + partitions.append([]) + for idx, item in enumerate(lst): + worker_id = idx % num_workers + partitions[worker_id].append(item) + return partitions + + +if __name__ == "__main__": + + args = parse_arguments() + + # set the log name + gmt = time.gmtime() + ts = calendar.timegm(gmt) + + os.environ.setdefault("log_logger_folder_name", f"{args.log_folder}") + os.environ.setdefault("log_file_name", f"{args.log_name}_{args.dataset}_wkid_{args.worker_id}_{ts}.log") + os.environ.setdefault("base_dir", args.base_dir) + + from src.logger import logger + from src.eva_engine.phase2.algo.trainer import ModelTrainer + from src.search_space.init_search_space import init_search_space + from src.dataset_utils.structure_data_loader import libsvm_dataloader + from src.tools.io_tools import write_json, read_json + + search_space_ins = init_search_space(args) + search_space_ins.load() + + # 1. data loader + logger.info(f" Loading data....") + train_loader, val_loader, test_loader = libsvm_dataloader( + args=args, + data_dir=os.path.join(args.base_dir, "data", "structure_data", args.dataset), + nfield=args.nfield, + batch_size=args.batch_size) + + res = read_json(args.pre_partitioned_file) + + all_partition = partition_list_by_worker_id(list(res.keys()), args.total_workers) + + if args.total_models_per_worker == -1: + logger.info( + f" ---- begin exploring, current worker have " + f"{len(all_partition[args.worker_id])} models. explore all those models ") + else: + logger.info(f" ---- begin exploring, current worker have " + f"{len(all_partition[args.worker_id])} models. but explore {args.total_models_per_worker} models ") + + # read the checkpoint + checkpoint_file_name = f"{args.result_dir}/train_baseline_{args.dataset}_wkid_{args.worker_id}.json" + visited = read_json(checkpoint_file_name) + if visited == {}: + visited = {args.dataset: {}} + logger.info(f" ---- initialize checkpointing with {visited} . ") + else: + logger.info(f" ---- recovery from checkpointing with {len(visited[args.dataset])} model. ") + + explored_arch_num = 0 + for arch_index in all_partition[args.worker_id]: + print(f"begin to train the {arch_index}") + if res[arch_index] in visited[args.dataset]: + logger.info(f" ---- model {res[arch_index]} already visited") + continue + model = search_space_ins.new_architecture(res[arch_index]) + model.init_embedding(requires_grad=True) + model.to(args.device) + valid_auc, total_run_time, train_log = ModelTrainer.fully_train_arch( + model=model, + use_test_acc=False, + epoch_num=args.epoch, + train_loader=train_loader, + val_loader=val_loader, + test_loader=test_loader, + args=args) + + logger.info(f' ----- model id: {res[arch_index]}, Val_AUC : {valid_auc} Total running time: ' + f'{total_run_time}-----') + + # update the shared model eval res + logger.info(f" ---- exploring {explored_arch_num} model. ") + logger.info(f" ---- info: {json.dumps({res[arch_index]: train_log})}") + visited[args.dataset][res[arch_index]] = train_log + explored_arch_num += 1 + + if args.total_models_per_worker != -1 and explored_arch_num > args.total_models_per_worker: + break + + logger.info(f" Saving result to: {checkpoint_file_name}") + write_json(checkpoint_file_name, visited) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py new file mode 100644 index 0000000000..64226d9b91 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py @@ -0,0 +1,116 @@ +import calendar +import json +import os +import random +import time +from exps.shared_args import parse_arguments +from datetime import datetime +import gc + +# import tracemalloc +# tracemalloc.start() +# +# +# def print_memory_usg(): +# snapshot = tracemalloc.take_snapshot() +# top_stats = snapshot.statistics('lineno') +# for stat in top_stats[:10]: # top 10 memory-consuming lines +# print(stat) + + +def generate_data_loader(): + if args.dataset in [Config.c10, Config.c100, Config.imgNet]: + train_loader, val_loader, class_num = dataset.get_dataloader( + train_batch_size=args.batch_size, + test_batch_size=args.batch_size, + dataset=args.dataset, + num_workers=1, + datadir=os.path.join(args.base_dir, "data")) + test_loader = val_loader + else: + train_loader, val_loader, test_loader = libsvm_dataloader( + args=args, + data_dir=os.path.join(args.base_dir, "data", "structure_data", args.dataset), + nfield=args.nfield, + batch_size=args.batch_size) + class_num = args.num_labels + + return train_loader, val_loader, test_loader, class_num + + +if __name__ == "__main__": + args = parse_arguments() + + # set the log name + gmt = time.gmtime() + ts = calendar.timegm(gmt) + os.environ.setdefault("log_logger_folder_name", f"{args.log_folder}") + os.environ.setdefault("log_file_name", args.log_name + "_" + str(ts) + ".log") + os.environ.setdefault("base_dir", args.base_dir) + + from src.common.constant import Config + from src.common.structure import ModelAcquireData + from src.controller.sampler_all.seq_sampler import SequenceSampler + from src.eva_engine.phase1.evaluator import P1Evaluator + from src.logger import logger + from src.search_space.init_search_space import init_search_space + from src.dataset_utils.structure_data_loader import libsvm_dataloader + from src.tools.io_tools import write_json, read_json + from src.dataset_utils import dataset + from src.common.constant import Config, CommonVars + + search_space_ins = init_search_space(args) + + train_loader, val_loader, test_loader, class_num = generate_data_loader() + + _evaluator = P1Evaluator(device=args.device, + num_label=args.num_labels, + dataset_name=args.dataset, + search_space_ins=search_space_ins, + train_loader=train_loader, + is_simulate=False, + metrics=args.tfmem, + enable_cache=args.embedding_cache_filtering) + + sampler = SequenceSampler(search_space_ins) + + explored_n = 0 + output_file = f"{args.result_dir}/score_{args.search_space}_{args.dataset}_batch_size_{args.batch_size}_{args.device}.json" + result = read_json(output_file) + print(f"begin to score all, currently we already explored {len(result.keys())}") + logger.info(f"begin to score all, currently we already explored {len(result.keys())}") + while True: + arch_id, arch_micro = sampler.sample_next_arch() + if arch_id is None: + logger.info("Stop exploring, meet None arch id") + break + if arch_id in result: + continue + if args.models_explore != -1 and explored_n > args.models_explore: + logger.info(f"Stop exploring, {explored_n} > {args.models_explore}") + break + # run the model selection + model_encoding = search_space_ins.serialize_model_encoding(arch_micro) + model_acquire_data = ModelAcquireData(model_id=arch_id, + model_encoding=model_encoding, + is_last=False) + data_str = model_acquire_data.serialize_model() + model_score = _evaluator.p1_evaluate(data_str) + explored_n += 1 + result[arch_id] = model_score + # print(f" {datetime.now()} finish arch = {arch_id}, model_score = {model_score}") + if explored_n % 1000 == 0: + # print_memory_usg() + # _evaluator.force_gc() + print("3. [trails] Phase 1: filter phase explored " + str(explored_n) + + "Total explored " + str(len(result)) + + " model, model_id = " + str(arch_id) + + " model_scores = " + json.dumps(model_score)) + if explored_n % 1000 == 0: + # print_memory_usg() + # _evaluator.force_gc() + logger.info("3. [trails] Phase 1: filter phase explored " + str(explored_n) + + " model, model_id = " + str(arch_id) + + " model_scores = " + json.dumps(model_score)) + write_json(output_file, result) + write_json(output_file, result) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_ecdf.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_ecdf.py new file mode 100644 index 0000000000..bb11058e62 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_ecdf.py @@ -0,0 +1,119 @@ + +import numpy as np +import matplotlib +import matplotlib.pyplot as plt +import os +from src.tools.io_tools import read_json + +# lines' mark size +set_marker_size = 15 +# points' mark size +set_marker_point = 14 +# points' mark size +set_font_size = 25 +set_lgend_size = 15 +set_tick_size = 20 + +frontinsidebox = 23 + +# update tick size +matplotlib.rc('xtick', labelsize=set_tick_size) +matplotlib.rc('ytick', labelsize=set_tick_size) + +plt.rcParams['axes.labelsize'] = set_tick_size + +mark_list = ["o", "*", "<", "^", "s", "d", "D", ">", "h"] +mark_size_list = [set_marker_size, set_marker_size + 1, set_marker_size + 1, set_marker_size, + set_marker_size, set_marker_size, set_marker_size, set_marker_size + 1, set_marker_size + 2] +line_shape_list = ['-.', '--', '-', ':'] +shade_degree = 0.2 +base_dir = "../exp_data/" + + +def export_legend(ori_fig, filename="any_time_legend", colnum=9, unique_labels=None): + if unique_labels is None: + unique_labels = [] + fig2 = plt.figure(figsize=(5, 0.3)) + lines_labels = [ax.get_legend_handles_labels() for ax in ori_fig.axes] + lines, labels = [sum(lol, []) for lol in zip(*lines_labels)] + # grab unique labels + if len(unique_labels) == 0: + unique_labels = set(labels) + # assign labels and legends in dict + legend_dict = dict(zip(labels, lines)) + # query dict based on unique labels + unique_lines = [legend_dict[x] for x in unique_labels] + fig2.legend(unique_lines, unique_labels, loc='center', + ncol=colnum, + fancybox=True, + shadow=True, scatterpoints=1, fontsize=set_lgend_size) + fig2.tight_layout() + fig2.savefig(f"{filename}.pdf", bbox_inches='tight') + + +def draw_edcf(): + # extract train_auc and valid_auc into separate lists + for dataset, architectures in data_dict.items(): + + fig, ax = plt.subplots(figsize=(6.4, 3.5)) + print(dataset) + train_auc = [] + valid_auc = [] + for architecture, epochs in architectures.items(): + for epoch, metrics in epochs.items(): + if str(epoch_sampled[dataset]) == epoch: + train_auc.append(metrics["train_auc"]) + valid_auc.append(metrics["valid_auc"]) + break + + # calculate and plot ECDF for train_auc + sorted_train_auc = np.sort(train_auc) + y_train = np.arange(1, len(sorted_train_auc) + 1) / len(sorted_train_auc) + plt.plot(sorted_train_auc, y_train, label='Training AUC', linewidth=3, linestyle='--') + + # calculate and plot ECDF for valid_auc + sorted_valid_auc = np.sort(valid_auc) + y_valid = np.arange(1, len(sorted_valid_auc) + 1) / len(sorted_valid_auc) + plt.plot(sorted_valid_auc, y_valid, label='Validation AUC', linewidth=3, linestyle='-') + + y_m = np.quantile(sorted_valid_auc, .5, axis=0) + print("medium", y_m, "best", max(sorted_valid_auc)) + # plt.xlim(left=0.45) + + plt.grid() + plt.xlabel('Accuracy') + plt.ylabel('ECDF') + # plt.legend(loc='upper left', fontsize=set_lgend_size) + plt.tight_layout() + export_legend(ori_fig=fig, colnum=5) + fig.savefig(f"space_{dataset}.pdf", bbox_inches='tight') + + +# dataset_used = "frappe" +dataset_used = "uci_diabetes" +# dataset_used = "criteo" + + +epoch_sampled = {"frappe": 19, "uci_diabetes": 35, "criteo": 9} + +if dataset_used == "frappe": + mlp_train_frappe = os.path.join( + base_dir, + "tab_data/frappe/all_train_baseline_frappe.json") + data_dict = read_json(mlp_train_frappe) +elif dataset_used == "uci_diabetes": + mlp_train_uci_diabetes = os.path.join( + base_dir, + "tab_data/uci_diabetes/all_train_baseline_uci_160k_40epoch.json") + + data_dict = read_json(mlp_train_uci_diabetes) +elif dataset_used == "criteo": + mlp_train_criteo = os.path.join( + base_dir, + "tab_data/criteo/all_train_baseline_criteo.json") + + data_dict = read_json(mlp_train_criteo) +else: + print("err") + +draw_edcf() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_param_auc.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_param_auc.py new file mode 100644 index 0000000000..327b665774 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/exps/nas_bench_tabular/measure_param_auc.py @@ -0,0 +1,126 @@ +import numpy as np +import matplotlib +import matplotlib.pyplot as plt +import os +from src.tools.io_tools import read_json + +# lines' mark size +set_marker_size = 15 +# points' mark size +set_marker_point = 14 +# points' mark size +set_font_size = 25 +set_lgend_size = 15 +set_tick_size = 20 +import matplotlib.ticker as ticker + +frontinsidebox = 23 + +# update tick size +matplotlib.rc('xtick', labelsize=set_tick_size) +matplotlib.rc('ytick', labelsize=set_tick_size) + +plt.rcParams['axes.labelsize'] = set_tick_size + +mark_list = ["o", "*", "<", "^", "s", "d", "D", ">", "h"] +mark_size_list = [set_marker_size, set_marker_size + 1, set_marker_size + 1, set_marker_size, + set_marker_size, set_marker_size, set_marker_size, set_marker_size + 1, set_marker_size + 2] +line_shape_list = ['-.', '--', '-', ':'] +shade_degree = 0.2 +base_dir = "../exp_data/" + + +def export_legend(ori_fig, filename="any_time_legend", colnum=9, unique_labels=None): + if unique_labels is None: + unique_labels = [] + fig2 = plt.figure(figsize=(5, 0.3)) + lines_labels = [ax.get_legend_handles_labels() for ax in ori_fig.axes] + lines, labels = [sum(lol, []) for lol in zip(*lines_labels)] + # grab unique labels + if len(unique_labels) == 0: + unique_labels = set(labels) + # assign labels and legends in dict + legend_dict = dict(zip(labels, lines)) + # query dict based on unique labels + unique_lines = [legend_dict[x] for x in unique_labels] + fig2.legend(unique_lines, unique_labels, loc='center', + ncol=colnum, + fancybox=True, + shadow=True, scatterpoints=1, fontsize=set_lgend_size) + fig2.tight_layout() + fig2.savefig(f"{filename}.pdf", bbox_inches='tight') + + +# Function to compute number of parameters for an architecture +def compute_params(architecture): + layers = [int(layer) for layer in architecture.split('-')] + params = 0 + for i in range(len(layers) - 1): + params += layers[i] * layers[i + 1] + # Add bias terms + params += sum(layers[1:]) + return params + + +# Function to convert large number into a string with 'k' for thousands +def func(x, pos): # formatter function takes tick label and tick position + if x == 0: + return f"0" + else: + s = f'{x / 1000000}M' + return s + + +def draw_parameter_performance(): + # extract train_auc and valid_auc into separate lists + for dataset, architectures in data_dict.items(): + fig, ax = plt.subplots(figsize=(6.4, 4)) + print(dataset) + param_sizes = [] + valid_auc = [] + for architecture, epochs in architectures.items(): + for epoch, metrics in epochs.items(): + if str(epoch_sampled[dataset]) == epoch: + param_sizes.append(compute_params(architecture)) + valid_auc.append(metrics["valid_auc"]) + break + + plt.scatter(param_sizes, valid_auc) + y_format = ticker.FuncFormatter(func) + ax.xaxis.set_major_formatter(y_format) + plt.grid() + plt.xlabel('Parameter Size') + plt.ylabel('Validation AUC') + # plt.legend(loc='upper left', fontsize=set_lgend_size) + plt.tight_layout() + export_legend(ori_fig=fig, colnum=5) + fig.savefig(f"para_{dataset}.jpg", bbox_inches='tight') + + +dataset_used = "frappe" +# dataset_used = "uci_diabetes" +# dataset_used = "criteo" + +epoch_sampled = {"frappe": 19, "uci_diabetes": 35, "criteo": 9} + +if dataset_used == "frappe": + mlp_train_frappe = os.path.join( + base_dir, + "tab_data/frappe/all_train_baseline_frappe.json") + data_dict = read_json(mlp_train_frappe) +elif dataset_used == "uci_diabetes": + mlp_train_uci_diabetes = os.path.join( + base_dir, + "tab_data/uci_diabetes/all_train_baseline_uci_160k_40epoch.json") + + data_dict = read_json(mlp_train_uci_diabetes) +elif dataset_used == "criteo": + mlp_train_criteo = os.path.join( + base_dir, + "tab_data/criteo/all_train_baseline_criteo.json") + + data_dict = read_json(mlp_train_criteo) +else: + print("err") + +draw_parameter_performance() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/init_env b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/init_env new file mode 100644 index 0000000000..b3204ea062 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/init_env @@ -0,0 +1,12 @@ + + + + +export PYTHONPATH=$PYTHONPATH:/project/TRAILS/internal/ml/model_selection +conda activate trails + + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/main.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/main.py new file mode 100644 index 0000000000..7b9e3f7884 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/main.py @@ -0,0 +1,60 @@ +# this is the main function of model selection. + +import calendar +import os +import time +from src.common.constant import Config +from src.dataset_utils.structure_data_loader import libsvm_dataloader +from exps.shared_args import parse_arguments + + +def generate_data_loader(): + if args.dataset in [Config.c10, Config.c100, Config.imgNet]: + train_loader, val_loader, class_num = dataset.get_dataloader( + train_batch_size=args.batch_size, + test_batch_size=args.batch_size, + dataset=args.dataset, + num_workers=1, + datadir=os.path.join(args.base_dir, "data")) + test_loader = val_loader + else: + train_loader, val_loader, test_loader = libsvm_dataloader( + args=args, + data_dir=os.path.join(args.base_dir, "data", "structure_data", args.dataset), + nfield=args.nfield, + batch_size=args.batch_size) + class_num = args.num_labels + + return train_loader, val_loader, test_loader, class_num + + +def run_with_time_budget(time_budget: float): + """ + :param time_budget: the given time budget, in second + :return: + """ + + # define dataLoader, and sample a mini-batch + train_loader, val_loader, test_loader, class_num = generate_data_loader() + args.num_labels = class_num + data_loader = [train_loader, val_loader, test_loader] + + rms = RunModelSelection(args.search_space, args, is_simulate=False) + best_arch, _, _, _, _, _, _, _ = rms.select_model_online(time_budget, data_loader) + + return best_arch + + +if __name__ == "__main__": + args = parse_arguments() + + # set the log name + gmt = time.gmtime() + ts = calendar.timegm(gmt) + os.environ.setdefault("log_file_name", args.log_name + "_" + str(ts) + ".log") + os.environ.setdefault("base_dir", args.base_dir) + + from src.eva_engine.run_ms import RunModelSelection + from src.dataset_utils import dataset + + run_with_time_budget(args.budget) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/pg_interface.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/pg_interface.py new file mode 100644 index 0000000000..9f9d1b4feb --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/pg_interface.py @@ -0,0 +1,617 @@ +import calendar +import os +import time +import requests +import json +from typing import List, Dict +import traceback +import orjson +from argparse import Namespace +from shared_config import parse_config_arguments + + +def exception_catcher(func): + def wrapper(encoded_str: str): + try: + # each functon accepts a json string + params = json.loads(encoded_str) + config_file = params.get("config_file") + + # Parse the config file + args = parse_config_arguments(config_file) + + # Set the environment variables + ts = calendar.timegm(time.gmtime()) + os.environ.setdefault("base_dir", args.base_dir) + os.environ.setdefault("log_logger_folder_name", args.log_folder) + os.environ.setdefault("log_file_name", args.log_name + "_" + str(ts) + ".log") + + # Call the original function with the parsed parameters + return func(params, args) + except Exception as e: + return orjson.dumps( + {"Errored": traceback.format_exc()}).decode('utf-8') + + return wrapper + +from torch.utils.data import Dataset +import torch +class LibsvmDataset(Dataset): + """ Dataset loader for Libsvm data format """ + + @staticmethod + def decode_libsvm(columns): + map_func = lambda pair: (int(pair[0]), float(pair[1])) + id, value = zip(*map(lambda col: map_func(col.split(':')), columns[:-1])) + sample = {'id': torch.LongTensor(id), + 'value': torch.FloatTensor(value), + 'y': float(columns[-1])} + return sample + + @staticmethod + def pre_processing(mini_batch_data: List[Dict]): + sample_lines = len(mini_batch_data) + nfields = len(mini_batch_data[0].keys()) - 1 + feat_id = torch.LongTensor(sample_lines, nfields) + feat_value = torch.FloatTensor(sample_lines, nfields) + y = torch.FloatTensor(sample_lines) + + for i in range(sample_lines): + row_value = mini_batch_data[i].values() + sample = LibsvmDataset.decode_libsvm(list(row_value)) + feat_id[i] = sample['id'] + feat_value[i] = sample['value'] + y[i] = sample['y'] + return feat_id, feat_value, y, sample_lines + + def __init__(self, mini_batch_data: List[Dict]): + self.feat_id, self.feat_value, self.y, self.nsamples = \ + LibsvmDataset.pre_processing(mini_batch_data) + + def __len__(self): + return self.nsamples + + def __getitem__(self, idx): + return {'id': self.feat_id[idx], + 'value': self.feat_value[idx], + 'y': self.y[idx]} + + +def generate_dataloader(mini_batch_data, args): + from src.logger import logger + from torch.utils.data import DataLoader + logger.info(f"Begin to preprocessing dataset") + begin_time = time.time() + dataloader = DataLoader(LibsvmDataset(mini_batch_data), + batch_size=args.batch_size, + shuffle=True) + logger.info(f"Preprocessing dataset Done ! time_usage = {time.time() - begin_time}") + return dataloader + + +@exception_catcher +def model_selection(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run model_selection on UDF runtime with CPU only") + + mini_batch_data = json.loads(params["mini_batch"]) + budget = float(params["budget"]) + + from src.eva_engine.run_ms import RunModelSelection + + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + + data_loader = [dataloader, dataloader, dataloader] + + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + best_arch, best_arch_performance, time_usage, _, p1_trace_highest_score, p1_trace_highest_scored_models_id = \ + rms.select_model_online_clean( + budget=budget, + data_loader=data_loader, + only_phase1=False, + run_workers=1) + + return orjson.dumps( + {"best_arch": best_arch, + "best_arch_performance": best_arch_performance, + "time_usage": time_usage}).decode('utf-8') + + +@exception_catcher +def profiling_filtering_phase(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run profiling_filtering_phase CPU only") + + mini_batch_m = params["mini_batch"] + + from src.eva_engine.run_ms import RunModelSelection + + logger.info(f"begin run filtering phase at {os.getcwd()}, with {mini_batch_m}") + + mini_batch_data = json.loads(mini_batch_m) + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + data_loader = [dataloader, dataloader, dataloader] + + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + score_time_per_model = rms.profile_filtering(data_loader=data_loader) + + return orjson.dumps({"time": score_time_per_model}).decode('utf-8') + + +@exception_catcher +def profiling_refinement_phase(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run profiling_refinement_phase CPU only") + + mini_batch_m = params["mini_batch"] + + from src.eva_engine.run_ms import RunModelSelection + + mini_batch_data = json.loads(mini_batch_m) + + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + data_loader = [dataloader, dataloader, dataloader] + + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + train_time_per_epoch = rms.profile_refinement(data_loader=data_loader) + + return orjson.dumps({"time": train_time_per_epoch}).decode('utf-8') + + +@exception_catcher +def coordinator(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run coordinator") + # print (f"begin run coordinator") + + budget = float(params["budget"]) + score_time_per_model = float(params["score_time_per_model"]) + train_time_per_epoch = float(params["train_time_per_epoch"]) + only_phase1 = True if params["only_phase1"].lower() == "true" else False + + from src.eva_engine.run_ms import RunModelSelection + + logger.info(f"coordinator params: budget={budget}, " + f"score_time_per_model={score_time_per_model}, " + f"train_time_per_epoch={train_time_per_epoch}, " + f"only_phase1={only_phase1}") + + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + K, U, N = rms.coordination( + budget=budget, + score_time_per_model=score_time_per_model, + train_time_per_epoch=train_time_per_epoch, + only_phase1=only_phase1) + + return orjson.dumps( + {"k": K, "u": U, "n": N}).decode('utf-8') + + +@exception_catcher +def filtering_phase(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run filtering_phase CPU only") + + # mini_batch_m = params["mini_batch"] + n = int(params["n"]) + k = int(params["k"]) + + from src.eva_engine.run_ms import RunModelSelection + + # mini_batch_data = json.loads(mini_batch_m) + # dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + k_models, _, _, _ = rms.filtering_phase(N=n, K=k) + + return orjson.dumps({"k_models": k_models}).decode('utf-8') + + +@exception_catcher +def filtering_phase_dataLoader(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run filtering_phase CPU only") + # print (f"begin run filtering_phase CPU only") + + mini_batch_m = params["mini_batch"] + # print ("mini_batch_m: ", mini_batch_m) + + + n = int(params["n"]) + k = int(params["k"]) + + from src.eva_engine.run_ms import RunModelSelection + + mini_batch_data = json.loads(mini_batch_m) + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + k_models, _, _, _ = rms.filtering_phase(N=n, K=k, train_loader=dataloader) + + return orjson.dumps({"k_models": k_models}).decode('utf-8') + + +@exception_catcher +def refinement_phase(params: dict, args: Namespace): + mini_batch_m = params["mini_batch"] + return orjson.dumps( + {"k_models": "k_models"}).decode('utf-8') + + +@exception_catcher +def model_selection_workloads(params: dict, args: Namespace): + """ + Run filtering (explore N models) and refinement phase (refine K models) for benchmarking latency. + """ + + mini_batch_m = params["mini_batch"] + n = int(params["n"]) + k = int(params["k"]) + + from src.logger import logger + logger.info(f"begin run model_selection_workloads on CPU only, explore N={n} and K={k}") + + from src.eva_engine.run_ms import RunModelSelection + + mini_batch_data = json.loads(mini_batch_m) + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + k_models, _, _, _ = rms.filtering_phase(N=n, K=k, train_loader=dataloader) + best_arch, best_arch_performance, _ = rms.refinement_phase( + U=1, + k_models=k_models, + train_loader=dataloader, + valid_loader=dataloader) + + return orjson.dumps( + {"best_arch": best_arch, + "best_arch_performance": best_arch_performance, + }).decode('utf-8') + + +@exception_catcher +def test_io(params: dict, args: Namespace): + return orjson.dumps({"inputs are": json.dumps(params)}).decode('utf-8') + + +@exception_catcher +def model_selection_trails(params: dict, args: Namespace): + from src.logger import logger + logger.info(f"begin run model_selection_trails CPU + GPU") + + mini_batch_data = json.loads(params["mini_batch"]) + budget = float(params["budget"]) + + # 1. launch cache service + columns = list(mini_batch_data[0].keys()) + requests.post(args.cache_svc_url, + json={'columns': columns, 'name_space': "train", 'table_name': "dummy", + "batch_size": len(mini_batch_data)}) + requests.post(args.cache_svc_url, + json={'columns': columns, 'name_space': "valid", 'table_name': "dummy", + "batch_size": len(mini_batch_data)}) + + from src.eva_engine.run_ms import RunModelSelection + + # 2. profiling & coordination + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + data_loader = [dataloader, dataloader, dataloader] + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + + begin_time = time.time() + score_time_per_model = rms.profile_filtering(data_loader) + train_time_per_epoch = rms.profile_refinement(data_loader) + K, U, N = rms.coordination(budget, score_time_per_model, train_time_per_epoch, False) + + # 3. filtering + k_models, all_models, p1_trace_highest_score, p1_trace_highest_scored_models_id = rms.filtering_phase( + N, K, train_loader=data_loader[0]) + + # 4. Run refinement pahse + data = {'u': 1, 'k_models': k_models, "table_name": "dummy", "config_file": args.config_file} + response = requests.post(args.refinement_url, json=data).json() + + best_arch, best_arch_performance = response["best_arch"], response["best_arch_performance"] + + end_time = time.time() + real_time_usage = end_time - begin_time + + return orjson.dumps( + {"best_arch": best_arch, + "best_arch_performance": best_arch_performance, + "time_usage": real_time_usage}).decode('utf-8') + + +@exception_catcher +def model_selection_trails_workloads(params: dict, args: Namespace): + """ + Run filtering (explore N models) and refinement phase (refine K models) for benchmarking latency. + """ + + begin_time = time.time() + mini_batch_data = json.loads(params["mini_batch"]) + n = int(params["n"]) + k = int(params["k"]) + + # 1. launch cache service, for both train and valid. + # todo: use real data table or others + columns = list(mini_batch_data[0].keys()) + requests.post(args.cache_svc_url, + json={'columns': columns, 'name_space': "train", 'table_name': "dummy", + "batch_size": len(mini_batch_data)}) + requests.post(args.cache_svc_url, + json={'columns': columns, 'name_space': "valid", 'table_name': "dummy", + "batch_size": len(mini_batch_data)}) + + from src.logger import logger + logger.info(f"begin run model_selection_trails_workloads CPU + GPU, explore N={n} and K={k}") + + from src.eva_engine.run_ms import RunModelSelection + + # 2. filtering + dataloader = generate_dataloader(mini_batch_data=mini_batch_data, args=args) + rms = RunModelSelection(args.search_space, args, is_simulate=args.is_simulate) + k_models, _, _, _ = rms.filtering_phase(N=n, K=k, train_loader=dataloader) + + # 3. Run refinement pahse + data = {'u': 1, 'k_models': k_models, "table_name": "dummy", "config_file": args.config_file} + response = requests.post(args.refinement_url, json=data).json() + best_arch, best_arch_performance = response["best_arch"], response["best_arch_performance"] + real_time_usage = time.time() - begin_time + + return orjson.dumps( + {"best_arch": best_arch, + "best_arch_performance": best_arch_performance, + "time_usage": real_time_usage + }).decode('utf-8') + + +# benchmarking code here +@exception_catcher +def benchmark_filtering_phase_latency(params: dict, args: Namespace): + from src.logger import logger + from src.common.structure import ModelAcquireData + from src.controller.sampler_all.seq_sampler import SequenceSampler + from src.eva_engine.phase1.evaluator import P1Evaluator + from src.search_space.init_search_space import init_search_space + from src.tools.io_tools import write_json, read_json + from src.tools.res_measure import print_cpu_gpu_usage + import torch + logger.info(f"begin run filtering_phase CPU only") + + args.models_explore = int(params["explore_models"]) + + output_file = f"{args.result_dir}/score_{args.search_space}_{args.dataset}_batch_size_{args.batch_size}_{args.device}_{args.tfmem}.json" + time_output_file = f"{args.result_dir}/time_score_{args.search_space}_{args.dataset}_batch_size_{args.batch_size}_{args.device}_{args.tfmem}.json" + res_output_file = f"{args.result_dir}/resource_score_{args.search_space}_{args.dataset}_batch_size_{args.batch_size}_{args.device}_{args.tfmem}.json" + + # start the resource monitor + stop_event, thread = print_cpu_gpu_usage(interval=0.5, output_file=res_output_file) + + db_config = { + "db_name": args.db_name, + "db_user": args.db_user, + "db_host": args.db_host, + "db_port": args.db_port, + } + + search_space_ins = init_search_space(args) + _evaluator = P1Evaluator(device=args.device, + num_label=args.num_labels, + dataset_name=args.dataset, + search_space_ins=search_space_ins, + train_loader=None, + is_simulate=False, + metrics=args.tfmem, + enable_cache=args.embedding_cache_filtering, + db_config=db_config) + + sampler = SequenceSampler(search_space_ins) + explored_n = 0 + result = read_json(output_file) + print(f"begin to score all, currently we already explored {len(result.keys())}") + logger.info(f"begin to score all, currently we already explored {len(result.keys())}") + + while True: + arch_id, arch_micro = sampler.sample_next_arch() + if arch_id is None: + break + if arch_id in result: + continue + if explored_n > args.models_explore: + break + # run the model selection + model_encoding = search_space_ins.serialize_model_encoding(arch_micro) + model_acquire_data = ModelAcquireData(model_id=arch_id, + model_encoding=model_encoding, + is_last=False) + data_str = model_acquire_data.serialize_model() + model_score = _evaluator.p1_evaluate(data_str) + explored_n += 1 + result[arch_id] = model_score + if explored_n % 50 == 0: + logger.info(f"Evaluate {explored_n} models") + print(f"Evaluate {explored_n} models") + + if _evaluator.if_cuda_avaiable(): + torch.cuda.synchronize() + + # the first two are used for warming up + _evaluator.time_usage["io_latency"] = \ + sum(_evaluator.time_usage["track_io_model_load"][2:]) + \ + sum(_evaluator.time_usage["track_io_model_release_each_50"]) + \ + sum(_evaluator.time_usage["track_io_model_init"][2:]) + \ + sum(_evaluator.time_usage["track_io_res_load"][2:]) + \ + sum(_evaluator.time_usage["track_io_data_retrievel"][2:]) + \ + sum(_evaluator.time_usage["track_io_data_preprocess"][2:]) + + _evaluator.time_usage["compute_latency"] = sum(_evaluator.time_usage["track_compute"][2:]) + _evaluator.time_usage["latency"] = _evaluator.time_usage["io_latency"] + _evaluator.time_usage["compute_latency"] + + _evaluator.time_usage["avg_compute_latency"] = \ + _evaluator.time_usage["compute_latency"] \ + / len(_evaluator.time_usage["track_compute"][2:]) + + write_json(output_file, result) + # compute time + write_json(time_output_file, _evaluator.time_usage) + + # Then, at the end of your program, you can stop the thread: + print("Done, time sleep for 10 seconds") + # wait the resource montor flush + time.sleep(10) + stop_event.set() + thread.join() + + return orjson.dumps({"Write to": time_output_file}).decode('utf-8') + + +# Micro benchmarking filterting phaes +search_space_ins = None +_evaluator = None +sampler = None + + +@exception_catcher +def in_db_filtering_state_init(params: dict, args: Namespace): + global search_space_ins, _evaluator, sampler + from src.logger import logger + from src.controller.sampler_all.seq_sampler import SequenceSampler + from src.eva_engine.phase1.evaluator import P1Evaluator + from src.search_space.init_search_space import init_search_space + + db_config = { + "db_name": args.db_name, + "db_user": args.db_user, + "db_host": args.db_host, + "db_port": args.db_port, + } + + # init once + # params["eva_results"] == "null" means it a new job + if params["eva_results"] == "null" or (search_space_ins is None and _evaluator is None and sampler is None): + logger.info(f'New job = {params["eva_results"]}, search_space_ins = {search_space_ins}') + search_space_ins = init_search_space(args) + _evaluator = P1Evaluator(device=args.device, + num_label=args.num_labels, + dataset_name=params["dataset"], + search_space_ins=search_space_ins, + train_loader=None, + is_simulate=False, + metrics=args.tfmem, + enable_cache=args.embedding_cache_filtering, + db_config=db_config, + data_retrievel="spi") + sampler = SequenceSampler(search_space_ins) + + arch_id, arch_micro = sampler.sample_next_arch() + model_encoding = search_space_ins.serialize_model_encoding(arch_micro) + + return orjson.dumps({"model_encoding": model_encoding, "arch_id": arch_id}).decode('utf-8') + + +@exception_catcher +def in_db_filtering_evaluate(params: dict, args: Namespace): + global search_space_ins, _evaluator, sampler + from src.common.structure import ModelAcquireData + from src.logger import logger + try: + if search_space_ins is None and _evaluator is None and sampler is None: + logger.info("search_space_ins, _evaluator, sampler is None") + return orjson.dumps({"error": "erroed, plz call init first"}).decode('utf-8') + + sampled_result = json.loads(params["sample_result"]) + arch_id, model_encoding = str(sampled_result["arch_id"]), str(sampled_result["model_encoding"]) + + mini_batch = json.loads(params["mini_batch"]) + if mini_batch["status"] == "error": + return orjson.dumps({"error": mini_batch["message"]}).decode('utf-8') + logger.info(f"Begin evaluate {params['model_index']}, " + f"with size of batch = {len(mini_batch['data'])}, " + f"size of columns = {len(mini_batch['data'][0])}") + model_acquire_data = ModelAcquireData(model_id=arch_id, + model_encoding=model_encoding, + is_last=False, + spi_seconds=float(params["spi_seconds"]), + spi_mini_batch=mini_batch["data"], + ) + + model_score = _evaluator._p1_evaluate_online(model_acquire_data) + logger.info(f'Done evaluate {params["model_index"]}, ' + f'with {orjson.dumps({"index": params["model_index"], "score": model_score}).decode("utf-8")}') + except: + logger.info(orjson.dumps( + {"Errored": traceback.format_exc()}).decode('utf-8')) + + return orjson.dumps( + {"Errored": traceback.format_exc()}).decode('utf-8') + + return orjson.dumps({"index": params["model_index"], "score": model_score}).decode('utf-8') + + +@exception_catcher +def records_results(params: dict, args: Namespace): + global search_space_ins, _evaluator, sampler + from src.tools.io_tools import write_json + from src.logger import logger + + try: + time_output_file = f"{args.result_dir}/time_score_{args.search_space}_{params['dataset']}_batch_size_{args.batch_size}_{args.device}_{args.tfmem}.json" + _evaluator.time_usage["io_latency"] = \ + sum(_evaluator.time_usage["track_io_model_load"][2:]) + \ + sum(_evaluator.time_usage["track_io_model_release_each_50"]) + \ + sum(_evaluator.time_usage["track_io_model_init"][2:]) + \ + sum(_evaluator.time_usage["track_io_res_load"][2:]) + \ + sum(_evaluator.time_usage["track_io_data_retrievel"][2:]) + \ + sum(_evaluator.time_usage["track_io_data_preprocess"][2:]) + + _evaluator.time_usage["compute_latency"] = sum(_evaluator.time_usage["track_compute"][2:]) + _evaluator.time_usage["latency"] = _evaluator.time_usage["io_latency"] + _evaluator.time_usage[ + "compute_latency"] + + _evaluator.time_usage["avg_compute_latency"] = \ + _evaluator.time_usage["compute_latency"] \ + / len(_evaluator.time_usage["track_compute"][2:]) + + logger.info(f"Saving time usag to {time_output_file}") + # compute time + write_json(time_output_file, _evaluator.time_usage) + except: + logger.info(orjson.dumps( + {"Errored": traceback.format_exc()}).decode('utf-8')) + + return orjson.dumps( + {"Errored": traceback.format_exc()}).decode('utf-8') + + return orjson.dumps({"Done": 1}).decode('utf-8') + + +if __name__ == "__main__": + params = {} + params["budget"] = 10 + params["score_time_per_model"] = 0.0211558125 + params["train_time_per_epoch"] = 5.122203075885773 + params["only_phase1"] = 'true' + params["config_file"] = './internal/ml/model_selection/config.ini' + print(coordinator(json.dumps(params))) + + params = {} + params[ + "mini_batch"] = '[{"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}]' + params["n"] = 10 + params["k"] = 1 + params["config_file"] = './internal/ml/model_selection/config.ini' + print(filtering_phase_dataLoader(json.dumps(params))) + + # params = {} + # params[ + # "mini_batch"] = '[{"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}]' + # params["config_file"] = './internal/ml/model_selection/config.ini' + # print(profiling_refinement_phase(json.dumps(params))) + # + # params = {} + # params["budget"] = 10 + # params[ + # "mini_batch"] = '[{"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"1"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}, {"col1":"123:123","col2":"123:123","col3":"123:123","label":"0"}]' + # params["config_file"] = './internal/ml/model_selection/config.ini' + # print(model_selection(json.dumps(params))) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/requirement.txt b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/requirement.txt new file mode 100644 index 0000000000..591daefa59 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/requirement.txt @@ -0,0 +1,54 @@ +aiofiles==23.1.0 +blessed==1.20.0 +certifi==2023.7.22 +charset-normalizer==3.2.0 +ConfigSpace==0.7.1 +contourpy==1.1.0 +cycler==0.11.0 +fonttools==4.41.0 +gpustat==1.1 +html5tagger==1.3.0 +httptools==0.6.0 +idna==3.4 +importlib-resources==6.0.0 +joblib==1.3.1 +kiwisolver==1.4.4 +matplotlib==3.7.2 +more-itertools==9.1.0 +multidict==6.0.4 +numpy==1.24.4 +nvidia-ml-py==12.535.77 +orjson==3.9.2 +packaging==23.1 +palettable==3.3.3 +pandas==2.0.3 +Pillow==10.0.0 +psutil==5.9.5 +psycopg2-binary==2.9.6 +pyparsing==3.0.9 +python-dateutil==2.8.2 +pytz==2023.3 +requests==2.31.0 +sanic==23.6.0 +sanic-routing==23.6.0 +scikit-learn==1.3.0 +scipy==1.10.1 +seaborn==0.12.2 +six==1.16.0 +sklearn==0.0 +thop @ git+https://github.com/Lyken17/pytorch-OpCounter.git@43c064afb71383501e41eaef9e8c8407265cf77f +threadpoolctl==3.1.0 +torch==1.8.1 +torchaudio==0.8.1 +torchinfo==1.8.0 +torchvision==0.9.1 +tqdm==4.47.0 +tracerite==1.1.0 +typing_extensions==4.7.1 +tzdata==2023.3 +ujson==5.8.0 +urllib3==2.0.4 +uvloop==0.17.0 +wcwidth==0.2.6 +websockets==11.0.3 +zipp==3.16.2 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_img_w_baseline.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_img_w_baseline.sh new file mode 100644 index 0000000000..e54e823bc7 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_img_w_baseline.sh @@ -0,0 +1,42 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + +############## c10 dataset ############## +# run both 2phase-MS and training-free MS +python internal/ml/model_selection/exps/macro/anytime_img.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar10 \ + --num_labels 10 \ + --base_dir ../exp_data/ \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## c100 dataset ############## +python internal/ml/model_selection/exps/macro/anytime_img.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar100 \ + --num_labels 100 \ + --base_dir ../exp_data/ \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## imageNet dataset ############## +python internal/ml/model_selection/exps/macro/anytime_img.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset ImageNet16-120 \ + --num_labels 120 \ + --base_dir ../exp_data/ \ + --result_dir ./internal/ml/model_selection/exp_result/ + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_tab.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_tab.sh new file mode 100644 index 0000000000..9811b467f8 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/anytime_tab.sh @@ -0,0 +1,126 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +############## frappe dataset ############## + +# run the 2phase-MS +python internal/ml/model_selection/exps/macro/anytime_simulate.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --batch_size 128 \ + --nfeat 5500 \ + --nfield 10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset frappe \ + --num_labels 2 \ + --only_phase1 False \ + --is_simulate True \ + --device cpu \ + --log_folder any_time_frappe \ + --result_dir ./internal/ml/model_selection/exp_result/ \ + --num_points 5 + + +# run the training-free MS +python internal/ml/model_selection/exps/macro/anytime_simulate.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --batch_size 128 \ + --nfeat 5500 \ + --nfield 10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset frappe \ + --num_labels 2 \ + --only_phase1 True \ + --is_simulate True \ + --device cpu \ + --log_folder any_time_frappe \ + --result_dir ./internal/ml/model_selection/exp_result/ \ + --num_points 5 + + +############## uci dataset ############## + +# run the 2phase-MS +python internal/ml/model_selection/exps/macro/anytime_simulate.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --batch_size 128 \ + --nfeat 369 \ + --nfield 43 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset uci_diabetes \ + --num_labels 2 \ + --only_phase1 False \ + --is_simulate True \ + --device cpu \ + --log_folder any_time_uci_diabetes \ + --result_dir ./internal/ml/model_selection/exp_result/ \ + --num_points 5 + + +# run the training-free MS +python internal/ml/model_selection/exps/macro/anytime_simulate.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --batch_size 128 \ + --nfeat 369 \ + --nfield 43 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset uci_diabetes \ + --num_labels 2 \ + --only_phase1 True \ + --is_simulate True \ + --device cpu \ + --log_folder any_time_uci_diabetes \ + --result_dir ./internal/ml/model_selection/exp_result/ \ + --num_points 5 + + +############## criteo dataset ############## + +# run the 2phase-MS +python internal/ml/model_selection/exps/macro/anytime_simulate.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 10 \ + --batch_size 128 \ + --nfeat 2100000 \ + --nfield 39 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset criteo \ + --num_labels 2 \ + --only_phase1 False \ + --is_simulate True \ + --device cpu \ + --log_folder any_time_criteo \ + --result_dir ./internal/ml/model_selection/exp_result/ \ + --num_points 5 + + +# run the training-free MS +python internal/ml/model_selection/exps/macro/anytime_simulate.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 10 \ + --batch_size 128 \ + --nfeat 2100000 \ + --nfield 39 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset criteo \ + --num_labels 2 \ + --only_phase1 True \ + --is_simulate True \ + --device cpu \ + --log_folder any_time_criteo \ + --result_dir ./internal/ml/model_selection/exp_result/ \ + --num_points 5 + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_img.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_img.sh new file mode 100644 index 0000000000..52467c9b70 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_img.sh @@ -0,0 +1,45 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + +# run both training-based MS +############## c10 dataset ############## +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar10 \ + --num_labels 10 \ + --base_dir ../exp_data/ \ + --log_folder log_baseline_c10 \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## c100 dataset ############## +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar100 \ + --num_labels 100 \ + --base_dir ../exp_data/ \ + --log_folder log_baseline_c100 \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## ImgNet dataset ############## +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset ImageNet16-120 \ + --num_labels 120 \ + --base_dir ../exp_data/ \ + --log_folder log_baseline_imgnet \ + --result_dir ./internal/ml/model_selection/exp_result/ + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_tab.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_tab.sh new file mode 100644 index 0000000000..c1f6d19250 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/baseline_system_tab.sh @@ -0,0 +1,67 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + +# run both training-based MS +############## frappe dataset ############## +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --epoch 19 \ + --batch_size=512 \ + --lr=0.001 \ + --iter_per_epoch=200 \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --base_dir ../exp_data/ \ + --dataset frappe \ + --num_labels 2 \ + --device=cpu \ + --log_folder baseline_frappe \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## uci dataset ############## +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --epoch 0 \ + --batch_size=1024 \ + --lr=0.001 \ + --iter_per_epoch=200 \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --base_dir ../exp_data/ \ + --dataset uci_diabetes \ + --num_labels 2 \ + --device=cpu \ + --log_folder baseline_uci_diabetes \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## criteo dataset ############## +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 10 \ + --epoch 9 \ + --batch_size=1024 \ + --lr=0.001 \ + --iter_per_epoch=2000 \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --base_dir ../exp_data/ \ + --dataset criteo \ + --num_labels 2 \ + --device=cpu \ + --log_folder baseline_criteo \ + --result_dir ./internal/ml/model_selection/exp_result/ + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/benchmark_weight_sharing.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/benchmark_weight_sharing.sh new file mode 100644 index 0000000000..7b13ac8c1b --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/benchmark_weight_sharing.sh @@ -0,0 +1,25 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +python ./internal/ml/model_selection/exps/micro/resp/benchmark_weight_sharing.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=512 \ + --lr=0.001 \ + --epoch=20 \ + --iter_per_epoch=200 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --log_folder=log_frappe \ + --total_models_per_worker=-1 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/database/load_data_to_db.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/database/load_data_to_db.sh new file mode 100644 index 0000000000..b7555bbe7d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/database/load_data_to_db.sh @@ -0,0 +1,67 @@ +#!/bin/bash + +# Check for proper number of command line args +if [[ $# -ne 2 ]]; then + echo "Usage: $0 " + exit 1 +fi + +# Configurations +DATA_PATH="$1" +DB_NAME="$2" + +# Connection details +HOST="localhost" +PORT="28814" +USERNAME="postgres" +DBNAME="pg_extension" + +# Create the database +echo "Creating database..." +createdb -h $HOST -p $PORT -U $USERNAME $DBNAME + +# Define datasets to process +datasets=("train" "valid" "test") + +# Loop over each dataset +for dataset in "${datasets[@]}"; do + rm "${DATA_PATH}/${dataset}.csv" + + # 1. Identify the number of columns + num_columns=$(awk 'NF > max { max = NF } END { print max }' "${DATA_PATH}/${dataset}.libsvm") + + # 2. Create the table dynamically + create_table_cmd="CREATE TABLE ${DB_NAME}_${dataset} (id SERIAL PRIMARY KEY, label INTEGER" + + for (( i=2; i<=$num_columns; i++ )); do + create_table_cmd+=", col$(($i-1)) TEXT" + done + create_table_cmd+=");" + + echo "Creating ${dataset} table..." + echo $create_table_cmd | psql -h $HOST -p $PORT -U $USERNAME -d $DBNAME + + # 3. Transform the libsvm format to CSV + echo "Transforming ${dataset} to CSV format..." + + awk '{ + for (i = 1; i <= NF; i++) { + printf "%s", $i; # print each field as-is + if (i < NF) { + printf " "; # if its not the last field, print a space + } + } + printf "\n"; # end of line + }' "${DATA_PATH}/${dataset}.libsvm" > "${DATA_PATH}/${dataset}.csv" + + # 4. Import into PostgreSQL + columns="label" + for (( i=2; i<=$num_columns; i++ )); do + columns+=", col$(($i-1))" + done + + echo "Loading ${dataset} into PostgreSQL..." + psql -h $HOST -p $PORT -U $USERNAME -d $DBNAME -c "\COPY ${DB_NAME}_${dataset}($columns) FROM '${DATA_PATH}/${dataset}.csv' DELIMITER ' '" +done + +echo "Data load complete." diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache.sh new file mode 100644 index 0000000000..29390b0bbc --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache.sh @@ -0,0 +1,122 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ + +########################## CPU ############################## +# this is run on cpu, only change the device==cpu for all above + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache_concurrent.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache_concurrent.sh new file mode 100644 index 0000000000..993e8a2a8d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_embedding_cache_concurrent.sh @@ -0,0 +1,139 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + +########################## CPU ############################## +# this is run on cpu, only change the device==cpu for all above + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_concurrent.py \ + --concurrency=8 \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_current_filter_cache/ \ + --log_folder=log_score_time_frappe_cache + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_concurrent.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_current_filter_cache/ \ + --log_folder=log_score_time_frappe_cache + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_concurrent.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_current_filter_cache/ \ + --log_folder=log_score_time_frappe_cache + + +# here is concurrent run but no embedding cache +####################################################################################### + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_concurrent.py \ + --tfmem=express_flow \ + --models_explore=5000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_current_filter_no_cache/ \ + --log_folder=log_score_time_frappe_cache + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_concurrent.py \ + --tfmem=express_flow \ + --models_explore=5000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_current_filter_no_cache/ \ + --log_folder=log_score_time_frappe_cache + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_concurrent.py \ + --tfmem=express_flow \ + --models_explore=5000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_filter_exp_current_filter_no_cachecache/ \ + --log_folder=log_score_time_frappe_cache + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_cpu_gpu.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_cpu_gpu.sh new file mode 100644 index 0000000000..c7c6a0be48 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_cpu_gpu.sh @@ -0,0 +1,211 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + + +# cifar 10 +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=10 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=cifar10 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + + +# cifar 100 +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=100 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=cifar100 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + + +# imageNet +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=120 \ + --device=cuda:0 \ + --batch_size=32 \ + --dataset=ImageNet16-120 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + +########################## CPU ############################## +# this is run on cpu, only change the device==cpu for all above + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + +# criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + + +# cifar 10 +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=10 \ + --device=cpu \ + --batch_size=32 \ + --dataset=cifar10 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + + +# cifar 100 +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=100 \ + --device=cpu \ + --batch_size=32 \ + --dataset=cifar100 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ + + +# imageNet +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency.py \ + --embedding_cache_filtering=False \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=120 \ + --device=cpu \ + --batch_size=32 \ + --dataset=ImageNet16-120 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_wo_cache/ diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_in_db.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_in_db.sh new file mode 100644 index 0000000000..608fd94b69 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase1_in_db.sh @@ -0,0 +1,61 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_sql.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_cache_sql/ + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_sql.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_cache_sql/ + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_sql.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_cache_sql/ diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase2.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase2.sh new file mode 100644 index 0000000000..608fd94b69 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/latency_phase2.sh @@ -0,0 +1,61 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +# frappe +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_sql.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_cache_sql/ + +#criteo +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_sql.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_cache_sql/ + +# uci +python3 ./internal/ml/model_selection/exps/micro/benchmark_filtering_latency_sql.py \ + --embedding_cache_filtering=True \ + --tfmem=express_flow \ + --models_explore=5000 \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result_sever_cache_sql/ diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_budget_aware_alg.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_budget_aware_alg.sh new file mode 100644 index 0000000000..33e9b9de5c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_budget_aware_alg.sh @@ -0,0 +1,44 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + +############## c10 dataset ############## +# run both 2phase-MS and training-free MS +python internal/ml/model_selection/exps/micro/benchmark_budget_aware_alg.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --dataset cifar10 \ + --epoch 200 \ + --base_dir ../exp_data/ \ + --log_name logs_default \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## c100 dataset ############## +python internal/ml/model_selection/exps/micro/benchmark_budget_aware_alg.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --dataset cifar100 \ + --epoch 200 \ + --base_dir ../exp_data/ \ + --log_name logs_default \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## imageNet dataset ############## +python internal/ml/model_selection/exps/micro/benchmark_budget_aware_alg.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --dataset ImageNet16-120 \ + --epoch 200 \ + --base_dir ../exp_data/ \ + --log_name logs_default \ + --result_dir ./internal/ml/model_selection/exp_result/ + + + +############## draw graphs ############## +python internal/ml/model_selection/exps/micro/draw_budget_aware_alg.py diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_nku_tradeoff.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_nku_tradeoff.sh new file mode 100644 index 0000000000..ad392f625e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_nku_tradeoff.sh @@ -0,0 +1,163 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + + +# ==================================== +# ==================================== +# determine the K and U tradeoff +# ==================================== +# ==================================== +# frappe +python internal/ml/model_selection/exps/micro/benchmark_ku.py \ + --search_space mlp_sp \ + --epoch 20 \ + --hidden_choice_len 20 \ + --dataset frappe \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# uci +python internal/ml/model_selection/exps/micro/benchmark_ku.py \ + --search_space mlp_sp \ + --hidden_choice_len 20 \ + --epoch 5 \ + --dataset uci_diabetes \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + +# criteo +python internal/ml/model_selection/exps/micro/benchmark_ku.py \ + --search_space mlp_sp \ + --hidden_choice_len 10 \ + --epoch 10 \ + --dataset criteo \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# c10 +python internal/ml/model_selection/exps/micro/benchmark_ku.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar10 \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# c100 +python internal/ml/model_selection/exps/micro/benchmark_ku.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar100 \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# imageNet +python internal/ml/model_selection/exps/micro/benchmark_ku.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset ImageNet16-120 \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + + +# ==================================== +# ==================================== +# determine the K and U tradeoff +# ==================================== +# ==================================== + + +python internal/ml/model_selection/exps/micro/benchmark_nk.py \ + --search_space mlp_sp \ + --epoch 20 \ + --hidden_choice_len 20 \ + --dataset frappe \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +#uci +python internal/ml/model_selection/exps/micro/benchmark_nk.py \ + --search_space mlp_sp \ + --hidden_choice_len 20 \ + --epoch 5 \ + --dataset uci_diabetes \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# criteo +python internal/ml/model_selection/exps/micro/benchmark_nk.py \ + --search_space mlp_sp \ + --hidden_choice_len 10 \ + --epoch 10 \ + --dataset criteo \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + + +# c10 +python internal/ml/model_selection/exps/micro/benchmark_nk.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar10 \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# c100 +python internal/ml/model_selection/exps/micro/benchmark_nk.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset cifar100 \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + + +# imageNet +python internal/ml/model_selection/exps/micro/benchmark_nk.py \ + --search_space nasbench201 \ + --api_loc NAS-Bench-201-v1_1-096897.pth \ + --epoch 200 \ + --dataset ImageNet16-120 \ + --base_dir ../exp_data/ \ + --only_phase1 True \ + --is_simulate True \ + --log_folder log_ku_tradeoff + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_score_metrics_relation.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_score_metrics_relation.sh new file mode 100644 index 0000000000..1593cfe084 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_score_metrics_relation.sh @@ -0,0 +1,38 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +############## Frappe ############## +# run both 2phase-MS and training-free MS +python ./internal/ml/model_selection/exps/micro/benchmark_score_metrics.py \ + --tfmem=express_flow \ + --search_space mlp_sp \ + --dataset frappe \ + --base_dir ../exp_data/ \ + --log_name logs_default \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## criteo dataset ############## +python ./internal/ml/model_selection/exps/micro/benchmark_score_metrics.py \ + --tfmem=express_flow \ + --search_space mlp_sp \ + --dataset criteo \ + --base_dir ../exp_data/ \ + --log_name logs_default \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## Uci dataset ############## +python ./internal/ml/model_selection/exps/micro/benchmark_score_metrics.py \ + --tfmem=express_flow \ + --search_space=mlp_sp \ + --dataset uci_diabetes \ + --base_dir ../exp_data/ \ + --log_name logs_default \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +############## draw graphs ############## +python ./internal/ml/model_selection/exps/micro/draw_score_metric_relation.py diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_search_strategy.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_search_strategy.sh new file mode 100644 index 0000000000..4bdb01129f --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/micro_search_strategy.sh @@ -0,0 +1,54 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + +# rs +python internal/ml/model_selection/exps/baseline/train_with_random.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --epoch 19 \ + --batch_size=512 \ + --lr=0.001 \ + --iter_per_epoch=200 \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --base_dir ../exp_data/ \ + --dataset frappe \ + --num_labels 2 \ + --device=cpu \ + --log_folder baseline_frappe \ + --result_dir ./internal/ml/model_selection/exp_result/ + + +# rl +python internal/ml/model_selection/exps/baseline/train_with_rl.py + + +# re +python internal/ml/model_selection/exps/baseline/train_with_ea.py \ + --search_space mlp_sp \ + --num_layers 4 \ + --hidden_choice_len 20 \ + --epoch 19 \ + --batch_size=512 \ + --lr=0.001 \ + --iter_per_epoch=200 \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --base_dir ../exp_data/ \ + --dataset frappe \ + --num_labels 2 \ + --device=cpu \ + --log_folder baseline_frappe \ + --result_dir ./internal/ml/model_selection/exp_result/ + +# bohb +python internal/ml/model_selection/exps/baseline/train_bohb.py + +############## draw the graph ############## +python internal/ml/model_selection/exps/baseline/draw_benchmark_train_based.py --dataset frappe + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/convert_api_2_json.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/convert_api_2_json.sh new file mode 100644 index 0000000000..1c166f68fa --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/convert_api_2_json.sh @@ -0,0 +1,12 @@ + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +# pip install nats_bench + +python internal/ml/model_selection/exps/nas_bench_img/0_characterize_gt.py +python internal/ml/model_selection/exps/nas_bench_img/0_parse_testacc_101.py +python internal/ml/model_selection/exps/nas_bench_img/0_parse_testacc_201.py + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/explore_all_models.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/explore_all_models.sh new file mode 100644 index 0000000000..d700c2baaa --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/explore_all_models.sh @@ -0,0 +1,60 @@ + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +# cifar10 + nb101 +python ./internal/ml/model_selection/exps/nas_bench_img/1_explore_models_100_run.py \ + --search_space=nasbench101 \ + --api_loc=nasbench_only108.pkl \ + --base_dir=../exp_data/ \ + --dataset=cifar10 \ + --num_labels=10 \ + --device=cpu \ + --log_folder=log_img_explore_ea \ + --result_dir=./internal/ml/model_selection/exp_result/ + + +# cifar10 + nb201 +python ./internal/ml/model_selection/exps/nas_bench_img/1_explore_models_100_run.py \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=../exp_data/ \ + --dataset=cifar10 \ + --init_channels=16 \ + --num_stacks=3 \ + --num_modules_per_stack=3 \ + --num_labels=10 \ + --device=cpu \ + --log_folder=log_img_explore_ea \ + --result_dir=./internal/ml/model_selection/exp_result/ + + +# cifar100 + nb201 +python ./internal/ml/model_selection/exps/nas_bench_img/1_explore_models_100_run.py \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=../exp_data/ \ + --dataset=cifar100 \ + --init_channels=16 \ + --num_stacks=3 \ + --num_modules_per_stack=3 \ + --num_labels=100 \ + --device=cpu \ + --log_folder=log_img_explore_ea \ + --result_dir=./internal/ml/model_selection/exp_result/ + + +# imgnet + nb201 +python ./internal/ml/model_selection/exps/nas_bench_img/1_explore_models_100_run.py \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=../exp_data/ \ + --dataset=ImageNet16-120 \ + --init_channels=16 \ + --num_stacks=3 \ + --num_modules_per_stack=3 \ + --num_labels=120 \ + --device=cpu \ + --log_folder=log_img_explore_ea \ + --result_dir=./internal/ml/model_selection/exp_result/ diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/score_all_models.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/score_all_models.sh new file mode 100644 index 0000000000..d4c18386bd --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-img/score_all_models.sh @@ -0,0 +1,60 @@ + + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +for i in {1..4} +do + # cifar10 + nb101 +# /home/xingnaili/miniconda3/envs/trails/bin/python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ +# --models_explore=1200 \ +# --search_space=nasbench101 \ +# --api_loc=nasbench_only108.pkl \ +# --base_dir=/hdd1/xingnaili/exp_data/ \ +# --dataset=cifar10 \ +# --batch_size=32 \ +# --num_labels=10 \ +# --device=cuda:0 \ +# --log_folder=log_score_all_img10_101 \ +# --result_dir=./internal/ml/model_selection/exp_result/ + + # cifar10 + nb201 + /home/xingnaili/miniconda3/envs/trails/bin/python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --models_explore=1200 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset=cifar10 \ + --batch_size=32 \ + --num_labels=10 \ + --device=cpu \ + --log_folder=log_score_all_img10 \ + --result_dir=./internal/ml/model_selection/exp_result/ + + # cifar100 + nb201 + /home/xingnaili/miniconda3/envs/trails/bin/python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --models_explore=1200 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset=cifar100 \ + --batch_size=32 \ + --num_labels=100 \ + --device=cpu \ + --log_folder=log_score_all_img100 \ + --result_dir=./internal/ml/model_selection/exp_result/ + + # imgnet + nb201 + /home/xingnaili/miniconda3/envs/trails/bin/python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --models_explore=1200 \ + --search_space=nasbench201 \ + --api_loc=NAS-Bench-201-v1_1-096897.pth \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --dataset=ImageNet16-120 \ + --batch_size=32 \ + --num_labels=120 \ + --device=cpu \ + --log_folder=log_score_all_img_imgnet \ + --result_dir=./internal/ml/model_selection/exp_result/ +done \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_criteo.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_criteo.sh new file mode 100644 index 0000000000..fd1a9be75b --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_criteo.sh @@ -0,0 +1,27 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + + +nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --embedding_cache_filtering=True \ + --models_explore=9999 \ + --tfmem=express_flow \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_score_time_criteo > outputCriScorAll.log& + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_frappe.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_frappe.sh new file mode 100644 index 0000000000..80cc39e49a --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_frappe.sh @@ -0,0 +1,27 @@ + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --embedding_cache_filtering=True \ + --models_explore=159999 \ + --tfmem=express_flow \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_score_time_frappe > output.log& + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_uci.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_uci.sh new file mode 100644 index 0000000000..9b910c1ac9 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/score_all_modesl_uci.sh @@ -0,0 +1,28 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --embedding_cache_filtering=True \ + --models_explore=159999 \ + --tfmem=express_flow \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cpu \ + --batch_size=32 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_score_time_uci > outputUciScoreALl.log& + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo.sh new file mode 100644 index 0000000000..8c25834738 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo.sh @@ -0,0 +1,46 @@ + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +worker_id=0 +GPU_NUM=9 +worker_each_gpu=6 +total_workers=$((worker_each_gpu*GPU_NUM)) + +for((gpu_id=0; gpu_id < GPU_NUM; ++gpu_id)); do +# echo "GPU id is $gpu_id" + for((i=0; i < worker_each_gpu; ++i)); do + echo "Assign task to worker id is $worker_id" + echo "nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:$gpu_id \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=10 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --worker_id=$worker_id \ + --total_workers=$total_workers \ + --workers=0 \ + --log_folder=log_train_criteo \ + --total_models_per_worker=-1 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --pre_partitioned_file=./internal/ml/model_selection/exps/nas_bench_tabular/sampled_models_10000_models.json & ">> train_all_models_criteo_seq.sh + +# sleep 1 + worker_id=$((worker_id+1)) + done +done + + +# pkill -9 -f 2.seq_train_online.py +# run with bash internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo.sh >criteobash & diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo_distirbuted.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo_distirbuted.sh new file mode 100644 index 0000000000..a0de85826c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_criteo_distirbuted.sh @@ -0,0 +1,48 @@ + + +# frappe +python exps/main_v2/ground_truth/2.seq_train_dist_online.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=../exp_data/ \ + --num_labels=1 \ + --device=gpu \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=10 \ + --iter_per_epoch=100 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --total_models_per_worker=10 \ + --workers=0 \ + --worker_each_gpu=1 \ + --gpu_num=8 \ + --log_folder=LogFrappee \ + --pre_partitioned_file=./exps/main_v2/ground_truth/sampled_models_10000_models.json & + +# criteo +python exps/main_v2/ground_truth/2.seq_train_dist_online.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=../exp_data/ \ + --num_labels=1 \ + --device=gpu \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=10 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --worker_each_gpu=9 \ + --gpu_num=8 \ + --log_folder=LogCriteo \ + --pre_partitioned_file=./exps/main_v2/ground_truth/sampled_models_10000_models.json & diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_diabetes.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_diabetes.sh new file mode 100644 index 0000000000..4f4c099d09 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_diabetes.sh @@ -0,0 +1,47 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +worker_id=0 +GPU_NUM=8 +worker_each_gpu=4 +total_workers=$((worker_each_gpu*GPU_NUM)) + +for((gpu_id=0; gpu_id < GPU_NUM; ++gpu_id)); do + for((i=0; i < worker_each_gpu; ++i)); do + + echo "nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:$gpu_id \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=40 \ + --iter_per_epoch=200 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --worker_id=$worker_id \ + --total_workers=$total_workers \ + --workers=0 \ + --log_folder=log_train_uci \ + --total_models_per_worker=-1 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --pre_partitioned_file=./internal/ml/model_selection/exps/nas_bench_tabular/uci_left_8k_models.json > outputuci.log& ">> train_all_models_diabetes_seq.sh + + worker_id=$((worker_id+1)) + done +done + + +# pkill -9 -f ./internal/ml/model_selection/exps/nas_bench_tabular//2.seq_train_online.py +# pkill -9 -f /home/naili/miniconda3/envs/firmest_torch11/bin/python + +# run with bash internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_diabetes.sh >ucibash & diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_frappe.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_frappe.sh new file mode 100644 index 0000000000..3ffd7bc64c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_frappe.sh @@ -0,0 +1,45 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +worker_id=0 +GPU_NUM=8 +worker_each_gpu=16 +total_workers=$((worker_each_gpu*GPU_NUM)) + +for((gpu_id=0; gpu_id < GPU_NUM; ++gpu_id)); do +# echo "GPU id is $gpu_id" + for((i=0; i < worker_each_gpu; ++i)); do + echo "nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/2.seq_train_online.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/home/shaofeng/naili/firmest_data/ \ + --num_labels=2 \ + --device=cuda:$gpu_id \ + --batch_size=512 \ + --lr=0.001 \ + --epoch=20 \ + --iter_per_epoch=200 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --worker_id=$worker_id \ + --total_workers=$total_workers \ + --workers=0 \ + --log_folder=log_frappe \ + --total_models_per_worker=-1 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --pre_partitioned_file=./internal/ml/model_selection/exps/nas_bench_tabular/sampled_models_all.json & ">> train_all_models_frappe_seq.sh + + sleep 1 + worker_id=$((worker_id+1)) + done +done + + +# pkill -9 -f internal/ml/model_selection/scripts/nas-bench-tabular/train_all_models_frappe.sh diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_one_model_dev.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_one_model_dev.sh new file mode 100644 index 0000000000..5115e874b3 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_one_model_dev.sh @@ -0,0 +1,25 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=512 \ + --lr=0.001 \ + --epoch=20 \ + --iter_per_epoch=200 \ + --dataset=frappe \ + --nfeat=5500 \ + --nfield=10 \ + --nemb=10 \ + --worker_id=0 \ + --total_workers=1 \ + --workers=1 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_frappe \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_criteo.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_criteo.sh new file mode 100644 index 0000000000..ea72391690 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_criteo.sh @@ -0,0 +1,144 @@ +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + +# default setting. +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=5 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_5.log & + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=10 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_10.log & + + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:1 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=20 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_20.log & + + + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:2 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=40 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_40.log & + + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:3 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=60 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_60.log & + + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:4 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=80 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_80.log & + + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:5 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=100 \ + --iter_per_epoch=2000 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_criteo_train_tune >criteo_100.log & + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_diabetes.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_diabetes.sh new file mode 100644 index 0000000000..661e6b9447 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/nas-bench-tabular/train_params_tune_diabetes.sh @@ -0,0 +1,70 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + +nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:0 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=3 \ + --iter_per_epoch=200 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_uci_train_tune >uci_3.log & + + + +nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:1 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=5 \ + --iter_per_epoch=200 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_uci_train_tune >uci_5.log & + + +# default setting. +nohup python ./internal/ml/model_selection/exps/nas_bench_tabular/0.train_one_model.py \ + --log_name=baseline_train_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=20 \ + --base_dir=../exp_data/ \ + --num_labels=2 \ + --device=cuda:2 \ + --batch_size=1024 \ + --lr=0.001 \ + --epoch=7 \ + --iter_per_epoch=200 \ + --dataset=uci_diabetes \ + --nfeat=369 \ + --nfield=43 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_uci_train_tune >uci_7.log & + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/pre_processing/pre_processing_data.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/pre_processing/pre_processing_data.sh new file mode 100644 index 0000000000..64b011c63e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/scripts/pre_processing/pre_processing_data.sh @@ -0,0 +1,47 @@ + + +export PYTHONPATH=$PYTHONPATH:./internal/ml/model_selection +conda activate trails + + + + +python ./internal/ml/model_selection/exps/nas_bench_tabular/4.seq_score_online.py \ + --models_explore=1000 \ + --log_name=score_based \ + --search_space=mlp_sp \ + --num_layers=4 \ + --hidden_choice_len=10 \ + --base_dir=/hdd1/xingnaili/exp_data/ \ + --num_labels=2 \ + --device=cuda:6 \ + --batch_size=32 \ + --dataset=criteo \ + --nfeat=2100000 \ + --nfield=39 \ + --nemb=10 \ + --workers=0 \ + --result_dir=./internal/ml/model_selection/exp_result/ \ + --log_folder=log_score_time_criteo > outputCriScorAll.log& + + + + + + + + + + + + + + + + + + + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/shared_config.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/shared_config.py new file mode 100644 index 0000000000..6731749976 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/shared_config.py @@ -0,0 +1,94 @@ +import calendar +import os +import time +import argparse +import configparser + + +def parse_config_arguments(config_path: str): + parser = configparser.ConfigParser() + parser.read(config_path) + + args = argparse.Namespace() + + # job config under DEFAULT + args.log_name = parser.get('DEFAULT', 'log_name') + args.budget = parser.getint('DEFAULT', 'budget') + args.device = parser.get('DEFAULT', 'device') + args.log_folder = parser.get('DEFAULT', 'log_folder') + args.result_dir = parser.get('DEFAULT', 'result_dir') + args.num_points = parser.getint('DEFAULT', 'num_points') + args.max_load = parser.getint('DEFAULT', 'max_load') + + # sampler args + args.search_space = parser.get('SAMPLER', 'search_space') + args.population_size = parser.getint('SAMPLER', 'population_size') + args.sample_size = parser.getint('SAMPLER', 'sample_size') + args.simple_score_sum = parser.getboolean('SAMPLER', 'simple_score_sum') + + # nb101 args + args.api_loc = parser.get('NB101', 'api_loc') + args.init_channels = parser.getint('NB101', 'init_channels') + args.bn = parser.getint('NB101', 'bn') + args.num_stacks = parser.getint('NB101', 'num_stacks') + args.num_modules_per_stack = parser.getint('NB101', 'num_modules_per_stack') + + # nb201 args + args.init_w_type = parser.get('NB201', 'init_w_type') + args.init_b_type = parser.get('NB201', 'init_b_type') + args.arch_size = parser.getint('NB201', 'arch_size') + + # mlp args + args.num_layers = parser.getint('MLP', 'num_layers') + args.hidden_choice_len = parser.getint('MLP', 'hidden_choice_len') + + # mlp_trainer args + args.epoch = parser.getint('MLP_TRAINER', 'epoch') + args.batch_size = parser.getint('MLP_TRAINER', 'batch_size') + args.lr = parser.getfloat('MLP_TRAINER', 'lr') + args.patience = parser.getint('MLP_TRAINER', 'patience') + args.iter_per_epoch = parser.getint('MLP_TRAINER', 'iter_per_epoch') + args.nfeat = parser.getint('MLP_TRAINER', 'nfeat') + args.nfield = parser.getint('MLP_TRAINER', 'nfield') + args.nemb = parser.getint('MLP_TRAINER', 'nemb') + args.report_freq = parser.getint('MLP_TRAINER', 'report_freq') + args.workers = parser.getint('MLP_TRAINER', 'workers') + + # dataset args + args.base_dir = parser.get('DATASET', 'base_dir') + args.dataset = parser.get('DATASET', 'dataset') + args.num_labels = parser.getint('DATASET', 'num_labels') + + # seq_train args + args.worker_id = parser.getint('SEQ_TRAIN', 'worker_id') + args.total_workers = parser.getint('SEQ_TRAIN', 'total_workers') + args.total_models_per_worker = parser.getint('SEQ_TRAIN', 'total_models_per_worker') + args.pre_partitioned_file = parser.get('SEQ_TRAIN', 'pre_partitioned_file') + + # dis_train args + args.worker_each_gpu = parser.getint('DIS_TRAIN', 'worker_each_gpu') + args.gpu_num = parser.getint('DIS_TRAIN', 'gpu_num') + + # tune_interval args + args.kn_rate = parser.getint('TUNE_INTERVAL', 'kn_rate') + + # anytime args + args.only_phase1 = parser.getboolean('ANYTIME', 'only_phase1') + args.is_simulate = parser.getboolean('ANYTIME', 'is_simulate') + + # system performance exps + args.models_explore = parser.getint('SYS_PERFORMANCE', 'models_explore') + args.tfmem = parser.get('SYS_PERFORMANCE', 'tfmem') + args.embedding_cache_filtering = parser.getboolean('SYS_PERFORMANCE', 'embedding_cache_filtering') + args.concurrency = parser.getint('SYS_PERFORMANCE', 'concurrency') + + args.refinement_url = parser.get('SERVER', 'refinement_url') + args.cache_svc_url = parser.get('SERVER', 'cache_svc_url') + + # db config + args.db_name = parser.get('DB_CONFIG', 'db_name') + args.db_user = parser.get('DB_CONFIG', 'db_user') + args.db_host = parser.get('DB_CONFIG', 'db_host') + args.db_port = parser.get('DB_CONFIG', 'db_port') + + return args \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/constant.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/constant.py new file mode 100644 index 0000000000..4d91cd3665 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/constant.py @@ -0,0 +1,64 @@ + + +class CommonVars: + + # SAMPLER + TEST_SAMPLER = "sequence" + RANDOM_SAMPLER = "random" + RL_SAMPLER = "rl" + EA_SAMPLER = "ea" + BOHB_SAMPLER = "bohb" + + # EVALUATOR + ExpressFlow = "express_flow" + + GRAD_NORM = "grad_norm" + GRAD_PLAIN = "grad_plain" + + JACOB_CONV = "jacob_conv" + NAS_WOT = "nas_wot" + + NTK_CONDNUM = "ntk_cond_num" + NTK_TRACE = "ntk_trace" + NTK_TRACE_APPROX = "ntk_trace_approx" + + PRUNE_FISHER = "fisher" + PRUNE_GRASP = "grasp" + PRUNE_SNIP = "snip" + PRUNE_SYNFLOW = "synflow" + + WEIGHT_NORM = "weight_norm" + + ALL_EVALUATOR = "all_matrix" + + # SEARCH SPACE + NASBENCH101 = "nas-bench-101" + NASBENCH201 = "nas-bench-201" + + # correlation coefficient metrics + KendallTau = "KendallTau" + Spearman = "Spearman" + Pearson = "Pearson" + AvgCorrelation = "average_correlation" + AllCorrelation = "all_correlation" + + +class Config: + + MLPSP = "mlp_sp" + NB101 = "nasbench101" + NB201 = "nasbench201" + DARTS = "darts" + NDS = "NDS" + + # vision dataset + c10_valid = "cifar10-valid" + c10 = "cifar10" + c100 = "cifar100" + imgNet = "ImageNet16-120" + + # struct dataset + Frappe = "frappe" + Criteo = "criteo" + UCIDataset = "uci_diabetes" + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/structure.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/structure.py new file mode 100644 index 0000000000..cf8f30e9e2 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/common/structure.py @@ -0,0 +1,89 @@ +import json + + +class ModelEvaData: + """ + Eva worker send score to search strategy + """ + + def __init__(self, model_id: str = None, model_score: dict = None): + if model_score is None: + model_score = {} + self.model_id = model_id + self.model_score = model_score + + def serialize_model(self) -> str: + data = {"model_id": self.model_id, + "model_score": self.model_score} + return json.dumps(data) + + @classmethod + def deserialize(cls, data_str: str): + data = json.loads(data_str) + res = cls( + data["model_id"], + data["model_score"]) + return res + + +class ModelAcquireData: + """ + Eva worker get model from search strategy + The serialize/deserialize is for good scalability. The project can be decouple into multiple service + """ + + def __init__(self, model_id: str, model_encoding: str, is_last: bool = False, + spi_seconds=None, spi_mini_batch=None): + self.is_last = is_last + self.model_id = model_id + self.model_encoding = model_encoding + + # this is when using spi + self.spi_seconds = spi_seconds + self.spi_mini_batch = spi_mini_batch + + def serialize_model(self) -> str: + data = {"is_last": self.is_last, + "model_id": self.model_id, + "model_encoding": self.model_encoding, + "spi_seconds": self.spi_seconds, + "spi_mini_batch": self.spi_mini_batch} + + return json.dumps(data) + + @classmethod + def deserialize(cls, data_str: str): + data = json.loads(data_str) + res = cls( + data["model_id"], + data["model_encoding"], + data["is_last"], + data["spi_mini_batch"], + data["spi_seconds"]) + return res + + +class ClientStruct: + """ + Client get data + """ + + def __init__(self, budget: float, dataset: str): + self.budget = budget + self.dataset = dataset + + @classmethod + def deserialize(cls, data_str: str): + data = json.loads(data_str) + res = cls( + data["budget"], + data["dataset"] + ) + return res + + +if __name__ == "__main__": + data = ModelEvaData("1", {"a": 1, "b": 2}) + data_str = data.serialize_model() + res = ModelEvaData.deserialize(data_str) + print(res) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/__init__.py new file mode 100644 index 0000000000..25e4ea30d1 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/__init__.py @@ -0,0 +1,17 @@ + + +from src.common.constant import CommonVars +from src.controller.sampler_ea.regularized_ea import RegularizedEASampler +from src.controller.sampler_all.seq_sampler import SequenceSampler +from src.controller.sampler_rl.reinforcement_learning import RLSampler +from src.controller.sampler_rand.random_sample import RandomSampler +from src.controller.sampler_all.seq_sampler import SequenceSampler + +sampler_register = { + CommonVars.TEST_SAMPLER: SequenceSampler, + # CommonVars.RANDOM_SAMPLER: RandomSampler, + CommonVars.RANDOM_SAMPLER: SequenceSampler, + CommonVars.RL_SAMPLER: RLSampler, + CommonVars.EA_SAMPLER: RegularizedEASampler, +} + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/controler.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/controler.py new file mode 100644 index 0000000000..c3d4036dc3 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/controler.py @@ -0,0 +1,173 @@ +import time + +from src.controller.core.sample import Sampler +from src.third_pkg.models import CellStructure + + +class ModelScore: + def __init__(self, model_id, score): + self.model_id = model_id + self.score = score + + def __repr__(self): + return "m_{}_s_{}".format(self.model_id, self.score) + + +# for binary insert +def binary_insert_get_rank(rank_list: list, new_item: ModelScore) -> int: + """ + Insert the new_item to rank_list, then get the rank of it. + :param rank_list: + :param new_item: + :return: + """ + index = search_position(rank_list, new_item) + # search the position to insert into + rank_list.insert(index, new_item) + return index + + +# O(logN) search the position to insert into +def search_position(rank_list_m: list, new_item: ModelScore): + if len(rank_list_m) == 0: + return 0 + left = 0 + right = len(rank_list_m) - 1 + while left + 1 < right: + mid = int((left + right) / 2) + if rank_list_m[mid].score <= new_item.score: + left = mid + else: + right = mid + + # consider the time. + if rank_list_m[right].score <= new_item.score: + return right + 1 + elif rank_list_m[left].score <= new_item.score: + return left + 1 + else: + return left + + +class SampleController(object): + """ + Controller control the sample-score flow in the 1st phase. + It records the results in the history. + """ + + def __init__(self, search_strategy: Sampler): + # Current ea is better than others. + self.search_strategy = search_strategy + + # the large the index, the better the model + self.ranked_models = [] + + # when simple_score_sum=False, records the model's score of each algorithm, + # use when simple_score_sum=True, record the model's sum score + self.history = {} + + def sample_next_arch(self) -> (str, CellStructure): + """ + Return a generator + :return: + """ + return self.search_strategy.sample_next_arch(self.ranked_models) + + def fit_sampler(self, arch_id: str, alg_score: dict, simple_score_sum: bool = False) -> float: + """ + :param arch_id: + :param alg_score: {alg_name1: score1, alg_name2: score2} + :param simple_score_sum: if simply sum multiple scores (good performing), + or sum over their rank (worse performing) + :return: + """ + if simple_score_sum or len(alg_score.keys()) == 1: + score = self._use_pure_score_as_final_res(arch_id, alg_score) + else: + score = self._use_vote_rank_as_final_res(arch_id, alg_score) + self.search_strategy.fit_sampler(score) + return score + + def _use_vote_rank_as_final_res(self, model_id: str, alg_score: dict): + """ + :param model_id: + :param alg_score: {alg_name1: score1, alg_name2: score2} + """ + # todo: bug: only all scores' under all arg is greater than previous one, then treat it as greater. + for alg in alg_score: + if alg not in self.history: + self.history[alg] = [] + + # add model and score to local list + for alg, score in alg_score.items(): + binary_insert_get_rank(self.history[alg], ModelScore(model_id, score)) + + new_rank_score = self._re_rank_model_id(model_id, alg_score) + return new_rank_score + + def _use_pure_score_as_final_res(self, model_id: str, alg_score: dict): + # get the key and sum the score of various alg + score_sum_key = "_".join(list(alg_score.keys())) + if score_sum_key not in self.history: + self.history[score_sum_key] = [] + final_score = 0 + for alg in alg_score: + final_score += float(alg_score[alg]) + # insert and get rank + index = binary_insert_get_rank(self.history[score_sum_key], ModelScore(model_id, final_score)) + self.ranked_models.insert(index, model_id) + return final_score + + def _re_rank_model_id(self, model_id: str, alg_score: dict): + # todo: re-rank everything, to make it self.ranked_models more accurate. + model_new_rank_score = {} + current_explored_models = 0 + for alg, score in alg_score.items(): + for rank_index in range(len(self.history[alg])): + current_explored_models = len(self.history[alg]) + ms_ins = self.history[alg][rank_index] + # rank = index + 1, since index can be 0 + if ms_ins.model_id in model_new_rank_score: + model_new_rank_score[ms_ins.model_id] += rank_index + 1 + else: + model_new_rank_score[ms_ins.model_id] = rank_index + 1 + + for ele in model_new_rank_score.keys(): + model_new_rank_score[ele] = model_new_rank_score[ele] / current_explored_models + + self.ranked_models = [k for k, v in sorted(model_new_rank_score.items(), key=lambda item: item[1])] + new_rank_score = model_new_rank_score[model_id] + return new_rank_score + + def get_current_top_k_models(self, k=-1): + """ + The model is already scored by: low -> high + :param k: + :return: + """ + if k == -1: + # retur all models + return self.ranked_models + else: + return self.ranked_models[-k:] + + +if __name__ == "__main__": + + rank_list = [] + begin = time.time() + score_list = [1, 2, 3, 1, 2] + for i in range(5): + ms = ModelScore(i, score_list[i]) + binary_insert_get_rank(rank_list, ms) + print(rank_list) + print(time.time() - begin) + + rank_list = [] + begin = time.time() + score_list = [1, 1, 1, 1, 1] + for i in range(5): + ms = ModelScore(i, score_list[i]) + binary_insert_get_rank(rank_list, ms) + print(rank_list) + print(time.time() - begin) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/metrics.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/metrics.py new file mode 100644 index 0000000000..78a6d86ebd --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/metrics.py @@ -0,0 +1,28 @@ +from enum import Enum, auto + + +class Metric(Enum): + RAW = auto() + ALL = auto() + + TRAIN_ACCURACY = auto() + VAL_ACCURACY = auto() + TEST_ACCURACY = auto() + + TRAIN_LOSS = auto() + VAL_LOSS = auto() + TEST_LOSS = auto() + + TRAIN_TIME = auto() + VAL_TIME = auto() + TEST_TIME = auto() + + FLOPS = auto() + LATENCY = auto() + PARAMETERS = auto() + EPOCH = auto() + HP = auto() + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/sample.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/sample.py new file mode 100644 index 0000000000..a4b304ce6f --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/core/sample.py @@ -0,0 +1,28 @@ +from abc import abstractmethod + +from src.search_space.core.model_params import ModelMicroCfg +from src.search_space.core.space import SpaceWrapper + + +class Sampler: + + def __init__(self, space: SpaceWrapper): + self.space = space + + @abstractmethod + def sample_next_arch(self, sorted_model: list) -> (str, ModelMicroCfg): + """ + Sample next architecture, + :param sorted_model: the scoted model, + :return: + """ + raise NotImplementedError + + @abstractmethod + def fit_sampler(self, score: float): + """ + Fit the sampler with architecture's score. + :param score: + :return: + """ + raise NotImplementedError diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_EA/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_EA/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_EA/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_RL/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_RL/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_RL/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/seq_sampler.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/seq_sampler.py new file mode 100644 index 0000000000..0dfdc5e077 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_all/seq_sampler.py @@ -0,0 +1,32 @@ +import random + +from src.controller.core.sample import Sampler +from src.search_space.core.model_params import ModelMicroCfg +from src.search_space.core.space import SpaceWrapper + + +class SequenceSampler(Sampler): + + def __init__(self, space: SpaceWrapper): + super().__init__(space) + + self.arch_gene = self.space.sample_all_models() + + def sample_next_arch(self, sorted_model: list = None) -> (str, ModelMicroCfg): + """ + Sample one random architecture, can sample max 10k architectures. + :return: arch_id, architecture + """ + + try: + arch_id, arch_micro = self.arch_gene.__next__() + return arch_id, arch_micro + except Exception as e: + if "StopIteration" in str(e): + print("the end") + return None, None + else: + raise e + + def fit_sampler(self, score: float): + pass diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_bohb/bohb_or.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_bohb/bohb_or.py new file mode 100644 index 0000000000..97266f6ef2 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_bohb/bohb_or.py @@ -0,0 +1,285 @@ +################################################## +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2020 # +################################################################### +# BOHB: Robust and Efficient Hyperparameter Optimization at Scale # +# required to install hpbandster ################################## +# pip install hpbandster ################################## +################################################################### +# OMP_NUM_THREADS=4 python exps/NATS-algos/bohb.py --search_space tss --dataset cifar10 --num_samples 4 --random_fraction 0.0 --bandwidth_factor 3 --rand_seed 1 +# OMP_NUM_THREADS=4 python exps/NATS-algos/bohb.py --search_space sss --dataset cifar10 --num_samples 4 --random_fraction 0.0 --bandwidth_factor 3 --rand_seed 1 +################################################################### +import os, sys, time, random, argparse, collections +from copy import deepcopy + +from src.tools.env_tools import prepare_seed +from src.logger import logger + +from models import CellStructure, get_search_spaces + +# BOHB: Robust and Efficient Hyperparameter Optimization at Scale, ICML 2018 +import ConfigSpace +from hpbandster.optimizers.bohb import BOHB +import hpbandster.core.nameserver as hpns +from hpbandster.core.worker import Worker + +from nats_bench import create + + +def time_string(): + ISOTIMEFORMAT = "%Y-%m-%d %X" + string = "[{:}]".format(time.strftime(ISOTIMEFORMAT, time.gmtime(time.time()))) + return string + + +def get_topology_config_space(search_space, max_nodes=4): + cs = ConfigSpace.ConfigurationSpace() + # edge2index = {} + for i in range(1, max_nodes): + for j in range(i): + node_str = "{:}<-{:}".format(i, j) + cs.add_hyperparameter( + ConfigSpace.CategoricalHyperparameter(node_str, search_space) + ) + return cs + + +def get_size_config_space(search_space): + cs = ConfigSpace.ConfigurationSpace() + for ilayer in range(search_space["numbers"]): + node_str = "layer-{:}".format(ilayer) + cs.add_hyperparameter( + ConfigSpace.CategoricalHyperparameter(node_str, search_space["candidates"]) + ) + return cs + + +def config2topology_func(max_nodes=4): + def config2structure(config): + genotypes = [] + for i in range(1, max_nodes): + xlist = [] + for j in range(i): + node_str = "{:}<-{:}".format(i, j) + op_name = config[node_str] + xlist.append((op_name, j)) + genotypes.append(tuple(xlist)) + return CellStructure(genotypes) + + return config2structure + + +def config2size_func(search_space): + def config2structure(config): + channels = [] + for ilayer in range(search_space["numbers"]): + node_str = "layer-{:}".format(ilayer) + channels.append(str(config[node_str])) + return ":".join(channels) + + return config2structure + + +class MyWorker(Worker): + def __init__(self, *args, convert_func=None, dataset=None, api=None, **kwargs): + super().__init__(*args, **kwargs) + self.convert_func = convert_func + self._dataset = dataset + self._api = api + self.total_times = [] + self.trajectory = [] + + def compute(self, config, budget, **kwargs): + arch = self.convert_func(config) + accuracy, latency, time_cost, total_time = self._api.simulate_train_eval( + arch, self._dataset, iepoch=int(budget) - 1, hp="12" + ) + self.trajectory.append((accuracy, arch)) + self.total_times.append(total_time) + return {"loss": 100 - accuracy, "info": self._api.query_index_by_arch(arch)} + + +def main(xargs, api): + import torch + torch.set_num_threads(4) + prepare_seed(xargs.rand_seed) + + logger.info("{:} use api : {:}".format(time_string(), api)) + api.reset_time() + search_space = get_search_spaces(xargs.search_space, "nats-bench") + if xargs.search_space == "tss": + cs = get_topology_config_space(search_space) + config2structure = config2topology_func() + else: + cs = get_size_config_space(search_space) + config2structure = config2size_func(search_space) + + hb_run_id = "0" + + NS = hpns.NameServer(run_id=hb_run_id, host="localhost", port=0) + ns_host, ns_port = NS.start() + num_workers = 1 + + workers = [] + for i in range(num_workers): + w = MyWorker( + nameserver=ns_host, + nameserver_port=ns_port, + convert_func=config2structure, + dataset=xargs.dataset, + api=api, + run_id=hb_run_id, + id=i, + ) + w.run(background=True) + workers.append(w) + + start_time = time.time() + bohb = BOHB( + configspace=cs, + run_id=hb_run_id, + eta=3, + min_budget=1, + max_budget=12, + nameserver=ns_host, + nameserver_port=ns_port, + num_samples=xargs.num_samples, + random_fraction=xargs.random_fraction, + bandwidth_factor=xargs.bandwidth_factor, + ping_interval=10, + min_bandwidth=xargs.min_bandwidth, + ) + + results = run(xargs.n_iters, min_n_workers=num_workers) + + bohb.shutdown(shutdown_workers=True) + NS.shutdown() + + # print('There are {:} runs.'.format(len(results.get_all_runs()))) + # workers[0].total_times + # workers[0].trajectory + current_best_index = [] + for idx in range(len(workers[0].trajectory)): + trajectory = workers[0].trajectory[: idx + 1] + arch = max(trajectory, key=lambda x: x[0])[1] + current_best_index.append(api.query_index_by_arch(arch)) + + best_arch = max(workers[0].trajectory, key=lambda x: x[0])[1] + logger.log( + "Best found configuration: {:} within {:.3f} s".format( + best_arch, workers[0].total_times[-1] + ) + ) + info = api.query_info_str_by_arch( + best_arch, "200" if xargs.search_space == "tss" else "90" + ) + logger.log("{:}".format(info)) + logger.log("-" * 100) + logger.close() + + return logger.log_dir, current_best_index, workers[0].total_times + + +if __name__ == "__main__": + parser = argparse.ArgumentParser( + "BOHB: Robust and Efficient Hyperparameter Optimization at Scale" + ) + parser.add_argument( + "--dataset", + default="cifar10", + type=str, + choices=["cifar10", "cifar100", "ImageNet16-120"], + help="Choose between Cifar10/100 and ImageNet-16.", + ) + # general arg + parser.add_argument( + "--search_space", + default="tss", + type=str, + choices=["tss", "sss"], + help="Choose the search space.", + ) + parser.add_argument( + "--time_budget", + type=int, + default=20000, + help="The total time cost budge for searching (in seconds).", + ) + parser.add_argument( + "--loops_if_rand", type=int, default=500, help="The total runs for evaluation." + ) + # BOHB + parser.add_argument( + "--strategy", + default="sampling", + type=str, + nargs="?", + help="optimization strategy for the acquisition function", + ) + parser.add_argument( + "--min_bandwidth", + default=0.3, + type=float, + nargs="?", + help="minimum bandwidth for KDE", + ) + parser.add_argument( + "--num_samples", + default=64, + type=int, + nargs="?", + help="number of samples for the acquisition function", + ) + parser.add_argument( + "--random_fraction", + default=0.33, + type=float, + nargs="?", + help="fraction of random configurations", + ) + parser.add_argument( + "--bandwidth_factor", + default=3, + type=int, + nargs="?", + help="factor multiplied to the bandwidth", + ) + parser.add_argument( + "--n_iters", + default=300, + type=int, + nargs="?", + help="number of iterations for optimization method", + ) + # log + parser.add_argument( + "--save_dir", + type=str, + default="./output/search", + help="Folder to save checkpoints and log.", + ) + parser.add_argument("--rand_seed", type=int, default=-1, help="manual seed") + args = parser.parse_args() + + api = create(None, args.search_space, fast_mode=False, verbose=False) + + args.save_dir = os.path.join( + "{:}-{:}".format(args.save_dir, args.search_space), + "{:}-T{:}".format(args.dataset, args.time_budget), + "BOHB", + ) + print("save-dir : {:}".format(args.save_dir)) + + import torch + if args.rand_seed < 0: + save_dir, all_info = None, collections.OrderedDict() + for i in range(args.loops_if_rand): + print("{:} : {:03d}/{:03d}".format(time_string(), i, args.loops_if_rand)) + args.rand_seed = random.randint(1, 100000) + save_dir, all_archs, all_total_times = main(args, api) + all_info[i] = {"all_archs": all_archs, "all_total_times": all_total_times} + save_path = save_dir / "results.pth" + print("save into {:}".format(save_path)) + torch.save(all_info, save_path) + else: + main(args, api) + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_ea/regularized_ea.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_ea/regularized_ea.py new file mode 100644 index 0000000000..d8a862a1fc --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_ea/regularized_ea.py @@ -0,0 +1,130 @@ +import collections +from src.search_space.core.model_params import ModelMicroCfg +from src.controller.core.sample import Sampler +import random +from src.search_space.core.space import SpaceWrapper + + +class Model(object): + def __init__(self): + self.arch = None + self.score = None + + def __str__(self): + """Prints a readable version of this bitstring.""" + return "{:}".format(self.arch) + + +class RegularizedEASampler(Sampler): + + def __init__(self, space: SpaceWrapper, population_size: int, sample_size: int): + super().__init__(space) + + self.population_size = population_size + # list of object, + self.population = collections.deque() + # list of str, for duplicate checking + self.population_model_ids = collections.deque() + + self.space = space + self.sample_size = sample_size + self.current_sampled = 0 + + # id here is to match the outside value. + self.current_arch_id = None + self.current_arch_micro = None + + # use the visited to reduce the collapse + self.visited = {} + self.max_mutate_time = 2 + self.max_mutate_sampler_time = 2 + + def sample_next_arch(self, sorted_model_ids: list) -> (str, ModelMicroCfg): + """ + This function performs one evolution cycle. It produces a model and removes another. + Models are sampled randomly from the current population. If the population size is less than the + desired population size, a random architecture is added to the population. + + :param sorted_model_ids: List of model ids sorted based on some criterion (not used here directly). + :return: Tuple of the architecture id and the architecture configuration (micro). + """ + # Case 1: If population hasn't reached desired size, add random architectures + if len(self.population) < self.population_size: + while True: + arch_id, arch_micro = self.space.random_architecture_id() + # Ensure that EA population has no repeated value + if str(arch_id) not in self.population_model_ids: + break + self.current_arch_micro = arch_micro + self.current_arch_id = arch_id + return arch_id, arch_micro + + # Case 2: If population has reached desired size, evolve population + else: + cur_mutate_sampler_time = 0 + is_found_new = False + + # Keep attempting mutations for a maximum of 'max_mutate_sampler_time' times + while cur_mutate_sampler_time < self.max_mutate_sampler_time: + cur_mutate_time = 0 + + # Randomly select a sample of models from the population + sample = [] + sample_ids = [] + while len(sample) < self.sample_size: + candidate = random.choice(list(self.population)) + candidate_id = self.population_model_ids[self.population.index(candidate)] + sample.append(candidate) + sample_ids.append(candidate_id) + + # Select the best parent from the sample (based on the order in sorted_model_ids) + parent_id = max(sample_ids, key=lambda _id: sorted_model_ids.index(str(_id))) + parent = sample[sample_ids.index(parent_id)] + + # Try to mutate the parent up to 'max_mutate_time' times + while cur_mutate_time < self.max_mutate_time: + arch_id, arch_micro = self.space.mutate_architecture(parent.arch) + + # If the mutated architecture hasn't been visited or we've visited all possible architectures, stop + if arch_id not in self.visited or len(self.space) == len(self.visited): + self.visited[arch_id] = True + is_found_new = True + break + cur_mutate_time += 1 + + # If we've found a new architecture, stop sampling + if is_found_new: + break + + cur_mutate_sampler_time += 1 + + # If we've hit the maximum number of mutation attempts, do nothing + if cur_mutate_time * cur_mutate_sampler_time == self.max_mutate_time * self.max_mutate_sampler_time: + pass + + # Update current architecture details + self.current_arch_micro = arch_micro + self.current_arch_id = arch_id + + return arch_id, arch_micro + + def fit_sampler(self, score: float): + # if it's in Initialize stage, add to the population with random models. + if len(self.population) < self.population_size: + model = Model() + model.arch = self.current_arch_micro + model.score = score + self.population.append(model) + self.population_model_ids.append(self.current_arch_id) + + # if it's in mutation stage + else: + child = Model() + child.arch = self.current_arch_micro + child.score = score + + self.population.append(child) + self.population_model_ids.append(self.current_arch_id) + # Remove the oldest model. + self.population.popleft() + self.population_model_ids.popleft() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/random_sample.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/random_sample.py new file mode 100644 index 0000000000..fb927cf3ec --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rand/random_sample.py @@ -0,0 +1,22 @@ +from src.controller.core.sample import Sampler +from src.search_space.core.space import SpaceWrapper +from src.search_space.core.model_params import ModelMicroCfg + + +class RandomSampler(Sampler): + + def __init__(self, space: SpaceWrapper): + super().__init__(space) + self.visited = [] + + def sample_next_arch(self, sorted_model: list = None) -> (str, ModelMicroCfg): + while True: + arch_id, model_micro = self.space.random_architecture_id() + + if arch_id not in self.visited: + self.visited.append(arch_id) + return str(arch_id), model_micro + + def fit_sampler(self, score: float): + # random sampler can skip this. + pass diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rl/reinforcement_learning.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rl/reinforcement_learning.py new file mode 100644 index 0000000000..bdd49e3758 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/controller/sampler_rl/reinforcement_learning.py @@ -0,0 +1,48 @@ + +from src.controller.core.sample import Sampler +from src.search_space.core.space import SpaceWrapper +from src.search_space.core.model_params import ModelMicroCfg +from src.third_pkg.models import CellStructure + + +class ExponentialMovingAverage(object): + """Class that maintains an exponential moving average.""" + + def __init__(self, momentum): + self._numerator = 0 + self._denominator = 0 + self._momentum = momentum + + def update(self, value): + self._numerator = ( + self._momentum * self._denominator + (1 - self._momentum) * value + ) + self._denominator = self._momentum * self._denominator + (1 - self._momentum) + + def value(self): + """Return the current value of the moving average""" + return self._numerator / self._denominator + + +class RLSampler(Sampler): + + def __init__(self, space: SpaceWrapper, args): + + super().__init__(space) + + self.policy = self.space.get_reinforcement_learning_policy(args.rl_learning_rate) + # update policy's parameters + self.baseline = ExponentialMovingAverage(args.rl_EMA_momentum) + self.log_prob = 0 + + def sample_next_arch(self, max_nodes: int) -> (str, ModelMicroCfg): + while True: + self.log_prob, action = self.policy.select_action() + arch_struct = self.policy.generate_arch(action) + arch_id = self.space.arch_to_id(arch_struct) + yield arch_id, arch_struct + + def fit_sampler(self, score: float): + reward = score + self.baseline.update(reward) + self.policy.update_policy(reward, self.baseline.value(), self.log_prob) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/__init__.py new file mode 100644 index 0000000000..139597f9cb --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/__init__.py @@ -0,0 +1,2 @@ + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/download_critero_and_avazu.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/download_critero_and_avazu.py new file mode 100644 index 0000000000..418a71f831 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/download_critero_and_avazu.py @@ -0,0 +1,42 @@ +# adapted from AFN-AAAI-20 +import os +import zipfile +import urllib.request +from tqdm import tqdm + + +class DownloadProgressBar(tqdm): + def update_to(self, b=1, bsize=1, tsize=None): + if tsize is not None: + self.total = tsize + self.update(b * bsize - self.n) + + +def download(url, output_path): + with DownloadProgressBar(unit='B', unit_scale=True, + miniters=1, desc=url.split('/')[-1]) as t: + urllib.request.urlretrieve(url, filename=output_path, reporthook=t.update_to) + + +if __name__ == "__main__": + # if not os.path.exists('../data/avazu/'): + # os.mkdir('../data/avazu/') + # print("Begin to download avazu data, the total size is 683MB...") + # download('https://worksheets.codalab.org/rest/bundles/0xf5ab597052744680b1a55986557472c7/contents/blob/', '../data/avazu/avazu.zip') + # print("Unzipping avazu dataset...") + # with zipfile.ZipFile('../data/avazu/avazu.zip', 'r') as zip_ref: + # zip_ref.extractall('../data/avazu/') + # print("Done.") + + if not os.path.exists('../exp_data/data/structure_data/criteo/'): + os.mkdir('../exp_data/data/structure_data/criteo/') + print("Begin to download criteo data, the total size is 3GB...") + + output_path = '../exp_data/data/structure_data/criteo/criteo.zip' + if not os.path.exists(output_path): + download('https://worksheets.codalab.org/rest/bundles/0x8dca5e7bac42470aa445f9a205d177c6/contents/blob/', + output_path) + print("Unzipping criteo dataset...") + with zipfile.ZipFile('../exp_data/data/structure_data/criteo/criteo.zip', 'r') as zip_ref: + zip_ref.extractall('../exp_data/data/structure_data/criteo/') + print("Done.") diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/sequence_dataloader.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/sequence_dataloader.py new file mode 100644 index 0000000000..4bb494c967 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/sequence_dataloader.py @@ -0,0 +1,78 @@ +import queue +import threading +import requests +import time +from src.logger import logger + + +class SquenceDataLoader: + """ + This will preoritically query data from cache-service + """ + + def __init__(self, cache_svc_url, table_name, name_space): + self.last_fetch_time = 0 + self.table_name = table_name + # train, valid, test + self.name_space = name_space + self.end_signal = "end_position" + self.cache_svc_url = cache_svc_url + self.data_queue = queue.Queue(maxsize=10) + self.stop_event = threading.Event() + self.thread = threading.Thread(target=self.fetch_data, daemon=True) + self.thread.start() + + def fetch_data(self): + while not self.stop_event.is_set(): + response = requests.get( + f'{self.cache_svc_url}/', + params={ + 'table_name': self.table_name, + 'name_space': self.name_space}) + + if response.status_code == 200: + batch = response.json() + + # in trianing, we use iteraiton-per-epoch to control the end + if batch == self.end_signal: + if self.name_space == "valid": + # end_signal in inference, stop ! + logger.info("[StreamingDataLoader]: last iteration in valid is meet!") + self.data_queue.put({self.end_signal: True}) + else: + # end_signal in trianing, then keep training + continue + else: + import torch + # convert to tensor again + id_tensor = torch.LongTensor(batch['id']) + value_tensor = torch.FloatTensor(batch['value']) + y_tensor = torch.FloatTensor(batch['y']) + data_tensor = {'id': id_tensor, 'value': value_tensor, 'y': y_tensor} + self.data_queue.put(data_tensor) + else: + print(response.json()) + time.sleep(5) + + def __iter__(self): + return self + + def __next__(self): + print("compute time = ", time.time() - self.last_fetch_time) + self.last_fetch_time = time.time() + if self.data_queue.empty() and not self.thread.is_alive(): + raise StopIteration + else: + data = self.data_queue.get(block=True) + if self.end_signal in data: + raise StopIteration + else: + return data + + def __len__(self): + return self.data_queue.qsize() + + def stop(self): + self.stop_event.set() + self.thread.join() + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/stream_dataloader.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/stream_dataloader.py new file mode 100644 index 0000000000..3c2abe7a3c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/dataset_utils/stream_dataloader.py @@ -0,0 +1,78 @@ +import queue +import threading +import requests +import time +from src.logger import logger + + +class StreamingDataLoader: + """ + This will preoritically query data from cache-service + """ + + def __init__(self, cache_svc_url, table_name, name_space): + self.last_fetch_time = 0 + self.table_name = table_name + # train, valid, test + self.name_space = name_space + self.end_signal = "end_position" + self.cache_svc_url = cache_svc_url + self.data_queue = queue.Queue(maxsize=10) + self.stop_event = threading.Event() + self.thread = threading.Thread(target=self.fetch_data, daemon=True) + self.thread.start() + + def fetch_data(self): + while not self.stop_event.is_set(): + response = requests.get( + f'{self.cache_svc_url}/', + params={ + 'table_name': self.table_name, + 'name_space': self.name_space}) + + if response.status_code == 200: + batch = response.json() + + # in trianing, we use iteraiton-per-epoch to control the end + if batch == self.end_signal: + if self.name_space == "valid": + # end_signal in inference, stop ! + logger.info("[StreamingDataLoader]: last iteration in valid is meet!") + self.data_queue.put({self.end_signal: True}) + else: + # end_signal in trianing, then keep training + continue + else: + # convert to tensor again + import torch + id_tensor = torch.LongTensor(batch['id']) + value_tensor = torch.FloatTensor(batch['value']) + y_tensor = torch.FloatTensor(batch['y']) + data_tensor = {'id': id_tensor, 'value': value_tensor, 'y': y_tensor} + self.data_queue.put(data_tensor) + else: + print(response.json()) + time.sleep(5) + + def __iter__(self): + return self + + def __next__(self): + print("compute time = ", time.time() - self.last_fetch_time) + self.last_fetch_time = time.time() + if self.data_queue.empty() and not self.thread.is_alive(): + raise StopIteration + else: + data = self.data_queue.get(block=True) + if self.end_signal in data: + raise StopIteration + else: + return data + + def __len__(self): + return self.data_queue.qsize() + + def stop(self): + self.stop_event.set() + self.thread.join() + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/__init__.py new file mode 100644 index 0000000000..ca0cf53448 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/__init__.py @@ -0,0 +1,10 @@ +from src.common.constant import * +from src.eva_engine.phase1.algo.prune_synflow import SynFlowEvaluator + +# evaluator mapper to register many existing evaluation algorithms +evaluator_register = { + + # prune based + CommonVars.PRUNE_SYNFLOW: SynFlowEvaluator(), + +} diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/coordinator.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/coordinator.py new file mode 100644 index 0000000000..331aec57a8 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/coordinator.py @@ -0,0 +1,98 @@ +from src.common.constant import Config +from src.eva_engine.phase2.run_sh import BudgetAwareControllerSH +from src.logger import logger +from src.search_space.core.space import SpaceWrapper + +eta = 3 + + +def min_budget_calculation(search_space_ins: SpaceWrapper, dataset: str, + N_K_ratio: int, sh: BudgetAwareControllerSH, t1_: float): + # Calculate the minimum budget requirements for both phases + K_max = int(len(search_space_ins) / N_K_ratio) + + if search_space_ins.name == Config.NB101: + U_options = [4, 12, 16, 108] + elif search_space_ins.name == Config.NB201: + U_options = list(range(1, 200)) + elif search_space_ins.name == Config.MLPSP: + # TODO: This is for benchmark only + if dataset == Config.Frappe: + MaxEpochTrained = 20 + elif dataset == Config.Criteo: + MaxEpochTrained = 10 + elif dataset == Config.UCIDataset: + MaxEpochTrained = 40 + else: + raise NotImplementedError + U_options = list(range(1, MaxEpochTrained)) + else: + raise NotImplementedError + + U_min = U_options[0] + min_budget_required_both_phase = sh.pre_calculate_time_required(K=1, U=U_min)[1] + N_K_ratio * t1_ + + return K_max, U_options, U_min, min_budget_required_both_phase + + +def schedule(dataset: str, sh: BudgetAwareControllerSH, T_: float, t1_: float, t2_: float, w_: int, + search_space_ins: SpaceWrapper, N_K_ratio: int, + only_phase1: bool = False): + """ + :param dataset + :param sh: BudgetAwareControllerSH instnace + :param T_: user given time budget + :param t1_: time to score one model + :param t2_: time to train one model + :param w_: number of workers, for parallelly running. + :param search_space_ins: search spcae instance + :param N_K_ratio: N/K = N_K_ratio + :param only_phase1: Only use filtering phase. + """ + if T_ < 1: + raise ValueError('Total time budget must be greater than 1 second') + + K_max, U_options, U_min, min_budget_required_both_phase = min_budget_calculation( + search_space_ins, dataset, N_K_ratio, sh, t1_) + + # collection of (best_K, best_U, best_N) + history = [] + + # Calculate phase 1 + time_used = t1_ + enable_phase2_at_least = sh.pre_calculate_time_required(K=2, U=U_min)[1] + 2 * N_K_ratio * t1_ + + if only_phase1 or enable_phase2_at_least > T_: + # all time give to phase1, explore n models + N_only = min(int(T_ / t1_), len(search_space_ins)) + history.extend([(1, U_min, i) for i in range(1, N_only + 1) if i * t1_ <= T_]) + if not history: + raise ValueError( + f' [trails] Only p1, Budget {T_} is too small, it\'s at least >= {time_used} with current worker, ' + f'{t1_}, {t2_}, eta') + + # Calculate phase 2, start from min U, if user given budget is larger enough, then evaluat each mode with more epoch + else: + # record all possible K, U pair meeting the SLO ( time used < T) + for K_ in range(2, min(int(T_ / t1_), K_max) + 1): + N_ = K_ * N_K_ratio + for U in U_options: + time_used = sh.pre_calculate_time_required(K=K_, U=U)[1] + N_ * t1_ + if time_used > T_: + break + else: + history.append((K_, U, N_)) + if not history: + raise ValueError( + f' [trails] Budget {T_} is too small, it\'s at least >= {min_budget_required_both_phase}' + f' with current worker, {t1_}, {t2_}, eta') + + best_K, best_U, best_N = history[-1] + N_scored = best_N + B1_time_used = N_scored * t1_ + B2_all_epoch, B2_time_used = sh.pre_calculate_time_required(K=best_K, U=best_U) + + logger.info( + f' [trails] The schedule result: when T = {T_} second, N = {N_scored}, K = {best_K}, best_U = {best_U}, ' + f'time_used = {B1_time_used + B2_time_used}') + return best_K, best_U, N_scored, B1_time_used, B2_time_used, B2_all_epoch diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/prune_synflow.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/prune_synflow.py new file mode 100644 index 0000000000..8f3373c448 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/prune_synflow.py @@ -0,0 +1,407 @@ +from src.eva_engine.phase1.algo.alg_base import Evaluator +from src.common.constant import Config + +from singa import singa_wrap as singa +from singa import device as singa_device +from singa import tensor +from singa import opt +from singa import autograd +from singa.opt import Optimizer +from singa.opt import DecayScheduler +from singa.opt import Constant +import numpy as np +import time +import argparse +from PIL import Image +from numpy import linalg as LA + +np_dtype = {"float16": np.float16, "float32": np.float32} + +# singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} +singa_dtype = {"float32": tensor.float32} + +### MSOptimizer +class MSOptimizer(Optimizer): + def __call__(self, loss): + pn_p_g_list = self.call_with_returns(loss) + # print ("optimizer1 before self.step()") + # print ("optimizer1 before print len(pn_p_g_list): \n", len(pn_p_g_list)) + self.step() + # print ("optimizer1 after print len(pn_p_g_list): \n", len(pn_p_g_list)) + # print ("optimizer1 after self.step()") + return pn_p_g_list + + def call_with_returns(self, loss): + # print ("call_with_returns before apply loss.data: \n", loss.data) + pn_p_g_list = [] + for p, g in autograd.backward(loss): + if p.name is None: + p.name = id(p) + self.apply(p.name, p, g) + # print ("call with returns") + # print ("p.name: \n", p.name) + # print ("p.data: \n", p.data) + # print ("g.data: \n", g.data) + pn_p_g_list.append([p.name, p, g]) # need iterables + # print ("call_with_returns after apply loss.data: \n", loss.data) + return pn_p_g_list + +# MSSGD -- actually no change of code +class MSSGD(MSOptimizer): + """Implements stochastic gradient descent (optionally with momentum). + + Nesterov momentum is based on the formula from `On the importance of initialization and momentum in deep learning`__. + + Args: + lr(float): learning rate + momentum(float, optional): momentum factor(default: 0) + weight_decay(float, optional): weight decay(L2 penalty)(default: 0) + dampening(float, optional): dampening for momentum(default: 0) + nesterov(bool, optional): enables Nesterov momentum(default: False) + + Typical usage example: + >> > from singa import opt + >> > optimizer = opt.SGD(lr=0.1, momentum=0.9) + >> > optimizer.update() + + __ http: // www.cs.toronto.edu / %7Ehinton / absps / momentum.pdf + + .. note:: + The implementation of SGD with Momentum / Nesterov subtly differs from + Sutskever et. al. and implementations in some other frameworks. + + Considering the specific case of Momentum, the update can be written as + + .. math:: + v = \rho * v + g \\ + p = p - lr * v + + where p, g, v and: math: `\rho` denote the parameters, gradient, + velocity, and momentum respectively. + + This is in contrast to Sutskever et. al. and + other frameworks which employ an update of the form + + .. math:: + v = \rho * v + lr * g \\ + p = p - v + + The Nesterov version is analogously modified. + """ + + def __init__(self, + lr=0.1, + momentum=0, + dampening=0, + weight_decay=0, + nesterov=False, + dtype=tensor.float32): + super(MSSGD, self).__init__(lr) + + # init momentum + if type(momentum) == float or type(momentum) == int: + if momentum < 0.0: + raise ValueError("Invalid momentum value: {}".format(momentum)) + self.momentum = Constant(momentum) + elif isinstance(momentum, DecayScheduler): + self.momentum = momentum + momentum = momentum.init_value + else: + raise TypeError("Wrong momentum type") + # self.dtype = dtype + # self.mom_value = self.momentum(self.step_counter).as_type(self.dtype) + self.mom_value = self.momentum(self.step_counter) + + # init dampening + if type(dampening) == float or type(dampening) == int: + self.dampening = Constant(dampening) + elif isinstance(dampening, DecayScheduler): + self.dampening = dampening + dampening = dampening.init_value + else: + raise TypeError("Wrong dampening type") + # self.dam_value = self.dampening(self.step_counter).as_type(self.dtype) + self.dam_value = self.dampening(self.step_counter) + + # init weight_decay + if type(weight_decay) == float or type(weight_decay) == int: + if weight_decay < 0.0: + raise ValueError( + "Invalid weight_decay value: {}".format(weight_decay)) + self.weight_decay = Constant(weight_decay) + elif isinstance(weight_decay, DecayScheduler): + self.weight_decay = weight_decay + else: + raise TypeError("Wrong weight_decay type") + # self.decay_value = self.weight_decay(self.step_counter).as_type(self.dtype) + self.decay_value = self.weight_decay(self.step_counter) + + # init other params + self.nesterov = nesterov + self.moments = dict() + + # check value + if nesterov and (momentum <= 0 or dampening != 0): + raise ValueError( + "Nesterov momentum requires a momentum and zero dampening") + + def apply(self, param_name, param_value, param_grad): + """Performs a single optimization step. + + Args: + param_name(String): the name of the param + param_value(Tensor): param values to be update in-place + grad(Tensor): param gradients; the values may be updated + in this function; cannot use it anymore + """ + assert param_value.shape == param_grad.shape, ("shape mismatch", + param_value.shape, + param_grad.shape) + self.device_check(param_value, self.step_counter, self.lr_value, + self.mom_value, self.dam_value, self.decay_value) + + # derive dtype from input + # assert param_value.dtype == self.dtype + + # TODO add branch operator + # if self.decay_value != 0: + if self.weight_decay.init_value != 0: + singa.Axpy(self.decay_value.data, param_value.data, param_grad.data) + + if self.momentum.init_value != 0: + if param_name not in self.moments: + flag = param_value.device.graph_enabled() + param_value.device.EnableGraph(False) + self.moments[param_name] = tensor.zeros_like(param_value) + param_value.device.EnableGraph(flag) + + buf = self.moments[param_name] + buf *= self.mom_value + alpha = 1.0 - self.dam_value + singa.Axpy(alpha.data, param_grad.data, buf.data) + + if self.nesterov: + singa.Axpy(self.mom_value.data, buf.data, param_grad.data) + else: + param_grad = buf + + minus_lr = 0.0 - self.lr_value + singa.Axpy(minus_lr.data, param_grad.data, param_value.data) + + def step(self): + # increment step counter, lr and moment + # print ("before super step") + super().step() + # print ("after super step") + # print ("before custiomized step") + # mom_value = self.momentum(self.step_counter).as_type(self.dtype) + # dam_value = self.dampening(self.step_counter).as_type(self.dtype) + # decay_value = self.weight_decay(self.step_counter).as_type(self.dtype) + mom_value = self.momentum(self.step_counter) + dam_value = self.dampening(self.step_counter) + decay_value = self.weight_decay(self.step_counter) + self.mom_value.copy_from(mom_value) + self.dam_value.copy_from(dam_value) + self.decay_value.copy_from(decay_value) + # print ("after customized step") + + def get_states(self): + states = super().get_states() + if self.mom_value > 0: + states[ + 'moments'] = self.moments # a dict for 1st order moments tensors + return states + + def set_states(self, states): + super().set_states(states) + if 'moments' in states: + self.moments = states['moments'] + self.mom_value = self.momentum(self.step_counter) + +# Data augmentation +def augmentation(x, batch_size): + xpad = np.pad(x, [[0, 0], [0, 0], [4, 4], [4, 4]], 'symmetric') + for data_num in range(0, batch_size): + offset = np.random.randint(8, size=2) + x[data_num, :, :, :] = xpad[data_num, :, + offset[0]:offset[0] + x.shape[2], + offset[1]:offset[1] + x.shape[2]] + if_flip = np.random.randint(2) + if (if_flip): + x[data_num, :, :, :] = x[data_num, :, :, ::-1] + return x + + +# Calculate accuracy +def accuracy(pred, target): + # y is network output to be compared with ground truth (int) + y = np.argmax(pred, axis=1) + a = y == target + correct = np.array(a, "int").sum() + return correct + + +# Data partition according to the rank +def partition(global_rank, world_size, train_x, train_y, val_x, val_y): + # Partition training data + data_per_rank = train_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + train_x = train_x[idx_start:idx_end] + train_y = train_y[idx_start:idx_end] + + # Partition evaluation data + data_per_rank = val_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + val_x = val_x[idx_start:idx_end] + val_y = val_y[idx_start:idx_end] + return train_x, train_y, val_x, val_y + + +# Function to all reduce NUMPY accuracy and loss from multiple devices +def reduce_variable(variable, dist_opt, reducer): + reducer.copy_from_numpy(variable) + dist_opt.all_reduce(reducer.data) + dist_opt.wait() + output = tensor.to_numpy(reducer) + return output + + +def resize_dataset(x, image_size): + num_data = x.shape[0] + dim = x.shape[1] + X = np.zeros(shape=(num_data, dim, image_size, image_size), + dtype=np.float32) + for n in range(0, num_data): + for d in range(0, dim): + X[n, d, :, :] = np.array(Image.fromarray(x[n, d, :, :]).resize( + (image_size, image_size), Image.BILINEAR), + dtype=np.float32) + return X + +import torch +class SynFlowEvaluator(Evaluator): + + def __init__(self): + super().__init__() + + def evaluate(self, arch, device, batch_data: object, batch_labels: torch.Tensor, space_name: str) -> float: + """ + This is implementation of paper + "Pruning neural networks without any data by iteratively conserving synaptic flow" + The score takes 5 steps: + 1. For each layer, for each parameter, calculate the absolute value |0| + 2. Use a single all-one-vector with dim = [1, c, h, w] to run a forward, + Since only consider linear and Con2d operation, the forward output is multiple( [ |0l| for l in L] ) + 3. New loss function R = sum(output), and then run backward + 4. for each layer, calculate Sl = Hadamard product( df/dw, w), where Sij=aij×bij + 5. score = sum( [ Sl for l in layers ] ) + Comments: + 1. this is data-Agnostic + 2. only compute on a single example + """ + + ### singa configs + mssgd = MSSGD(lr=0.005, momentum=0.9, weight_decay=1e-5, dtype=singa_dtype['float32']) + device_id = 0 + max_epoch = 1 + model = arch + graph = True + verbosity = 0 + dist_option='plain' + spars=None + precision = 'float32' + global_rank = 0 + world_size = 1 + + ### singa setups + # print ("device: \n", device) + if device == 'cpu': + dev = singa_device.get_default_device() + else: # GPU + dev = singa_device.create_cuda_gpu_on(local_rank) # need to change to CPU device for CPU-only machines + dev.SetRandSeed(0) + np.random.seed(0) + + # For distributed training, sequential has better performance + if hasattr(mssgd, "communicator"): + DIST = True + sequential = True + else: + DIST = False + sequential = False + + model.train() + + ### process batch_data + x = batch_data.cpu().numpy() # Size([1, 100]) and all ones + x = x.astype(np_dtype[precision]) + y = np.ones(x.shape[0], dtype=np.int32) + if model.dimension == 2: # input data dimension + tx = tensor.Tensor(x.shape, dev, singa_dtype[precision]) + ty = tensor.Tensor((x.shape[0],), dev, tensor.int32) + + model.set_optimizer(mssgd) + model.compile([tx], is_train=True, use_graph=graph, sequential=sequential) + dev.SetVerbosity(verbosity) + + + # 1. Convert params to their abs. + synflow_flag = True ### just change the model to the absolute value + tx.copy_from_numpy(x) # dtype=np.float32 + ty.copy_from_numpy(y) + # print ("before model forward ...") + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + # print ("---------------------------------------") + # print ("before absolute prune_synflow !!!nemb input vector!!! tensor.to_numpy(loss)[0]: ", tensor.to_numpy(loss)[0]) + # print ("before absolute prune_synflow !!!nemb input vector!!! tensor.to_numpy(loss): ", tensor.to_numpy(loss)) + # train_correct += accuracy(tensor.to_numpy(out), y) + # train_loss += tensor.to_numpy(loss)[0] + # all params turned to positive + for pn_p_g_item in pn_p_g_list: + # print ("absolute value parameter name: \n", pn_p_g_item[0]) + param_np = tensor.to_numpy(pn_p_g_item[1]) + # print ("param_np shape: \n", param_np.shape) + # print ("param_np sqrt norm: \n", np.sqrt(LA.norm(param_np)/param_np.size)) + # print ("before abs np.min(tensor.to_numpy(pn_p_g_item[1])): \n", np.min(tensor.to_numpy(pn_p_g_item[1]))) + pn_p_g_item[1] = tensor.abs(pn_p_g_item[1]) # tensor actually .. + # print ("after abs np.min(tensor.to_numpy(pn_p_g_item[1])): \n", np.min(tensor.to_numpy(pn_p_g_item[1]))) + # print ("after abs pn_p_g_item[1][0]: \n", pn_p_g_item[1][0]) + + # 2. Compute gradients with input of one dummy example ( 1-vector with dimension [1, c, h, w] ) + # 3.R = sum(output) + # 4. Select the gradients that we want to use for search/prune + # 5. Sum over all parameter's results to get the final score. + # score = sum([grad.sum() for grad in grads_abs]) + + # print ("calculate synflow") + synflow_flag = True + ### step 1: all one input + # Copy the patch data into input tensors + # tx.copy_from_numpy(np.ones(x.shape, dtype=np.float32)) + tx.copy_from_numpy(x) # dtype=np.float32 # actually it is all ones ... --> np.ones(x.shape, dtype=np.float32) + ty.copy_from_numpy(y) + ### step 2: all weights turned to positive (done) + ### step 3: new loss (done) + # print ("before model forward ...") + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + # print ("prune_synflow !!!nemb input vector!!! synflow step tensor.to_numpy(loss)[0]: ", tensor.to_numpy(loss)[0]) + ### step 4: calculate the multiplication of weights + score = 0.0 + for pn_p_g_item in pn_p_g_list: + # print ("calculate weight param * grad parameter name: \n", pn_p_g_item[0]) + if len(pn_p_g_item[1].shape) == 2: # param_value.data is "weight" + # print ("pn_p_g_item[1].shape: \n", pn_p_g_item[1].shape) + # print ("tensor.to_numpy(pn_p_g_item[1][0]): ", tensor.to_numpy(pn_p_g_item[1][0])) + # print ("calculate synflow parameter name: \n", pn_p_g_item[0]) + # print ("should be positive np.min(tensor.to_numpy(pn_p_g_item[1])): ", np.min(tensor.to_numpy(pn_p_g_item[1]))) + # print ("weight should be positive tensor.to_numpy(pn_p_g_item[1][0])[0, :10]: ", tensor.to_numpy(pn_p_g_item[1][0])[0, :10]) + # print ("gradients tensor.to_numpy(pn_p_g_item[2][0])[0, :10]: ", tensor.to_numpy(pn_p_g_item[2][0])[0, :10]) + # print () + score += np.sum(np.absolute(tensor.to_numpy(pn_p_g_item[1]) * tensor.to_numpy(pn_p_g_item[2]))) + # print ("layer_hidden_list: \n", layer_hidden_list) + # print ("prune_synflow !!!one-hot input vector!!! absolute step tensor.to_numpy(loss)[0]: ", tensor.to_numpy(loss)[0]) + print ("score: \n", score) + + return score diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/README.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/README.md new file mode 100644 index 0000000000..90e0d8e17e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/README.md @@ -0,0 +1,2 @@ +(1) copy cnn_ms/pkg_model_code/model.py to ~/miniconda3/lib/python3.6/site-packages/singa/model.py +(2) enter cnn_ms/ and run "python train_ms_model.py ms_model_mlp mnist" diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/README.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/README.md new file mode 100644 index 0000000000..b081affa7e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/README.md @@ -0,0 +1,46 @@ + + +# Image Classification using Convolutional Neural Networks + +Examples inside this folder show how to train CNN models using +SINGA for image classification. + +* `data` includes the scripts for preprocessing image datasets. + Currently, MNIST, CIFAR10 and CIFAR100 are included. + +* `model` includes the CNN model construction codes by creating + a subclass of `Module` to wrap the neural network operations + of each model. Then computational graph is enabled to optimized + the memory and efficiency. + +* `autograd` includes the codes to train CNN models by calling the + [neural network operations](../../python/singa/autograd.py) imperatively. + The computational graph is not created. + +* `train_cnn.py` is the training script, which controls the training flow by + doing BackPropagation and SGD update. + +* `train_multiprocess.py` is the script for distributed training on a single + node with multiple GPUs; it uses Python's multiprocessing module and NCCL. + +* `train_mpi.py` is the script for distributed training (among multiple nodes) + using MPI and NCCL for communication. + +* `benchmark.py` tests the training throughput using `ResNet50` as the workload. \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/cifar10_multiprocess.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/cifar10_multiprocess.py new file mode 100644 index 0000000000..4b3cb0f43b --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/cifar10_multiprocess.py @@ -0,0 +1,43 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from resnet_cifar10 import * +import multiprocessing +import sys + +if __name__ == '__main__': + + # Generate a NCCL ID to be used for collective communication + nccl_id = singa.NcclIdHolder() + + # Configure the number of GPUs to be used + world_size = int(sys.argv[1]) + + # Testing the experimental partial-parameter update asynchronous training + partial_update = True + + process = [] + for local_rank in range(0, world_size): + process.append( + multiprocessing.Process(target=train_cifar10, + args=(True, local_rank, world_size, nccl_id, + partial_update))) + + for p in process: + p.start() \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_cnn.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_cnn.py new file mode 100644 index 0000000000..16752ceabe --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_cnn.py @@ -0,0 +1,304 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import singa_wrap as singa +from singa import autograd +from singa import layer +from singa import tensor +from singa import device +from singa import opt +import numpy as np +import os +import sys +import gzip +import codecs +import time + + +class CNN: + + def __init__(self): + self.conv1 = layer.Conv2d(1, 20, 5, padding=0) + self.conv2 = layer.Conv2d(20, 50, 5, padding=0) + self.linear1 = layer.Linear(4 * 4 * 50, 500) + self.linear2 = layer.Linear(500, 10) + self.pooling1 = layer.MaxPool2d(2, 2, padding=0) + self.pooling2 = layer.MaxPool2d(2, 2, padding=0) + self.relu1 = layer.ReLU() + self.relu2 = layer.ReLU() + self.relu3 = layer.ReLU() + self.flatten = layer.Flatten() + + def forward(self, x): + y = self.conv1(x) + y = self.relu1(y) + y = self.pooling1(y) + y = self.conv2(y) + y = self.relu2(y) + y = self.pooling2(y) + y = self.flatten(y) + y = self.linear1(y) + y = self.relu3(y) + y = self.linear2(y) + return y + + +def check_dataset_exist(dirpath): + if not os.path.exists(dirpath): + print( + 'The MNIST dataset does not exist. Please download the mnist dataset using download_mnist.py (e.g. python3 download_mnist.py)' + ) + sys.exit(0) + return dirpath + + +def load_dataset(): + train_x_path = '/tmp/train-images-idx3-ubyte.gz' + train_y_path = '/tmp/train-labels-idx1-ubyte.gz' + valid_x_path = '/tmp/t10k-images-idx3-ubyte.gz' + valid_y_path = '/tmp/t10k-labels-idx1-ubyte.gz' + + train_x = read_image_file(check_dataset_exist(train_x_path)).astype( + np.float32) + train_y = read_label_file(check_dataset_exist(train_y_path)).astype( + np.float32) + valid_x = read_image_file(check_dataset_exist(valid_x_path)).astype( + np.float32) + valid_y = read_label_file(check_dataset_exist(valid_y_path)).astype( + np.float32) + return train_x, train_y, valid_x, valid_y + + +def read_label_file(path): + with gzip.open(path, 'rb') as f: + data = f.read() + assert get_int(data[:4]) == 2049 + length = get_int(data[4:8]) + parsed = np.frombuffer(data, dtype=np.uint8, offset=8).reshape((length)) + return parsed + + +def get_int(b): + return int(codecs.encode(b, 'hex'), 16) + + +def read_image_file(path): + with gzip.open(path, 'rb') as f: + data = f.read() + assert get_int(data[:4]) == 2051 + length = get_int(data[4:8]) + num_rows = get_int(data[8:12]) + num_cols = get_int(data[12:16]) + parsed = np.frombuffer(data, dtype=np.uint8, offset=16).reshape( + (length, 1, num_rows, num_cols)) + return parsed + + +def to_categorical(y, num_classes): + y = np.array(y, dtype="int") + n = y.shape[0] + categorical = np.zeros((n, num_classes)) + categorical[np.arange(n), y] = 1 + categorical = categorical.astype(np.float32) + return categorical + + +def accuracy(pred, target): + y = np.argmax(pred, axis=1) + t = np.argmax(target, axis=1) + a = y == t + return np.array(a, "int").sum() + + +# Function to all reduce NUMPY accuracy and loss from multiple devices +def reduce_variable(variable, dist_opt, reducer): + reducer.copy_from_numpy(variable) + dist_opt.all_reduce(reducer.data) + dist_opt.wait() + output = tensor.to_numpy(reducer) + return output + + +# Function to sychronize SINGA TENSOR initial model parameters +def synchronize(tensor, dist_opt): + dist_opt.all_reduce(tensor.data) + dist_opt.wait() + tensor /= dist_opt.world_size + + +# Data augmentation +def augmentation(x, batch_size): + xpad = np.pad(x, [[0, 0], [0, 0], [4, 4], [4, 4]], 'symmetric') + for data_num in range(0, batch_size): + offset = np.random.randint(8, size=2) + x[data_num, :, :, :] = xpad[data_num, :, offset[0]:offset[0] + 28, + offset[1]:offset[1] + 28] + if_flip = np.random.randint(2) + if (if_flip): + x[data_num, :, :, :] = x[data_num, :, :, ::-1] + return x + + +# Data partition +def data_partition(dataset_x, dataset_y, global_rank, world_size): + data_per_rank = dataset_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + return dataset_x[idx_start:idx_end], dataset_y[idx_start:idx_end] + + +def train_mnist_cnn(DIST=False, + local_rank=None, + world_size=None, + nccl_id=None, + spars=0, + topK=False, + corr=True): + + # Define the hypermeters for the mnist_cnn + max_epoch = 10 + batch_size = 64 + sgd = opt.SGD(lr=0.005, momentum=0.9, weight_decay=1e-5) + + # Prepare training and valadiation data + train_x, train_y, test_x, test_y = load_dataset() + IMG_SIZE = 28 + num_classes = 10 + train_y = to_categorical(train_y, num_classes) + test_y = to_categorical(test_y, num_classes) + + # Normalization + train_x = train_x / 255 + test_x = test_x / 255 + + if DIST: + # For distributed GPU training + sgd = opt.DistOpt(sgd, + nccl_id=nccl_id, + local_rank=local_rank, + world_size=world_size) + dev = device.create_cuda_gpu_on(sgd.local_rank) + + # Dataset partition for distributed training + train_x, train_y = data_partition(train_x, train_y, sgd.global_rank, + sgd.world_size) + test_x, test_y = data_partition(test_x, test_y, sgd.global_rank, + sgd.world_size) + world_size = sgd.world_size + else: + # For single GPU + dev = device.create_cuda_gpu() + world_size = 1 + + # Create model + model = CNN() + + tx = tensor.Tensor((batch_size, 1, IMG_SIZE, IMG_SIZE), dev, tensor.float32) + ty = tensor.Tensor((batch_size, num_classes), dev, tensor.int32) + num_train_batch = train_x.shape[0] // batch_size + num_test_batch = test_x.shape[0] // batch_size + idx = np.arange(train_x.shape[0], dtype=np.int32) + + if DIST: + #Sychronize the initial parameters + autograd.training = True + x = np.random.randn(batch_size, 1, IMG_SIZE, + IMG_SIZE).astype(np.float32) + y = np.zeros(shape=(batch_size, num_classes), dtype=np.int32) + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out = model.forward(tx) + loss = autograd.softmax_cross_entropy(out, ty) + for p, g in autograd.backward(loss): + synchronize(p, sgd) + + # Training and evaulation loop + for epoch in range(max_epoch): + start_time = time.time() + np.random.shuffle(idx) + + if ((DIST == False) or (sgd.global_rank == 0)): + print('Starting Epoch %d:' % (epoch)) + + # Training phase + autograd.training = True + train_correct = np.zeros(shape=[1], dtype=np.float32) + test_correct = np.zeros(shape=[1], dtype=np.float32) + train_loss = np.zeros(shape=[1], dtype=np.float32) + + for b in range(num_train_batch): + x = train_x[idx[b * batch_size:(b + 1) * batch_size]] + x = augmentation(x, batch_size) + y = train_y[idx[b * batch_size:(b + 1) * batch_size]] + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out = model.forward(tx) + loss = autograd.softmax_cross_entropy(out, ty) + train_correct += accuracy(tensor.to_numpy(out), y) + train_loss += tensor.to_numpy(loss)[0] + if DIST: + if (spars == 0): + sgd.backward_and_update(loss, threshold=50000) + else: + sgd.backward_and_sparse_update(loss, + spars=spars, + topK=topK, + corr=corr) + else: + sgd(loss) + + if DIST: + # Reduce the evaluation accuracy and loss from multiple devices + reducer = tensor.Tensor((1,), dev, tensor.float32) + train_correct = reduce_variable(train_correct, sgd, reducer) + train_loss = reduce_variable(train_loss, sgd, reducer) + + # Output the training loss and accuracy + if ((DIST == False) or (sgd.global_rank == 0)): + print('Training loss = %f, training accuracy = %f' % + (train_loss, train_correct / + (num_train_batch * batch_size * world_size)), + flush=True) + + # Evaluation phase + autograd.training = False + for b in range(num_test_batch): + x = test_x[b * batch_size:(b + 1) * batch_size] + y = test_y[b * batch_size:(b + 1) * batch_size] + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out_test = model.forward(tx) + test_correct += accuracy(tensor.to_numpy(out_test), y) + + if DIST: + # Reduce the evaulation accuracy from multiple devices + test_correct = reduce_variable(test_correct, sgd, reducer) + + # Output the evaluation accuracy + if ((DIST == False) or (sgd.global_rank == 0)): + print('Evaluation accuracy = %f, Elapsed Time = %fs' % + (test_correct / (num_test_batch * batch_size * world_size), + time.time() - start_time), + flush=True) + + +if __name__ == '__main__': + + DIST = False + train_mnist_cnn(DIST=DIST) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_dist.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_dist.py new file mode 100644 index 0000000000..3586127c42 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_dist.py @@ -0,0 +1,25 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from mnist_cnn import * + +if __name__ == '__main__': + + DIST = True + train_mnist_cnn(DIST=DIST) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_multiprocess.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_multiprocess.py new file mode 100644 index 0000000000..f51344ff09 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/mnist_multiprocess.py @@ -0,0 +1,39 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from mnist_cnn import * +import multiprocessing +import sys + +if __name__ == '__main__': + + # Generate a NCCL ID to be used for collective communication + nccl_id = singa.NcclIdHolder() + + # Number of GPUs to be used + world_size = int(sys.argv[1]) + + process = [] + for local_rank in range(0, world_size): + process.append( + multiprocessing.Process(target=train_mnist_cnn, + args=(True, local_rank, world_size, nccl_id))) + + for p in process: + p.start() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_cifar10.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_cifar10.py new file mode 100644 index 0000000000..7541736994 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_cifar10.py @@ -0,0 +1,292 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +try: + import pickle +except ImportError: + import cPickle as pickle + +from singa import singa_wrap as singa +from singa import autograd +from singa import tensor +from singa import device +from singa import opt +from PIL import Image +import numpy as np +import os +import sys +import time + + +def load_dataset(filepath): + with open(filepath, 'rb') as fd: + try: + cifar10 = pickle.load(fd, encoding='latin1') + except TypeError: + cifar10 = pickle.load(fd) + image = cifar10['data'].astype(dtype=np.uint8) + image = image.reshape((-1, 3, 32, 32)) + label = np.asarray(cifar10['labels'], dtype=np.uint8) + label = label.reshape(label.size, 1) + return image, label + + +def load_train_data(dir_path='cifar-10-batches-py', num_batches=5): + labels = [] + batchsize = 10000 + images = np.empty((num_batches * batchsize, 3, 32, 32), dtype=np.uint8) + for did in range(1, num_batches + 1): + fname_train_data = dir_path + "/data_batch_{}".format(did) + image, label = load_dataset(check_dataset_exist(fname_train_data)) + images[(did - 1) * batchsize:did * batchsize] = image + labels.extend(label) + images = np.array(images, dtype=np.float32) + labels = np.array(labels, dtype=np.int32) + return images, labels + + +def load_test_data(dir_path='cifar-10-batches-py'): + images, labels = load_dataset(check_dataset_exist(dir_path + "/test_batch")) + return np.array(images, dtype=np.float32), np.array(labels, dtype=np.int32) + + +def check_dataset_exist(dirpath): + if not os.path.exists(dirpath): + print( + 'Please download the cifar10 dataset using download_data.py (e.g. python ~/singa/examples/cifar10/download_data.py py)' + ) + sys.exit(0) + return dirpath + + +def normalize_for_resnet(train_x, test_x): + mean = [0.4914, 0.4822, 0.4465] + std = [0.2023, 0.1994, 0.2010] + train_x /= 255 + test_x /= 255 + for ch in range(0, 2): + train_x[:, ch, :, :] -= mean[ch] + train_x[:, ch, :, :] /= std[ch] + test_x[:, ch, :, :] -= mean[ch] + test_x[:, ch, :, :] /= std[ch] + return train_x, test_x + + +def resize_dataset(x, IMG_SIZE): + num_data = x.shape[0] + dim = x.shape[1] + X = np.zeros(shape=(num_data, dim, IMG_SIZE, IMG_SIZE), dtype=np.float32) + for n in range(0, num_data): + for d in range(0, dim): + X[n, d, :, :] = np.array(Image.fromarray(x[n, d, :, :]).resize( + (IMG_SIZE, IMG_SIZE), Image.BILINEAR), + dtype=np.float32) + return X + + +def augmentation(x, batch_size): + xpad = np.pad(x, [[0, 0], [0, 0], [4, 4], [4, 4]], 'symmetric') + for data_num in range(0, batch_size): + offset = np.random.randint(8, size=2) + x[data_num, :, :, :] = xpad[data_num, :, offset[0]:offset[0] + 32, + offset[1]:offset[1] + 32] + if_flip = np.random.randint(2) + if (if_flip): + x[data_num, :, :, :] = x[data_num, :, :, ::-1] + return x + + +def accuracy(pred, target): + y = np.argmax(pred, axis=1) + t = np.argmax(target, axis=1) + a = y == t + return np.array(a, "int").sum() + + +def to_categorical(y, num_classes): + y = np.array(y, dtype="int") + n = y.shape[0] + categorical = np.zeros((n, num_classes)) + for i in range(0, n): + categorical[i, y[i]] = 1 + categorical = categorical.astype(np.float32) + return categorical + + +# Function to all reduce NUMPY accuracy and loss from multiple devices +def reduce_variable(variable, dist_opt, reducer): + reducer.copy_from_numpy(variable) + dist_opt.all_reduce(reducer.data) + dist_opt.wait() + output = tensor.to_numpy(reducer) + return output + + +# Function to sychronize SINGA TENSOR initial model parameters +def synchronize(tensor, dist_opt): + dist_opt.all_reduce(tensor.data) + dist_opt.wait() + tensor /= dist_opt.world_size + + +# Data partition +def data_partition(dataset_x, dataset_y, global_rank, world_size): + data_per_rank = dataset_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + return dataset_x[idx_start:idx_end], dataset_y[idx_start:idx_end] + + +def train_cifar10(DIST=False, + local_rank=None, + world_size=None, + nccl_id=None, + partial_update=False): + + # Define the hypermeters for the train_cifar10 + sgd = opt.SGD(lr=0.005, momentum=0.9, weight_decay=1e-5) + max_epoch = 5 + batch_size = 32 + + train_x, train_y = load_train_data() + test_x, test_y = load_test_data() + train_x, test_x = normalize_for_resnet(train_x, test_x) + IMG_SIZE = 224 + num_classes = 10 + + if DIST: + # For distributed GPU training + sgd = opt.DistOpt(sgd, + nccl_id=nccl_id, + local_rank=local_rank, + world_size=world_size) + dev = device.create_cuda_gpu_on(sgd.local_rank) + + # Dataset partition for distributed training + train_x, train_y = data_partition(train_x, train_y, sgd.global_rank, + sgd.world_size) + test_x, test_y = data_partition(test_x, test_y, sgd.global_rank, + sgd.world_size) + world_size = sgd.world_size + else: + # For single GPU + dev = device.create_cuda_gpu() + world_size = 1 + + from resnet import resnet50 + model = resnet50(num_classes=num_classes) + + tx = tensor.Tensor((batch_size, 3, IMG_SIZE, IMG_SIZE), dev, tensor.float32) + ty = tensor.Tensor((batch_size,), dev, tensor.int32) + num_train_batch = train_x.shape[0] // batch_size + num_test_batch = test_x.shape[0] // batch_size + idx = np.arange(train_x.shape[0], dtype=np.int32) + + if DIST: + # Sychronize the initial parameters + autograd.training = True + x = np.random.randn(batch_size, 3, IMG_SIZE, + IMG_SIZE).astype(np.float32) + y = np.zeros(shape=(batch_size,), dtype=np.int32) + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out = model(tx) + loss = autograd.softmax_cross_entropy(out, ty) + param = [] + for p, _ in autograd.backward(loss): + synchronize(p, sgd) + param.append(p) + + for epoch in range(max_epoch): + start_time = time.time() + np.random.shuffle(idx) + + if ((DIST == False) or (sgd.global_rank == 0)): + print('Starting Epoch %d:' % (epoch)) + + # Training phase + autograd.training = True + train_correct = np.zeros(shape=[1], dtype=np.float32) + test_correct = np.zeros(shape=[1], dtype=np.float32) + train_loss = np.zeros(shape=[1], dtype=np.float32) + + for b in range(num_train_batch): + x = train_x[idx[b * batch_size:(b + 1) * batch_size]] + x = augmentation(x, batch_size) + x = resize_dataset(x, IMG_SIZE) + y = train_y[idx[b * batch_size:(b + 1) * batch_size]] + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out = model(tx) + loss = autograd.softmax_cross_entropy(out, ty) + train_correct += accuracy(tensor.to_numpy(out), + to_categorical(y, num_classes)).astype( + np.float32) + train_loss += tensor.to_numpy(loss)[0] + if not partial_update: + sgd.backward_and_update(loss) + else: + sgd.backward_and_partial_update(loss) + + if DIST: + # Reduce the evaluation accuracy and loss from multiple devices + reducer = tensor.Tensor((1,), dev, tensor.float32) + train_correct = reduce_variable(train_correct, sgd, reducer) + train_loss = reduce_variable(train_loss, sgd, reducer) + + # Output the training loss and accuracy + if ((DIST == False) or (sgd.global_rank == 0)): + print('Training loss = %f, training accuracy = %f' % + (train_loss, train_correct / + (num_train_batch * batch_size * world_size)), + flush=True) + + if partial_update: + # Sychronize parameters before evaluation phase + for p in param: + synchronize(p, sgd) + + # Evaulation phase + autograd.training = False + for b in range(num_test_batch): + x = test_x[b * batch_size:(b + 1) * batch_size] + x = resize_dataset(x, IMG_SIZE) + y = test_y[b * batch_size:(b + 1) * batch_size] + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out_test = model(tx) + test_correct += accuracy(tensor.to_numpy(out_test), + to_categorical(y, num_classes)) + + if DIST: + # Reduce the evaulation accuracy from multiple devices + test_correct = reduce_variable(test_correct, sgd, reducer) + + # Output the evaluation accuracy + if ((DIST == False) or (sgd.global_rank == 0)): + print('Evaluation accuracy = %f, Elapsed Time = %fs' % + (test_correct / (num_test_batch * batch_size * world_size), + time.time() - start_time), + flush=True) + + +if __name__ == '__main__': + + DIST = False + train_cifar10(DIST=DIST) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_dist.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_dist.py new file mode 100644 index 0000000000..6f9b56ceeb --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/resnet_dist.py @@ -0,0 +1,87 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +# the code is modified from +# https://github.com/pytorch/vision/blob/master/torchvision/models/resnet.py + +from singa import autograd +from singa import tensor +from singa import device +from singa import opt + +import numpy as np +from tqdm import trange + +if __name__ == "__main__": + sgd = opt.SGD(lr=0.1, momentum=0.9, weight_decay=1e-5) + sgd = opt.DistOpt(sgd) + + if (sgd.global_rank == 0): + print("Start intialization...........", flush=True) + + dev = device.create_cuda_gpu_on(sgd.local_rank) + + from resnet import resnet50 + model = resnet50() + + niters = 100 + batch_size = 32 + IMG_SIZE = 224 + + tx = tensor.Tensor((batch_size, 3, IMG_SIZE, IMG_SIZE), dev) + ty = tensor.Tensor((batch_size,), dev, tensor.int32) + autograd.training = True + x = np.random.randn(batch_size, 3, IMG_SIZE, IMG_SIZE).astype(np.float32) + y = np.random.randint(0, 1000, batch_size, dtype=np.int32) + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + + import time + + dev.Sync() + start = time.time() + fd = 0 + softmax = 0 + update = 0 + with trange(niters) as t: + for _ in t: + dev.Sync() + tick = time.time() + x = model(tx) + dev.Sync() + fd += time.time() - tick + tick = time.time() + loss = autograd.softmax_cross_entropy(x, ty) + dev.Sync() + softmax += time.time() - tick + sgd.backward_and_update(loss) + + dev.Sync() + end = time.time() + throughput = float(sgd.world_size * niters * batch_size) / (end - start) + titer = (end - start) / float(niters) + tforward = float(fd) / float(niters) + tsoftmax = float(softmax) / float(niters) + tbackward = titer - tforward - tsoftmax + + if (sgd.global_rank == 0): + print("\nThroughput = {} per second".format(throughput), flush=True) + print("Total={}, forward={}, softmax={}, backward={}".format( + titer, tforward, tsoftmax, tbackward), + flush=True) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/sparsification_mnist.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/sparsification_mnist.py new file mode 100644 index 0000000000..315605acd6 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/sparsification_mnist.py @@ -0,0 +1,45 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from mnist_cnn import * +import multiprocessing +import sys + +if __name__ == '__main__': + + # Generate a NCCL ID to be used for collective communication + nccl_id = singa.NcclIdHolder() + + # Number of GPUs to be used + world_size = int(sys.argv[1]) + + # Use sparsification with parameters + topK = False # When topK = False, Sparsification based on a constant absolute threshold + corr = True # If True, uses local accumulate gradient for the correction + sparsThreshold = 0.05 # The constant absolute threshold for sparsification + + process = [] + for local_rank in range(0, world_size): + process.append( + multiprocessing.Process(target=train_mnist_cnn, + args=(True, local_rank, world_size, nccl_id, + sparsThreshold, topK, corr))) + + for p in process: + p.start() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/xceptionnet.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/xceptionnet.py new file mode 100644 index 0000000000..8fb23d8cbb --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/autograd/xceptionnet.py @@ -0,0 +1,303 @@ +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# ============================================================================= + +from singa import autograd +from singa import tensor +from singa import device +from singa import layer +from singa import opt + +import numpy as np +from tqdm import trange + +# the code is modified from +# https://github.com/Cadene/pretrained-models.pytorch/blob/master/pretrainedmodels/models/xception.py + + +class Block(layer.Layer): + + def __init__(self, + in_filters, + out_filters, + reps, + strides=1, + padding=0, + start_with_relu=True, + grow_first=True): + super(Block, self).__init__() + + if out_filters != in_filters or strides != 1: + self.skip = layer.Conv2d(in_filters, + out_filters, + 1, + stride=strides, + padding=padding, + bias=False) + self.skipbn = layer.BatchNorm2d(out_filters) + else: + self.skip = None + + self.layers = [] + + filters = in_filters + if grow_first: + self.layers.append(layer.ReLU()) + self.layers.append( + layer.SeparableConv2d(in_filters, + out_filters, + 3, + stride=1, + padding=1, + bias=False)) + self.layers.append(layer.BatchNorm2d(out_filters)) + filters = out_filters + + for i in range(reps - 1): + self.layers.append(layer.ReLU()) + self.layers.append( + layer.SeparableConv2d(filters, + filters, + 3, + stride=1, + padding=1, + bias=False)) + self.layers.append(layer.BatchNorm2d(filters)) + + if not grow_first: + self.layers.append(layer.ReLU()) + self.layers.append( + layer.SeparableConv2d(in_filters, + out_filters, + 3, + stride=1, + padding=1, + bias=False)) + self.layers.append(layer.BatchNorm2d(out_filters)) + + if not start_with_relu: + self.layers = self.layers[1:] + else: + self.layers[0] = layer.ReLU() + + if strides != 1: + self.layers.append(layer.MaxPool2d(3, strides, padding + 1)) + + self.register_layers(*self.layers) + + self.add = layer.Add() + + def forward(self, x): + y = self.layers[0](x) + for layer in self.layers[1:]: + if isinstance(y, tuple): + y = y[0] + y = layer(y) + + if self.skip is not None: + skip = self.skip(x) + skip = self.skipbn(skip) + else: + skip = x + y = self.add(y, skip) + return y + + +__all__ = ['Xception'] + + +class Xception(layer.Layer): + """ + Xception optimized for the ImageNet dataset, as specified in + https://arxiv.org/pdf/1610.02357.pdf + """ + + def __init__(self, num_classes=1000): + """ Constructor + Args: + num_classes: number of classes + """ + super(Xception, self).__init__() + self.num_classes = num_classes + + self.conv1 = layer.Conv2d(3, 32, 3, 2, 0, bias=False) + self.bn1 = layer.BatchNorm2d(32) + self.relu1 = layer.ReLU() + + self.conv2 = layer.Conv2d(32, 64, 3, 1, 1, bias=False) + self.bn2 = layer.BatchNorm2d(64) + self.relu2 = layer.ReLU() + # do relu here + + self.block1 = Block(64, + 128, + 2, + 2, + padding=0, + start_with_relu=False, + grow_first=True) + self.block2 = Block(128, + 256, + 2, + 2, + padding=0, + start_with_relu=True, + grow_first=True) + self.block3 = Block(256, + 728, + 2, + 2, + padding=0, + start_with_relu=True, + grow_first=True) + + self.block4 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block5 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block6 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block7 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + + self.block8 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block9 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block10 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block11 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + + self.block12 = Block(728, + 1024, + 2, + 2, + start_with_relu=True, + grow_first=False) + + self.conv3 = layer.SeparableConv2d(1024, 1536, 3, 1, 1) + self.bn3 = layer.BatchNorm2d(1536) + self.relu3 = layer.ReLU() + + # Relu Layer + self.conv4 = layer.SeparableConv2d(1536, 2048, 3, 1, 1) + self.bn4 = layer.BatchNorm2d(2048) + + self.relu4 = layer.ReLU() + self.globalpooling = layer.MaxPool2d(10, 1) + self.flatten = layer.Flatten() + self.fc = layer.Linear(2048, num_classes) + + def features(self, input): + x = self.conv1(input) + x = self.bn1(x) + x = self.relu1(x) + + x = self.conv2(x) + x = self.bn2(x) + x = self.relu2(x) + + x = self.block1(x) + x = self.block2(x) + x = self.block3(x) + x = self.block4(x) + x = self.block5(x) + x = self.block6(x) + x = self.block7(x) + x = self.block8(x) + x = self.block9(x) + x = self.block10(x) + x = self.block11(x) + x = self.block12(x) + + x = self.conv3(x) + x = self.bn3(x) + x = self.relu3(x) + + x = self.conv4(x) + x = self.bn4(x) + return x + + def logits(self, features): + x = self.relu4(features) + x = self.globalpooling(x) + x = self.flatten(x) + x = self.fc(x) + return x + + def forward(self, input): + x = self.features(input) + x = self.logits(x) + return x + + +if __name__ == '__main__': + model = Xception(num_classes=1000) + print('Start intialization............') + dev = device.create_cuda_gpu_on(0) + #dev = device.create_cuda_gpu() + + niters = 20 + batch_size = 16 + IMG_SIZE = 299 + sgd = opt.SGD(lr=0.1, momentum=0.9, weight_decay=1e-5) + + tx = tensor.Tensor((batch_size, 3, IMG_SIZE, IMG_SIZE), dev) + ty = tensor.Tensor((batch_size,), dev, tensor.int32) + autograd.training = True + x = np.random.randn(batch_size, 3, IMG_SIZE, IMG_SIZE).astype(np.float32) + y = np.random.randint(0, 1000, batch_size, dtype=np.int32) + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + + with trange(niters) as t: + for _ in t: + x = model(tx) + loss = autograd.softmax_cross_entropy(x, ty) + sgd(loss) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/benchmark.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/benchmark.py new file mode 100644 index 0000000000..9f69feee08 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/benchmark.py @@ -0,0 +1,121 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +# the code is modified from +# https://github.com/pytorch/vision/blob/master/torchvision/models/resnet.py + +from singa import opt +from singa import device +from singa import tensor + +import argparse +import time +import numpy as np +from tqdm import trange + + +def train_resnet(DIST=True, graph=True, sequential=False, verbosity=0): + + # Define the hypermeters for the train_resnet + niters = 100 + batch_size = 32 + sgd = opt.SGD(lr=0.1, momentum=0.9, weight_decay=1e-5) + + IMG_SIZE = 224 + + # For distributed training, sequential has better throughput in the current version + if DIST == True: + sgd = opt.DistOpt(sgd) + world_size = sgd.world_size + local_rank = sgd.local_rank + global_rank = sgd.global_rank + sequential = True + else: + local_rank = 0 + world_size = 1 + global_rank = 0 + sequential = False + + dev = device.create_cuda_gpu_on(local_rank) + + tx = tensor.Tensor((batch_size, 3, IMG_SIZE, IMG_SIZE), dev) + ty = tensor.Tensor((batch_size,), dev, tensor.int32) + x = np.random.randn(batch_size, 3, IMG_SIZE, IMG_SIZE).astype(np.float32) + y = np.random.randint(0, 1000, batch_size, dtype=np.int32) + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + + dev.SetVerbosity(verbosity) + dev.SetSkipIteration(5) + + # Construct the model + from model import resnet + model = resnet.resnet50(num_channels=3, num_classes=1000) + + model.train() + model.set_optimizer(sgd) + model.compile([tx], is_train=True, use_graph=graph, sequential=sequential) + + # Train model + dev.Sync() + start = time.time() + with trange(niters) as t: + for _ in t: + model(tx, ty, dist_option='fp32', spars=None) + + dev.Sync() + end = time.time() + titer = (end - start) / float(niters) + throughput = float(niters * batch_size * world_size) / (end - start) + if global_rank == 0: + print("\nThroughput = {} per second".format(throughput), flush=True) + print("TotalTime={}".format(end - start), flush=True) + print("Total={}".format(titer), flush=True) + dev.PrintTimeProfiling() + + +if __name__ == "__main__": + + parser = argparse.ArgumentParser( + description='Throughput test using Resnet 50') + parser.add_argument('--dist', + '--enable-dist', + default='False', + action='store_true', + help='enable distributed training', + dest='DIST') + parser.add_argument('--no-graph', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('--verbosity', + '--log-verbosity', + default=0, + type=int, + help='logging verbosity', + dest='verbosity') + + args = parser.parse_args() + + train_resnet(DIST=args.DIST, + graph=args.graph, + sequential=False, + verbosity=args.verbosity) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar10.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar10.py new file mode 100644 index 0000000000..5caaf30f44 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar10.py @@ -0,0 +1,89 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +try: + import pickle +except ImportError: + import cPickle as pickle + +import numpy as np +import os +import sys + + +def load_dataset(filepath): + with open(filepath, 'rb') as fd: + try: + cifar10 = pickle.load(fd, encoding='latin1') + except TypeError: + cifar10 = pickle.load(fd) + image = cifar10['data'].astype(dtype=np.uint8) + image = image.reshape((-1, 3, 32, 32)) + label = np.asarray(cifar10['labels'], dtype=np.uint8) + label = label.reshape(label.size, 1) + return image, label + + +def load_train_data(dir_path='/tmp/cifar-10-batches-py', num_batches=5): # need to save to specific local directories + labels = [] + batchsize = 10000 + images = np.empty((num_batches * batchsize, 3, 32, 32), dtype=np.uint8) + for did in range(1, num_batches + 1): + fname_train_data = dir_path + "/data_batch_{}".format(did) + image, label = load_dataset(check_dataset_exist(fname_train_data)) + images[(did - 1) * batchsize:did * batchsize] = image + labels.extend(label) + images = np.array(images, dtype=np.float32) + labels = np.array(labels, dtype=np.int32) + return images, labels + + +def load_test_data(dir_path='/tmp/cifar-10-batches-py'): # need to save to specific local directories + images, labels = load_dataset(check_dataset_exist(dir_path + "/test_batch")) + return np.array(images, dtype=np.float32), np.array(labels, dtype=np.int32) + + +def check_dataset_exist(dirpath): + if not os.path.exists(dirpath): + print( + 'Please download the cifar10 dataset using python data/download_cifar10.py' + ) + sys.exit(0) + return dirpath + + +def normalize(train_x, val_x): + mean = [0.4914, 0.4822, 0.4465] + std = [0.2023, 0.1994, 0.2010] + train_x /= 255 + val_x /= 255 + for ch in range(0, 2): + train_x[:, ch, :, :] -= mean[ch] + train_x[:, ch, :, :] /= std[ch] + val_x[:, ch, :, :] -= mean[ch] + val_x[:, ch, :, :] /= std[ch] + return train_x, val_x + +def load(): + train_x, train_y = load_train_data() + val_x, val_y = load_test_data() + train_x, val_x = normalize(train_x, val_x) + train_y = train_y.flatten() + val_y = val_y.flatten() + return train_x, train_y, val_x, val_y diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar100.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar100.py new file mode 100644 index 0000000000..88b943f074 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/cifar100.py @@ -0,0 +1,81 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +try: + import pickle +except ImportError: + import cPickle as pickle + +import numpy as np +import os +import sys + + +def load_dataset(filepath): + with open(filepath, 'rb') as fd: + try: + cifar100 = pickle.load(fd, encoding='latin1') + except TypeError: + cifar100 = pickle.load(fd) + image = cifar100['data'].astype(dtype=np.uint8) + image = image.reshape((-1, 3, 32, 32)) + label = np.asarray(cifar100['fine_labels'], dtype=np.uint8) + label = label.reshape(label.size, 1) + return image, label + + +def load_train_data(dir_path='/tmp/cifar-100-python'): + images, labels = load_dataset(check_dataset_exist(dir_path + "/train")) + return np.array(images, dtype=np.float32), np.array(labels, dtype=np.int32) + + +def load_test_data(dir_path='/tmp/cifar-100-python'): + images, labels = load_dataset(check_dataset_exist(dir_path + "/test")) + return np.array(images, dtype=np.float32), np.array(labels, dtype=np.int32) + + +def check_dataset_exist(dirpath): + if not os.path.exists(dirpath): + print( + 'Please download the cifar100 dataset using python data/download_cifar100.py' + ) + sys.exit(0) + return dirpath + + +def normalize(train_x, val_x): + mean = [0.4914, 0.4822, 0.4465] + std = [0.2023, 0.1994, 0.2010] + train_x /= 255 + val_x /= 255 + for ch in range(0, 2): + train_x[:, ch, :, :] -= mean[ch] + train_x[:, ch, :, :] /= std[ch] + val_x[:, ch, :, :] -= mean[ch] + val_x[:, ch, :, :] /= std[ch] + return train_x, val_x + + +def load(): + train_x, train_y = load_train_data() + val_x, val_y = load_test_data() + train_x, val_x = normalize(train_x, val_x) + train_y = train_y.flatten() + val_y = val_y.flatten() + return train_x, train_y, val_x, val_y diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar10.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar10.py new file mode 100755 index 0000000000..8e44679218 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar10.py @@ -0,0 +1,49 @@ +#!/usr/bin/env python +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +from __future__ import print_function +from future import standard_library +standard_library.install_aliases() +import urllib.request, urllib.parse, urllib.error +import tarfile +import os +import sys + + +def extract_tarfile(filepath): + if os.path.exists(filepath): + print('The tar file does exist. Extracting it now..') + with tarfile.open(filepath, 'r') as f: + f.extractall('/tmp/') # need to specify a local directory + print('Finished!') + sys.exit(0) + + +def do_download(dirpath, gzfile, url): + print('Downloading CIFAR from %s' % (url)) + urllib.request.urlretrieve(url, gzfile) + extract_tarfile(gzfile) + print('Finished!') + + +if __name__ == '__main__': + dirpath = '/tmp/' # need to specify a local directory + gzfile = dirpath + 'cifar-10-python.tar.gz' + url = 'http://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz' + do_download(dirpath, gzfile, url) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar100.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar100.py new file mode 100755 index 0000000000..5f1e21b788 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_cifar100.py @@ -0,0 +1,26 @@ +#!/usr/bin/env python +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +from download_cifar10 import do_download + +if __name__ == '__main__': + dirpath = '/tmp/' # need to specify a local directory + gzfile = dirpath + 'cifar-100-python.tar.gz' + url = 'http://www.cs.toronto.edu/~kriz/cifar-100-python.tar.gz' + do_download(dirpath, gzfile, url) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_mnist.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_mnist.py new file mode 100644 index 0000000000..65acb0e28b --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/download_mnist.py @@ -0,0 +1,49 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +import os +import urllib.request + + +def check_exist_or_download(url): + + download_dir = '/tmp/' + name = url.rsplit('/', 1)[-1] + filename = os.path.join(download_dir, name) + + if not os.path.isfile(filename): + print("Downloading %s" % url) + urllib.request.urlretrieve(url, filename) + else: + print("Already Downloaded: %s" % url) + + +if __name__ == '__main__': + + #url of the mnist dataset + train_x_url = 'http://yann.lecun.com/exdb/mnist/train-images-idx3-ubyte.gz' + train_y_url = 'http://yann.lecun.com/exdb/mnist/train-labels-idx1-ubyte.gz' + valid_x_url = 'http://yann.lecun.com/exdb/mnist/t10k-images-idx3-ubyte.gz' + valid_y_url = 'http://yann.lecun.com/exdb/mnist/t10k-labels-idx1-ubyte.gz' + + #download the mnist dataset + check_exist_or_download(train_x_url) + check_exist_or_download(train_y_url) + check_exist_or_download(valid_x_url) + check_exist_or_download(valid_y_url) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/mnist.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/mnist.py new file mode 100644 index 0000000000..b25bf5e67d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/data/mnist.py @@ -0,0 +1,91 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +import numpy as np +import os +import sys +import gzip +import codecs + + +def check_dataset_exist(dirpath): + if not os.path.exists(dirpath): + print( + 'The MNIST dataset does not exist. Please download the mnist dataset using python data/download_mnist.py' + ) + sys.exit(0) + return dirpath + + +def load_dataset(): + train_x_path = '/tmp/train-images-idx3-ubyte.gz' # need to change to local disk + train_y_path = '/tmp/train-labels-idx1-ubyte.gz' # need to change to local disk + valid_x_path = '/tmp/t10k-images-idx3-ubyte.gz' # need to change to local disk + valid_y_path = '/tmp/t10k-labels-idx1-ubyte.gz' # need to change to local disk + + train_x = read_image_file(check_dataset_exist(train_x_path)).astype( + np.float32) + train_y = read_label_file(check_dataset_exist(train_y_path)).astype( + np.float32) + valid_x = read_image_file(check_dataset_exist(valid_x_path)).astype( + np.float32) + valid_y = read_label_file(check_dataset_exist(valid_y_path)).astype( + np.float32) + return train_x, train_y, valid_x, valid_y + + +def read_label_file(path): + with gzip.open(path, 'rb') as f: + data = f.read() + assert get_int(data[:4]) == 2049 + length = get_int(data[4:8]) + parsed = np.frombuffer(data, dtype=np.uint8, offset=8).reshape((length)) + return parsed + + +def get_int(b): + return int(codecs.encode(b, 'hex'), 16) + + +def read_image_file(path): + with gzip.open(path, 'rb') as f: + data = f.read() + assert get_int(data[:4]) == 2051 + length = get_int(data[4:8]) + num_rows = get_int(data[8:12]) + num_cols = get_int(data[12:16]) + parsed = np.frombuffer(data, dtype=np.uint8, offset=16).reshape( + (length, 1, num_rows, num_cols)) + return parsed + + +def normalize(train_x, val_x): + train_x /= 255 + val_x /= 255 + return train_x, val_x + + +def load(): + train_x, train_y, val_x, val_y = load_dataset() + train_x, val_x = normalize(train_x, val_x) + train_x = train_x.astype(np.float32) + val_x = val_x.astype(np.float32) + train_y = train_y.astype(np.int32) + val_y = val_y.astype(np.int32) + return train_x, train_y, val_x, val_y diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/alexnet.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/alexnet.py new file mode 100644 index 0000000000..cad7b1e3f3 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/alexnet.py @@ -0,0 +1,119 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import layer +from singa import model + + +class AlexNet(model.Model): + + def __init__(self, num_classes=10, num_channels=1): + super(AlexNet, self).__init__() + self.num_classes = num_classes + self.input_size = 224 + self.dimension = 4 + self.conv1 = layer.Conv2d(num_channels, 64, 11, stride=4, padding=2) + self.conv2 = layer.Conv2d(64, 192, 5, padding=2) + self.conv3 = layer.Conv2d(192, 384, 3, padding=1) + self.conv4 = layer.Conv2d(384, 256, 3, padding=1) + self.conv5 = layer.Conv2d(256, 256, 3, padding=1) + self.linear1 = layer.Linear(4096) + self.linear2 = layer.Linear(4096) + self.linear3 = layer.Linear(num_classes) + self.pooling1 = layer.MaxPool2d(2, 2, padding=0) + self.pooling2 = layer.MaxPool2d(2, 2, padding=0) + self.pooling3 = layer.MaxPool2d(2, 2, padding=0) + self.avg_pooling1 = layer.AvgPool2d(3, 2, padding=0) + self.relu1 = layer.ReLU() + self.relu2 = layer.ReLU() + self.relu3 = layer.ReLU() + self.relu4 = layer.ReLU() + self.relu5 = layer.ReLU() + self.relu6 = layer.ReLU() + self.relu7 = layer.ReLU() + self.flatten = layer.Flatten() + self.dropout1 = layer.Dropout() + self.dropout2 = layer.Dropout() + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + + def forward(self, x): + y = self.conv1(x) + y = self.relu1(y) + y = self.pooling1(y) + y = self.conv2(y) + y = self.relu2(y) + y = self.pooling2(y) + y = self.conv3(y) + y = self.relu3(y) + y = self.conv4(y) + y = self.relu4(y) + y = self.conv5(y) + y = self.relu5(y) + y = self.pooling3(y) + y = self.avg_pooling1(y) + y = self.flatten(y) + y = self.dropout1(y) + y = self.linear1(y) + y = self.relu6(y) + y = self.dropout2(y) + y = self.linear2(y) + y = self.relu7(y) + y = self.linear3(y) + return y + + def train_one_batch(self, x, y, dist_option, spars): + out = self.forward(x) + loss = self.softmax_cross_entropy(out, y) + + if dist_option == 'plain': + self.optimizer(loss) + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + return out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def create_model(pretrained=False, **kwargs): + """Constructs a AlexNet model. + + Args: + pretrained (bool): If True, returns a pre-trained model. + + Returns: + The created AlexNet model. + + """ + model = AlexNet(**kwargs) + + return model + + +__all__ = ['AlexNet', 'create_model'] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/cnn.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/cnn.py new file mode 100644 index 0000000000..3877e83af5 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/cnn.py @@ -0,0 +1,90 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import layer +from singa import model + + +class CNN(model.Model): + + def __init__(self, num_classes=10, num_channels=1): + super(CNN, self).__init__() + self.num_classes = num_classes + self.input_size = 28 + self.dimension = 4 + self.conv1 = layer.Conv2d(num_channels, 20, 5, padding=0, activation="RELU") + self.conv2 = layer.Conv2d(20, 50, 5, padding=0, activation="RELU") + self.linear1 = layer.Linear(500) + self.linear2 = layer.Linear(num_classes) + self.pooling1 = layer.MaxPool2d(2, 2, padding=0) + self.pooling2 = layer.MaxPool2d(2, 2, padding=0) + self.relu = layer.ReLU() + self.flatten = layer.Flatten() + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + + def forward(self, x): + y = self.conv1(x) + y = self.pooling1(y) + y = self.conv2(y) + y = self.pooling2(y) + y = self.flatten(y) + y = self.linear1(y) + y = self.relu(y) + y = self.linear2(y) + return y + + def train_one_batch(self, x, y, dist_option, spars): + out = self.forward(x) + loss = self.softmax_cross_entropy(out, y) + + if dist_option == 'plain': + self.optimizer(loss) + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + return out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def create_model(pretrained=False, **kwargs): + """Constructs a CNN model. + + Args: + pretrained (bool): If True, returns a pre-trained model. + + Returns: + The created CNN model. + """ + model = CNN(**kwargs) + + return model + + +__all__ = ['CNN', 'create_model'] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/resnet.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/resnet.py new file mode 100644 index 0000000000..28b5f99492 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/resnet.py @@ -0,0 +1,300 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +# the code is modified from +# https://github.com/pytorch/vision/blob/master/torchvision/models/resnet.py + +from singa import layer +from singa import model + + +def conv3x3(in_planes, out_planes, stride=1): + """3x3 convolution with padding""" + return layer.Conv2d( + in_planes, + out_planes, + 3, + stride=stride, + padding=1, + bias=False, + ) + + +class BasicBlock(layer.Layer): + expansion = 1 + + def __init__(self, inplanes, planes, stride=1, downsample=None): + super(BasicBlock, self).__init__() + self.conv1 = conv3x3(inplanes, planes, stride) + self.bn1 = layer.BatchNorm2d(planes) + self.conv2 = conv3x3(planes, planes) + self.bn2 = layer.BatchNorm2d(planes) + self.relu1 = layer.ReLU() + self.add = layer.Add() + self.relu2 = layer.ReLU() + self.downsample = downsample + self.stride = stride + + def forward(self, x): + residual = x + + out = self.conv1(x) + out = self.bn1(out) + out = self.relu1(out) + + out = self.conv2(out) + out = self.bn2(out) + + if self.downsample is not None: + residual = self.downsample(x) + + out = self.add(out, residual) + out = self.relu2(out) + + return out + + +class Bottleneck(layer.Layer): + expansion = 4 + + def __init__(self, inplanes, planes, stride=1, downsample=None): + super(Bottleneck, self).__init__() + self.conv1 = layer.Conv2d(inplanes, planes, 1, bias=False) + self.bn1 = layer.BatchNorm2d(planes) + self.relu1 = layer.ReLU() + self.conv2 = layer.Conv2d(planes, + planes, + 3, + stride=stride, + padding=1, + bias=False) + self.bn2 = layer.BatchNorm2d(planes) + self.relu2 = layer.ReLU() + self.conv3 = layer.Conv2d(planes, + planes * self.expansion, + 1, + bias=False) + self.bn3 = layer.BatchNorm2d(planes * self.expansion) + + self.add = layer.Add() + self.relu3 = layer.ReLU() + + self.downsample = downsample + self.stride = stride + + def forward(self, x): + residual = x + + out = self.conv1(x) + out = self.bn1(out) + out = self.relu1(out) + + out = self.conv2(out) + out = self.bn2(out) + out = self.relu2(out) + + out = self.conv3(out) + out = self.bn3(out) + + if self.downsample is not None: + residual = self.downsample(x) + + out = self.add(out, residual) + out = self.relu3(out) + + return out + + +__all__ = [ + 'ResNet', 'resnet18', 'resnet34', 'resnet50', 'resnet101', 'resnet152' +] + + +class ResNet(model.Model): + + def __init__(self, block, layers, num_classes=10, num_channels=3): + self.inplanes = 64 + super(ResNet, self).__init__() + self.num_classes = num_classes + self.input_size = 224 + self.dimension = 4 + self.conv1 = layer.Conv2d(num_channels, + 64, + 7, + stride=2, + padding=3, + bias=False) + self.bn1 = layer.BatchNorm2d(64) + self.relu = layer.ReLU() + self.maxpool = layer.MaxPool2d(kernel_size=3, stride=2, padding=1) + self.layer1, layers1 = self._make_layer(block, 64, layers[0]) + self.layer2, layers2 = self._make_layer(block, 128, layers[1], stride=2) + self.layer3, layers3 = self._make_layer(block, 256, layers[2], stride=2) + self.layer4, layers4 = self._make_layer(block, 512, layers[3], stride=2) + self.avgpool = layer.AvgPool2d(7, stride=1) + self.flatten = layer.Flatten() + self.fc = layer.Linear(num_classes) + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + + self.register_layers(*layers1, *layers2, *layers3, *layers4) + + def _make_layer(self, block, planes, blocks, stride=1): + downsample = None + if stride != 1 or self.inplanes != planes * block.expansion: + conv = layer.Conv2d( + self.inplanes, + planes * block.expansion, + 1, + stride=stride, + bias=False, + ) + bn = layer.BatchNorm2d(planes * block.expansion) + + def _downsample(x): + return bn(conv(x)) + + downsample = _downsample + + layers = [] + layers.append(block(self.inplanes, planes, stride, downsample)) + self.inplanes = planes * block.expansion + for i in range(1, blocks): + layers.append(block(self.inplanes, planes)) + + def forward(x): + for layer in layers: + x = layer(x) + return x + + return forward, layers + + def forward(self, x): + x = self.conv1(x) + x = self.bn1(x) + x = self.relu(x) + x = self.maxpool(x) + + x = self.layer1(x) + x = self.layer2(x) + x = self.layer3(x) + x = self.layer4(x) + + x = self.avgpool(x) + x = self.flatten(x) + x = self.fc(x) + + return x + + def train_one_batch(self, x, y, dist_option, spars): + out = self.forward(x) + loss = self.softmax_cross_entropy(out, y) + + if dist_option == 'plain': + self.optimizer(loss) + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + return out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def resnet18(pretrained=False, **kwargs): + """Constructs a ResNet-18 model. + + Args: + pretrained (bool): If True, returns a model pre-trained on ImageNet. + + Returns: + The created ResNet-18 model. + """ + model = ResNet(BasicBlock, [2, 2, 2, 2], **kwargs) + + return model + + +def resnet34(pretrained=False, **kwargs): + """Constructs a ResNet-34 model. + + Args: + pretrained (bool): If True, returns a model pre-trained on ImageNet. + + Returns: + The created ResNet-34 model. + """ + model = ResNet(BasicBlock, [3, 4, 6, 3], **kwargs) + + return model + + +def resnet50(pretrained=False, **kwargs): + """Constructs a ResNet-50 model. + + Args: + pretrained (bool): If True, returns a model pre-trained on ImageNet. + + Returns: + The created ResNet-50 model. + """ + model = ResNet(Bottleneck, [3, 4, 6, 3], **kwargs) + + return model + + +def resnet101(pretrained=False, **kwargs): + """Constructs a ResNet-101 model. + + Args: + pretrained (bool): If True, returns a model pre-trained on ImageNet. + + Returns: + The created ResNet-101 model. + """ + model = ResNet(Bottleneck, [3, 4, 23, 3], **kwargs) + + return model + + +def resnet152(pretrained=False, **kwargs): + """Constructs a ResNet-152 model. + + Args: + pretrained (bool): If True, returns a model pre-trained on ImageNet. + + Returns: + The created ResNet-152 model. + """ + model = ResNet(Bottleneck, [3, 8, 36, 3], **kwargs) + + return model + + +__all__ = [ + 'ResNet', 'resnet18', 'resnet34', 'resnet50', 'resnet101', 'resnet152' +] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/xceptionnet.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/xceptionnet.py new file mode 100644 index 0000000000..34440ab9df --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/model/xceptionnet.py @@ -0,0 +1,311 @@ +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# ============================================================================= + +# the code is modified from +# https://github.com/Cadene/pretrained-models.pytorch/blob/master/pretrainedmodels/models/xception.py + +from singa import layer +from singa import model + + +class Block(layer.Layer): + + def __init__(self, + in_filters, + out_filters, + reps, + strides=1, + padding=0, + start_with_relu=True, + grow_first=True): + super(Block, self).__init__() + + if out_filters != in_filters or strides != 1: + self.skip = layer.Conv2d(in_filters, + out_filters, + 1, + stride=strides, + padding=padding, + bias=False) + self.skipbn = layer.BatchNorm2d(out_filters) + else: + self.skip = None + + self.layers = [] + + filters = in_filters + if grow_first: + self.layers.append(layer.ReLU()) + self.layers.append( + layer.SeparableConv2d(in_filters, + out_filters, + 3, + stride=1, + padding=1, + bias=False)) + self.layers.append(layer.BatchNorm2d(out_filters)) + filters = out_filters + + for i in range(reps - 1): + self.layers.append(layer.ReLU()) + self.layers.append( + layer.SeparableConv2d(filters, + filters, + 3, + stride=1, + padding=1, + bias=False)) + self.layers.append(layer.BatchNorm2d(filters)) + + if not grow_first: + self.layers.append(layer.ReLU()) + self.layers.append( + layer.SeparableConv2d(in_filters, + out_filters, + 3, + stride=1, + padding=1, + bias=False)) + self.layers.append(layer.BatchNorm2d(out_filters)) + + if not start_with_relu: + self.layers = self.layers[1:] + else: + self.layers[0] = layer.ReLU() + + if strides != 1: + self.layers.append(layer.MaxPool2d(3, strides, padding + 1)) + + self.register_layers(*self.layers) + + self.add = layer.Add() + + def forward(self, x): + y = self.layers[0](x) + for layer in self.layers[1:]: + if isinstance(y, tuple): + y = y[0] + y = layer(y) + + if self.skip is not None: + skip = self.skip(x) + skip = self.skipbn(skip) + else: + skip = x + y = self.add(y, skip) + return y + + +class Xception(model.Model): + """ + Xception optimized for the ImageNet dataset, as specified in + https://arxiv.org/pdf/1610.02357.pdf + """ + + def __init__(self, num_classes=10, num_channels=3): + """ Constructor + Args: + num_classes: number of classes + """ + super(Xception, self).__init__() + self.num_classes = num_classes + self.input_size = 299 + self.dimension = 4 + + self.conv1 = layer.Conv2d(num_channels, 32, 3, 2, 0, bias=False) + self.bn1 = layer.BatchNorm2d(32) + self.relu1 = layer.ReLU() + + self.conv2 = layer.Conv2d(32, 64, 3, 1, 1, bias=False) + self.bn2 = layer.BatchNorm2d(64) + self.relu2 = layer.ReLU() + # do relu here + + self.block1 = Block(64, + 128, + 2, + 2, + padding=0, + start_with_relu=False, + grow_first=True) + self.block2 = Block(128, + 256, + 2, + 2, + padding=0, + start_with_relu=True, + grow_first=True) + self.block3 = Block(256, + 728, + 2, + 2, + padding=0, + start_with_relu=True, + grow_first=True) + + self.block4 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block5 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block6 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block7 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + + self.block8 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block9 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block10 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + self.block11 = Block(728, + 728, + 3, + 1, + start_with_relu=True, + grow_first=True) + + self.block12 = Block(728, + 1024, + 2, + 2, + start_with_relu=True, + grow_first=False) + + self.conv3 = layer.SeparableConv2d(1024, 1536, 3, 1, 1) + self.bn3 = layer.BatchNorm2d(1536) + self.relu3 = layer.ReLU() + + # do relu here + self.conv4 = layer.SeparableConv2d(1536, 2048, 3, 1, 1) + self.bn4 = layer.BatchNorm2d(2048) + + self.relu4 = layer.ReLU() + self.globalpooling = layer.MaxPool2d(10, 1) + self.flatten = layer.Flatten() + self.fc = layer.Linear(num_classes) + + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + + def features(self, input): + x = self.conv1(input) + x = self.bn1(x) + x = self.relu1(x) + + x = self.conv2(x) + x = self.bn2(x) + x = self.relu2(x) + + x = self.block1(x) + x = self.block2(x) + x = self.block3(x) + x = self.block4(x) + x = self.block5(x) + x = self.block6(x) + x = self.block7(x) + x = self.block8(x) + x = self.block9(x) + x = self.block10(x) + x = self.block11(x) + x = self.block12(x) + + x = self.conv3(x) + x = self.bn3(x) + x = self.relu3(x) + + x = self.conv4(x) + x = self.bn4(x) + return x + + def logits(self, features): + x = self.relu4(features) + x = self.globalpooling(x) + x = self.flatten(x) + x = self.fc(x) + return x + + def forward(self, x): + x = self.features(x) + x = self.logits(x) + return x + + def train_one_batch(self, x, y, dist_option, spars): + out = self.forward(x) + loss = self.softmax_cross_entropy(out, y) + if dist_option == 'plain': + self.optimizer(loss) + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + return out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def create_model(pretrained=False, **kwargs): + """Constructs a Xceptionnet model. + + Args: + pretrained (bool): If True, returns a pre-trained model. + + Returns: + The created Xceptionnet model. + """ + model = Xception(**kwargs) + + return model + + +__all__ = ['Xception', 'create_model'] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/pkg_model_code/model.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/pkg_model_code/model.py new file mode 100644 index 0000000000..98884584f6 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/pkg_model_code/model.py @@ -0,0 +1,357 @@ +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# ============================================================================= +''' +This script includes Model class for python users +to use Computational Graph in their model. +''' + +import os +import gc +import time +import json +import zipfile +import numpy as np +from functools import wraps +from collections import Iterable + +from singa import tensor +from singa import autograd +from singa import layer +from .tensor import Tensor +from . import singa_wrap as singa + + +class ModelMeta(layer.LayerMeta): + + def buffer_operation(func): + + def remove_creator(tensors): + if not tensors: + return + + if isinstance(tensors, Iterable): + if isinstance(tensors, str): + return + else: + for item in tensors: + if isinstance(item, Iterable): + remove_creator(item) + elif isinstance(item, tensor.Tensor): + item.creator = None + elif isinstance(tensors, tensor.Tensor): + tensors.creator = None + + @wraps(func) + def wrapper(self, *args, **kwargs): + if self.graph_mode and self.training: + if len(args) == 0: + raise ValueError('expect at least one input tensor') + + if isinstance(args[0], list): + assert isinstance( + args[0][0], + Tensor), ('function expects PlaceHolders or Tensors') + dev = args[0][0].device + else: + assert isinstance( + args[0], + Tensor), ('function expects PlaceHolders or Tensors') + dev = args[0].device + + if not self._buffered: + # buffer operations + dev.EnableGraph(True) + self._results = func(self, *args, **kwargs) + dev.Sync() + dev.EnableGraph(False) + self._buffered = True + + # deconstruct Operations before running the entire graph + remove_creator(self._results) + + # make sure all Operations are deallocated + gc.collect() + + # run graph + dev.RunGraph(self.sequential) + return self._results + else: + return func(self, *args, **kwargs) + + return wrapper + + def __new__(cls, name, bases, attr): + if 'train_one_batch' in attr: + attr['train_one_batch'] = ModelMeta.buffer_operation( + attr['train_one_batch']) + + return super(ModelMeta, cls).__new__(cls, name, bases, attr) + + +class Model(layer.Layer, metaclass=ModelMeta): + """ Base class for your neural network models. + + Example usage:: + + import numpy as np + from singa import opt + from singa import tensor + from singa import device + from singa import autograd + from singa import layer + from singa import model + + class MyModel(model.Model): + def __init__(self): + super(MyModel, self).__init__() + + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + self.conv1 = layer.Conv2d(1, 20, 5, padding=0) + self.conv2 = layer.Conv2d(20, 50, 5, padding=0) + self.sgd = opt.SGD(lr=0.01) + + def forward(self, x): + y = self.conv1(x) + y = self.conv2(y) + return y + + def train_one_batch(self, x, y): + out = self.forward(x) + loss = self.softmax_cross_entropy(out, y) + self.sgd(loss) + return out, loss + + """ + + # save load states constant + TENSOR_DICT_FILENAME = '/tensor_dict.npz' + STATES_ATTR_FILENAME = '/states_attr.json' + MODEL_STATE_TYPE = 0 + AUX_STATE_TYPE = 1 + + def __init__(self): + """ + Initializes internal Model state + """ + super(Model, self).__init__() + + self.training = True + self.graph_mode = True + self.sequential = False + self._buffered = False + self._results = None + + def compile(self, inputs, is_train=True, use_graph=False, sequential=False): + """ Compile and initialize the model + + This function will automatically derive the shape of parameters + in each sublayer based on the shape of input placeholders. It will + also do some settings. + + Args: + inputs(list): the list of input tensors(placeholders) + is_train(bool): when is_trainis True, this model will enter + training mode, otherwise it will enter the evaluation mode + use_graph(bool): when use_graph is True, computational graph + will be used to train this model + sequential(bool): when sequential is True, model will execute ops + in the graph follow the order of joining the graph + """ + assert len(inputs) > 0 and isinstance(inputs[0], Tensor), ( + 'compile function expects PlaceHolders or Tensors') + + dev = inputs[0].device + dev.EnableGraph(True) + self.forward(*inputs) + dev.EnableGraph(False) + dev.ResetGraph() + + autograd.training = is_train + self.training = is_train + self.graph_mode = use_graph + self.sequential = sequential + + def forward(self, *input): + """Defines the computation performed in every forward propagation. + + Should be overridden by all subclasses. + + Args: + *input: the input training data for the model + + Returns: + out: the outputs of the forward propagation. + """ + raise NotImplementedError + + def train_one_batch(self, *input, **kwargs): + """Defines the computation performed in every training iteration + + Should be overridden by all subclasses. + + Args: + *input: the arguments of train_one_batch + **kwargs: the keyword arguments of train_one_batch + """ + raise NotImplementedError + + def train(self, mode=True): + """Set the model in evaluation mode. + + Args: + mode(bool): when mode is True, this model will enter training mode + """ + self.training = mode + autograd.training = mode + + def eval(self): + """Sets the model in evaluation mode. + """ + self.train(mode=False) + + def graph(self, mode=True, sequential=False): + """ Turn on the computational graph. Specify execution mode. + + Args: + mode(bool): when mode is True, model will use computational graph + sequential(bool): when sequential is True, model will execute ops + in the graph follow the order of joining the graph + """ + self.graph_mode = mode + self.sequential = sequential + + def __get_name__(self): + return self.__class__.__name__ + + def __call__(self, *input, **kwargs): + if self.training: + return self.train_one_batch(*input, **kwargs) + else: + return self.forward(*input, **kwargs) + + def save_states(self, fpath, aux_states={}): + """Save states. + + Args: + fpath: output file path (without the extension) + aux_states(dict): values are standard data types or Tensor, + e.g., epoch ID, learning rate, optimizer states + """ + assert not os.path.isfile(fpath), ( + "Failed to save states, %s is already existed." % fpath) + + states = self.get_states() + + # save states data and attr + tensor_dict = {} + states_attr = {} + for k, v in states.items(): + assert isinstance(v, tensor.Tensor), "Only tensor state is allowed" + tensor_dict[k] = tensor.to_numpy(v) + states_attr[k] = { + 'state_type': self.MODEL_STATE_TYPE, + 'shape': v.shape, + 'dtype': v.dtype + } + + for k, v in aux_states.items(): + assert isinstance(v, + tensor.Tensor), "Only tensor aux state is allowed" + tensor_dict[k] = tensor.to_numpy(v) + states_attr[k] = { + 'state_type': self.AUX_STATE_TYPE, + 'shape': v.shape, + 'dtype': v.dtype + } + + # save to files + timestamp = time.time() + tmp_dir = '/tmp/singa_save_states_%s' % timestamp + os.mkdir(tmp_dir) + tensor_dict_fp = tmp_dir + self.TENSOR_DICT_FILENAME + states_attr_fp = tmp_dir + self.STATES_ATTR_FILENAME + + np.savez(tensor_dict_fp, **tensor_dict) + + with open(states_attr_fp, 'w') as fp: + json.dump(states_attr, fp) + + compression = zipfile.ZIP_DEFLATED + with zipfile.ZipFile(fpath, mode="w") as zf: + zf.write(tensor_dict_fp, + os.path.basename(tensor_dict_fp), + compress_type=compression) + zf.write(states_attr_fp, + os.path.basename(states_attr_fp), + compress_type=compression) + + # clean up tmp files + os.remove(tensor_dict_fp) + os.remove(states_attr_fp) + os.rmdir(tmp_dir) + + def load_states(self, fpath): + """Load the model states and auxiliary states from disk. + + Usage: + m = MyModel() + m.compile(...) + aux_states = m.load_states('mymodel.zip') + + Args: + path: input file path (without the extension) + Returns: + dict + """ + + assert os.path.isfile(fpath), ( + "Failed to load states, %s is not exist." % fpath) + + timestamp = time.time() + tmp_dir = '/tmp/singa_load_states_%s' % timestamp + os.mkdir(tmp_dir) + + with zipfile.ZipFile(fpath, 'r') as zf: + zf.extractall(tmp_dir) + + tensor_dict_fp = tmp_dir + self.TENSOR_DICT_FILENAME + states_attr_fp = tmp_dir + self.STATES_ATTR_FILENAME + + with open(states_attr_fp) as f: + states_attr = json.load(f) + + tensor_dict = np.load(tensor_dict_fp) + + # restore singa tensor from numpy + model_states = dict() + aux_states = dict() + + for k in tensor_dict.files: + if states_attr[k]['state_type'] == self.MODEL_STATE_TYPE: + model_states[k] = tensor.from_numpy(tensor_dict[k]) + elif states_attr[k]['state_type'] == self.AUX_STATE_TYPE: + aux_states[k] = tensor.from_numpy(tensor_dict[k]) + + # restore model_states + self.set_states(model_states) + + # clean up tmp files + os.remove(tensor_dict_fp) + os.remove(states_attr_fp) + os.rmdir(tmp_dir) + return aux_states diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/run.sh b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/run.sh new file mode 100644 index 0000000000..a536a1e81c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/run.sh @@ -0,0 +1,38 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +#!/usr/bin/env python -W ignore::DeprecationWarning + +### mnist +python train_cnn.py mlp mnist +python train_cnn.py cnn mnist +python train_cnn.py resnet mnist +python train_cnn.py alexnet mnist + +### cifar10 +python train_cnn.py mlp cifar10 +python train_cnn.py cnn cifar10 +python train_cnn.py resnet cifar10 +python train_cnn.py alexnet cifar10 + +### cifar100 +python train_cnn.py mlp cifar100 +python train_cnn.py cnn cifar100 +python train_cnn.py resnet cifar100 +python train_cnn.py alexnet cifar100 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_cnn.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_cnn.py new file mode 100644 index 0000000000..c17e1b6c42 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_cnn.py @@ -0,0 +1,564 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import singa_wrap as singa +from singa import device +from singa import tensor +from singa import opt +from singa import autograd +from singa.opt import Optimizer +from singa.opt import DecayScheduler +from singa.opt import Constant +import numpy as np +import time +import argparse +from PIL import Image + +np_dtype = {"float16": np.float16, "float32": np.float32} + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +### MSOptimizer +class MSOptimizer(Optimizer): + def __call__(self, loss): + pn_p_g_list = self.call_with_returns(loss) + self.step() + return pn_p_g_list + + def call_with_returns(self, loss): + # print ("call_with_returns loss.data: \n", loss.data) + pn_p_g_list = [] + for p, g in autograd.backward(loss): + if p.name is None: + p.name = id(p) + self.apply(p.name, p, g) + # print ("call with returns") + # print ("p.name: \n", p.name) + # print ("p.data: \n", p.data) + # print ("g.data: \n", g.data) + pn_p_g_list.append([p.name, p, g]) # need iterables + return pn_p_g_list + +# MSSGD -- actually no change of code +class MSSGD(MSOptimizer): + """Implements stochastic gradient descent (optionally with momentum). + + Nesterov momentum is based on the formula from `On the importance of initialization and momentum in deep learning`__. + + Args: + lr(float): learning rate + momentum(float, optional): momentum factor(default: 0) + weight_decay(float, optional): weight decay(L2 penalty)(default: 0) + dampening(float, optional): dampening for momentum(default: 0) + nesterov(bool, optional): enables Nesterov momentum(default: False) + + Typical usage example: + >> > from singa import opt + >> > optimizer = opt.SGD(lr=0.1, momentum=0.9) + >> > optimizer.update() + + __ http: // www.cs.toronto.edu / %7Ehinton / absps / momentum.pdf + + .. note:: + The implementation of SGD with Momentum / Nesterov subtly differs from + Sutskever et. al. and implementations in some other frameworks. + + Considering the specific case of Momentum, the update can be written as + + .. math:: + v = \rho * v + g \\ + p = p - lr * v + + where p, g, v and: math: `\rho` denote the parameters, gradient, + velocity, and momentum respectively. + + This is in contrast to Sutskever et. al. and + other frameworks which employ an update of the form + + .. math:: + v = \rho * v + lr * g \\ + p = p - v + + The Nesterov version is analogously modified. + """ + + def __init__(self, + lr=0.1, + momentum=0, + dampening=0, + weight_decay=0, + nesterov=False, + dtype=tensor.float32): + super(MSSGD, self).__init__(lr, dtype) + + # init momentum + if type(momentum) == float or type(momentum) == int: + if momentum < 0.0: + raise ValueError("Invalid momentum value: {}".format(momentum)) + self.momentum = Constant(momentum) + elif isinstance(momentum, DecayScheduler): + self.momentum = momentum + momentum = momentum.init_value + else: + raise TypeError("Wrong momentum type") + self.mom_value = self.momentum(self.step_counter).as_type(self.dtype) + + # init dampening + if type(dampening) == float or type(dampening) == int: + self.dampening = Constant(dampening) + elif isinstance(dampening, DecayScheduler): + self.dampening = dampening + dampening = dampening.init_value + else: + raise TypeError("Wrong dampening type") + self.dam_value = self.dampening(self.step_counter).as_type(self.dtype) + + # init weight_decay + if type(weight_decay) == float or type(weight_decay) == int: + if weight_decay < 0.0: + raise ValueError( + "Invalid weight_decay value: {}".format(weight_decay)) + self.weight_decay = Constant(weight_decay) + elif isinstance(weight_decay, DecayScheduler): + self.weight_decay = weight_decay + else: + raise TypeError("Wrong weight_decay type") + self.decay_value = self.weight_decay(self.step_counter).as_type( + self.dtype) + + # init other params + self.nesterov = nesterov + self.moments = dict() + + # check value + if nesterov and (momentum <= 0 or dampening != 0): + raise ValueError( + "Nesterov momentum requires a momentum and zero dampening") + + def apply(self, param_name, param_value, param_grad): + """Performs a single optimization step. + + Args: + param_name(String): the name of the param + param_value(Tensor): param values to be update in-place + grad(Tensor): param gradients; the values may be updated + in this function; cannot use it anymore + """ + assert param_value.shape == param_grad.shape, ("shape mismatch", + param_value.shape, + param_grad.shape) + self.device_check(param_value, self.step_counter, self.lr_value, + self.mom_value, self.dam_value, self.decay_value) + + # derive dtype from input + assert param_value.dtype == self.dtype + + # TODO add branch operator + # if self.decay_value != 0: + if self.weight_decay.init_value != 0: + singa.Axpy(self.decay_value.data, param_value.data, param_grad.data) + + if self.momentum.init_value != 0: + if param_name not in self.moments: + flag = param_value.device.graph_enabled() + param_value.device.EnableGraph(False) + self.moments[param_name] = tensor.zeros_like(param_value) + param_value.device.EnableGraph(flag) + + buf = self.moments[param_name] + buf *= self.mom_value + alpha = 1.0 - self.dam_value + singa.Axpy(alpha.data, param_grad.data, buf.data) + + if self.nesterov: + singa.Axpy(self.mom_value.data, buf.data, param_grad.data) + else: + param_grad = buf + + minus_lr = 0.0 - self.lr_value + singa.Axpy(minus_lr.data, param_grad.data, param_value.data) + + def step(self): + # increment step counter, lr and moment + super().step() + mom_value = self.momentum(self.step_counter).as_type(self.dtype) + dam_value = self.dampening(self.step_counter).as_type(self.dtype) + decay_value = self.weight_decay(self.step_counter).as_type(self.dtype) + self.mom_value.copy_from(mom_value) + self.dam_value.copy_from(dam_value) + self.decay_value.copy_from(decay_value) + + def get_states(self): + states = super().get_states() + if self.mom_value > 0: + states[ + 'moments'] = self.moments # a dict for 1st order moments tensors + return states + + def set_states(self, states): + super().set_states(states) + if 'moments' in states: + self.moments = states['moments'] + self.mom_value = self.momentum(self.step_counter) + + +# Data augmentation +def augmentation(x, batch_size): + xpad = np.pad(x, [[0, 0], [0, 0], [4, 4], [4, 4]], 'symmetric') + for data_num in range(0, batch_size): + offset = np.random.randint(8, size=2) + x[data_num, :, :, :] = xpad[data_num, :, + offset[0]:offset[0] + x.shape[2], + offset[1]:offset[1] + x.shape[2]] + if_flip = np.random.randint(2) + if (if_flip): + x[data_num, :, :, :] = x[data_num, :, :, ::-1] + return x + + +# Calculate accuracy +def accuracy(pred, target): + # y is network output to be compared with ground truth (int) + y = np.argmax(pred, axis=1) + a = y == target + correct = np.array(a, "int").sum() + return correct + + +# Data partition according to the rank +def partition(global_rank, world_size, train_x, train_y, val_x, val_y): + # Partition training data + data_per_rank = train_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + train_x = train_x[idx_start:idx_end] + train_y = train_y[idx_start:idx_end] + + # Partition evaluation data + data_per_rank = val_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + val_x = val_x[idx_start:idx_end] + val_y = val_y[idx_start:idx_end] + return train_x, train_y, val_x, val_y + + +# Function to all reduce NUMPY accuracy and loss from multiple devices +def reduce_variable(variable, dist_opt, reducer): + reducer.copy_from_numpy(variable) + dist_opt.all_reduce(reducer.data) + dist_opt.wait() + output = tensor.to_numpy(reducer) + return output + + +def resize_dataset(x, image_size): + num_data = x.shape[0] + dim = x.shape[1] + X = np.zeros(shape=(num_data, dim, image_size, image_size), + dtype=np.float32) + for n in range(0, num_data): + for d in range(0, dim): + X[n, d, :, :] = np.array(Image.fromarray(x[n, d, :, :]).resize( + (image_size, image_size), Image.BILINEAR), + dtype=np.float32) + return X + + +def run(global_rank, + world_size, + local_rank, + max_epoch, + batch_size, + model, + data, + mssgd, + graph, + verbosity, + dist_option='plain', + spars=None, + precision='float32'): + # dev = device.create_cuda_gpu_on(local_rank) # need to change to CPU device for CPU-only machines + dev = device.get_default_device() + dev.SetRandSeed(0) + np.random.seed(0) + + if data == 'cifar10': + from data import cifar10 + train_x, train_y, val_x, val_y = cifar10.load() + elif data == 'cifar100': + from data import cifar100 + train_x, train_y, val_x, val_y = cifar100.load() + elif data == 'mnist': + from data import mnist + train_x, train_y, val_x, val_y = mnist.load() + + + num_channels = train_x.shape[1] + image_size = train_x.shape[2] + data_size = np.prod(train_x.shape[1:train_x.ndim]).item() + num_classes = (np.max(train_y) + 1).item() + + if model == 'resnet': + from model import resnet + model = resnet.resnet50(num_channels=num_channels, + num_classes=num_classes) + elif model == 'xceptionnet': + from model import xceptionnet + model = xceptionnet.create_model(num_channels=num_channels, + num_classes=num_classes) + elif model == 'cnn': + from model import cnn + model = cnn.create_model(num_channels=num_channels, + num_classes=num_classes) + elif model == 'alexnet': + from model import alexnet + model = alexnet.create_model(num_channels=num_channels, + num_classes=num_classes) + elif model == 'mlp': + import os, sys, inspect + current = os.path.dirname( + os.path.abspath(inspect.getfile(inspect.currentframe()))) + parent = os.path.dirname(current) + sys.path.insert(0, parent) + from mlp import model + model = model.create_model(data_size=data_size, + num_classes=num_classes) + + elif model == 'msmlp': + import os, sys, inspect + current = os.path.dirname( + os.path.abspath(inspect.getfile(inspect.currentframe()))) + parent = os.path.dirname(current) + sys.path.insert(0, parent) + from msmlp import model + model = model.create_model(data_size=data_size, + num_classes=num_classes) + + # For distributed training, sequential has better performance + if hasattr(mssgd, "communicator"): + DIST = True + sequential = True + else: + DIST = False + sequential = False + + if DIST: + train_x, train_y, val_x, val_y = partition(global_rank, world_size, + train_x, train_y, val_x, + val_y) + + if model.dimension == 4: + tx = tensor.Tensor( + (batch_size, num_channels, model.input_size, model.input_size), dev, + singa_dtype[precision]) + elif model.dimension == 2: + tx = tensor.Tensor((batch_size, data_size), dev, singa_dtype[precision]) + np.reshape(train_x, (train_x.shape[0], -1)) + np.reshape(val_x, (val_x.shape[0], -1)) + + ty = tensor.Tensor((batch_size,), dev, tensor.int32) + num_train_batch = train_x.shape[0] // batch_size + num_val_batch = val_x.shape[0] // batch_size + idx = np.arange(train_x.shape[0], dtype=np.int32) + + # Attach model to graph + model.set_optimizer(mssgd) + model.compile([tx], is_train=True, use_graph=graph, sequential=sequential) + dev.SetVerbosity(verbosity) + + # Training and evaluation loop + for epoch in range(max_epoch): + start_time = time.time() + np.random.shuffle(idx) + + if global_rank == 0: + print('Starting Epoch %d:' % (epoch)) + + # Training phase + train_correct = np.zeros(shape=[1], dtype=np.float32) + test_correct = np.zeros(shape=[1], dtype=np.float32) + train_loss = np.zeros(shape=[1], dtype=np.float32) + + model.train() + print ("num_train_batch: \n", num_train_batch) + print () + for b in range(num_train_batch): + if b % 200 == 0: + print ("b: \n", b) + # Generate the patch data in this iteration + x = train_x[idx[b * batch_size:(b + 1) * batch_size]] + if model.dimension == 4: + x = augmentation(x, batch_size) + if (image_size != model.input_size): + x = resize_dataset(x, model.input_size) + x = x.astype(np_dtype[precision]) + y = train_y[idx[b * batch_size:(b + 1) * batch_size]] + + + synflow_flag = False + # Train the model + if epoch == (max_epoch - 1) and b == (num_train_batch - 1): ### synflow calcuation for the last batch + print ("last epoch calculate synflow") + synflow_flag = True + ### step 1: all one input + # Copy the patch data into input tensors + tx.copy_from_numpy(np.ones(x.shape, dtype=np.float32)) + ty.copy_from_numpy(y) + ### step 2: all weights turned to positive (done) + ### step 3: new loss (done) + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + ### step 4: calculate the multiplication of weights + synflow_score = 0.0 + for pn_p_g_item in pn_p_g_list: + print ("calculate weight param * grad parameter name: \n", pn_p_g_item[0]) + if len(pn_p_g_item[1].shape) == 2: # param_value.data is "weight" + print ("pn_p_g_item[1].shape: \n", pn_p_g_item[1].shape) + synflow_score += np.sum(np.absolute(tensor.to_numpy(pn_p_g_item[1]) * tensor.to_numpy(pn_p_g_item[2]))) + print ("synflow_score: \n", synflow_score) + elif epoch == (max_epoch - 1) and b == (num_train_batch - 2): # all weights turned to positive + # Copy the patch data into input tensors + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + train_correct += accuracy(tensor.to_numpy(out), y) + train_loss += tensor.to_numpy(loss)[0] + # all params turned to positive + for pn_p_g_item in pn_p_g_list: + print ("absolute value parameter name: \n", pn_p_g_item[0]) + pn_p_g_item[1] = tensor.abs(pn_p_g_item[1]) # tensor actually ... + else: # normal train steps + # Copy the patch data into input tensors + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + # print ("normal before model(tx, ty, synflow_flag, dist_option, spars)") + # print ("train_cnn tx: \n", tx) + # print ("train_cnn ty: \n", ty) + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + # print ("normal after model(tx, ty, synflow_flag, dist_option, spars)") + train_correct += accuracy(tensor.to_numpy(out), y) + train_loss += tensor.to_numpy(loss)[0] + + if DIST: + # Reduce the evaluation accuracy and loss from multiple devices + reducer = tensor.Tensor((1,), dev, tensor.float32) + train_correct = reduce_variable(train_correct, mssgd, reducer) + train_loss = reduce_variable(train_loss, mssgd, reducer) + + if global_rank == 0: + print('Training loss = %f, training accuracy = %f' % + (train_loss, train_correct / + (num_train_batch * batch_size * world_size)), + flush=True) + + # Evaluation phase + model.eval() + for b in range(num_val_batch): + x = val_x[b * batch_size:(b + 1) * batch_size] + if model.dimension == 4: + if (image_size != model.input_size): + x = resize_dataset(x, model.input_size) + x = x.astype(np_dtype[precision]) + y = val_y[b * batch_size:(b + 1) * batch_size] + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out_test = model(tx) + test_correct += accuracy(tensor.to_numpy(out_test), y) + + if DIST: + # Reduce the evaulation accuracy from multiple devices + test_correct = reduce_variable(test_correct, mssgd, reducer) + + # Output the evaluation accuracy + if global_rank == 0: + print('Evaluation accuracy = %f, Elapsed Time = %fs' % + (test_correct / (num_val_batch * batch_size * world_size), + time.time() - start_time), + flush=True) + + dev.PrintTimeProfiling() + + +if __name__ == '__main__': + # Use argparse to get command config: max_epoch, model, data, etc., for single gpu training + parser = argparse.ArgumentParser( + description='Training using the autograd and graph.') + parser.add_argument( + 'model', + choices=['cnn', 'resnet', 'xceptionnet', 'mlp', 'msmlp', 'alexnet'], + default='cnn') + parser.add_argument('data', + choices=['mnist', 'cifar10', 'cifar100'], + default='mnist') + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-m', + '--max-epoch', + default=3, + type=int, + help='maximum epochs', + dest='max_epoch') + parser.add_argument('-b', + '--batch-size', + default=64, + type=int, + help='batch size', + dest='batch_size') + parser.add_argument('-l', + '--learning-rate', + default=0.005, + type=float, + help='initial learning rate', + dest='lr') + # Determine which gpu to use + parser.add_argument('-i', + '--device-id', + default=0, + type=int, + help='which GPU to use', + dest='device_id') + parser.add_argument('-g', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('-v', + '--log-verbosity', + default=0, + type=int, + help='logging verbosity', + dest='verbosity') + + args = parser.parse_args() + + mssgd = MSSGD(lr=args.lr, momentum=0.9, weight_decay=1e-5, dtype=singa_dtype[args.precision]) + run(0, + 1, + args.device_id, + args.max_epoch, + args.batch_size, + args.model, + args.data, + mssgd, + args.graph, + args.verbosity, + precision=args.precision) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_mpi.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_mpi.py new file mode 100644 index 0000000000..563d4b2c51 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_mpi.py @@ -0,0 +1,91 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + + +from singa import singa_wrap as singa +from singa import opt +from singa import tensor +import argparse +import train_cnn + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +if __name__ == '__main__': + # Use argparse to get command config: max_epoch, model, data, etc., for single gpu training + parser = argparse.ArgumentParser( + description='Training using the autograd and graph.') + parser.add_argument('model', + choices=['cnn', 'resnet', 'xceptionnet', 'mlp'], + default='cnn') + parser.add_argument('data', choices=['mnist', 'cifar10', 'cifar100'], default='mnist') + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-m', + '--max-epoch', + default=10, + type=int, + help='maximum epochs', + dest='max_epoch') + parser.add_argument('-b', + '--batch-size', + default=64, + type=int, + help='batch size', + dest='batch_size') + parser.add_argument('-l', + '--learning-rate', + default=0.005, + type=float, + help='initial learning rate', + dest='lr') + parser.add_argument('-d', + '--dist-option', + default='plain', + choices=['plain','half','partialUpdate','sparseTopK','sparseThreshold'], + help='distibuted training options', + dest='dist_option') # currently partialUpdate support graph=False only + parser.add_argument('-s', + '--sparsification', + default='0.05', + type=float, + help='the sparsity parameter used for sparsification, between 0 to 1', + dest='spars') + parser.add_argument('-g', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('-v', + '--log-verbosity', + default=0, + type=int, + help='logging verbosity', + dest='verbosity') + + args = parser.parse_args() + + sgd = opt.SGD(lr=args.lr, momentum=0.9, weight_decay=1e-5, dtype=singa_dtype[args.precision]) + sgd = opt.DistOpt(sgd) + + train_cnn.run(sgd.global_rank, sgd.world_size, sgd.local_rank, args.max_epoch, + args.batch_size, args.model, args.data, sgd, args.graph, + args.verbosity, args.dist_option, args.spars, args.precision) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_ms_model.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_ms_model.py new file mode 100644 index 0000000000..8cdda8fe1d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_ms_model.py @@ -0,0 +1,592 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import singa_wrap as singa +from singa import device +from singa import tensor +from singa import opt +from singa import autograd +from singa.opt import Optimizer +from singa.opt import DecayScheduler +from singa.opt import Constant +import numpy as np +import time +import argparse +from PIL import Image + +np_dtype = {"float16": np.float16, "float32": np.float32} + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +### MSOptimizer +class MSOptimizer(Optimizer): + def __call__(self, loss): + pn_p_g_list = self.call_with_returns(loss) + self.step() + return pn_p_g_list + + def call_with_returns(self, loss): + # print ("call_with_returns loss.data: \n", loss.data) + pn_p_g_list = [] + for p, g in autograd.backward(loss): + if p.name is None: + p.name = id(p) + self.apply(p.name, p, g) + # print ("call with returns") + # print ("p.name: \n", p.name) + # print ("p.data: \n", p.data) + # print ("g.data: \n", g.data) + pn_p_g_list.append([p.name, p, g]) # need iterables + return pn_p_g_list + +# MSSGD -- actually no change of code +class MSSGD(MSOptimizer): + """Implements stochastic gradient descent (optionally with momentum). + + Nesterov momentum is based on the formula from `On the importance of initialization and momentum in deep learning`__. + + Args: + lr(float): learning rate + momentum(float, optional): momentum factor(default: 0) + weight_decay(float, optional): weight decay(L2 penalty)(default: 0) + dampening(float, optional): dampening for momentum(default: 0) + nesterov(bool, optional): enables Nesterov momentum(default: False) + + Typical usage example: + >> > from singa import opt + >> > optimizer = opt.SGD(lr=0.1, momentum=0.9) + >> > optimizer.update() + + __ http: // www.cs.toronto.edu / %7Ehinton / absps / momentum.pdf + + .. note:: + The implementation of SGD with Momentum / Nesterov subtly differs from + Sutskever et. al. and implementations in some other frameworks. + + Considering the specific case of Momentum, the update can be written as + + .. math:: + v = \rho * v + g \\ + p = p - lr * v + + where p, g, v and: math: `\rho` denote the parameters, gradient, + velocity, and momentum respectively. + + This is in contrast to Sutskever et. al. and + other frameworks which employ an update of the form + + .. math:: + v = \rho * v + lr * g \\ + p = p - v + + The Nesterov version is analogously modified. + """ + + def __init__(self, + lr=0.1, + momentum=0, + dampening=0, + weight_decay=0, + nesterov=False, + dtype=tensor.float32): + super(MSSGD, self).__init__(lr, dtype) + + # init momentum + if type(momentum) == float or type(momentum) == int: + if momentum < 0.0: + raise ValueError("Invalid momentum value: {}".format(momentum)) + self.momentum = Constant(momentum) + elif isinstance(momentum, DecayScheduler): + self.momentum = momentum + momentum = momentum.init_value + else: + raise TypeError("Wrong momentum type") + self.mom_value = self.momentum(self.step_counter).as_type(self.dtype) + + # init dampening + if type(dampening) == float or type(dampening) == int: + self.dampening = Constant(dampening) + elif isinstance(dampening, DecayScheduler): + self.dampening = dampening + dampening = dampening.init_value + else: + raise TypeError("Wrong dampening type") + self.dam_value = self.dampening(self.step_counter).as_type(self.dtype) + + # init weight_decay + if type(weight_decay) == float or type(weight_decay) == int: + if weight_decay < 0.0: + raise ValueError( + "Invalid weight_decay value: {}".format(weight_decay)) + self.weight_decay = Constant(weight_decay) + elif isinstance(weight_decay, DecayScheduler): + self.weight_decay = weight_decay + else: + raise TypeError("Wrong weight_decay type") + self.decay_value = self.weight_decay(self.step_counter).as_type( + self.dtype) + + # init other params + self.nesterov = nesterov + self.moments = dict() + + # check value + if nesterov and (momentum <= 0 or dampening != 0): + raise ValueError( + "Nesterov momentum requires a momentum and zero dampening") + + def apply(self, param_name, param_value, param_grad): + """Performs a single optimization step. + + Args: + param_name(String): the name of the param + param_value(Tensor): param values to be update in-place + grad(Tensor): param gradients; the values may be updated + in this function; cannot use it anymore + """ + assert param_value.shape == param_grad.shape, ("shape mismatch", + param_value.shape, + param_grad.shape) + self.device_check(param_value, self.step_counter, self.lr_value, + self.mom_value, self.dam_value, self.decay_value) + + # derive dtype from input + assert param_value.dtype == self.dtype + + # TODO add branch operator + # if self.decay_value != 0: + if self.weight_decay.init_value != 0: + singa.Axpy(self.decay_value.data, param_value.data, param_grad.data) + + if self.momentum.init_value != 0: + if param_name not in self.moments: + flag = param_value.device.graph_enabled() + param_value.device.EnableGraph(False) + self.moments[param_name] = tensor.zeros_like(param_value) + param_value.device.EnableGraph(flag) + + buf = self.moments[param_name] + buf *= self.mom_value + alpha = 1.0 - self.dam_value + singa.Axpy(alpha.data, param_grad.data, buf.data) + + if self.nesterov: + singa.Axpy(self.mom_value.data, buf.data, param_grad.data) + else: + param_grad = buf + + minus_lr = 0.0 - self.lr_value + singa.Axpy(minus_lr.data, param_grad.data, param_value.data) + + def step(self): + # increment step counter, lr and moment + super().step() + mom_value = self.momentum(self.step_counter).as_type(self.dtype) + dam_value = self.dampening(self.step_counter).as_type(self.dtype) + decay_value = self.weight_decay(self.step_counter).as_type(self.dtype) + self.mom_value.copy_from(mom_value) + self.dam_value.copy_from(dam_value) + self.decay_value.copy_from(decay_value) + + def get_states(self): + states = super().get_states() + if self.mom_value > 0: + states[ + 'moments'] = self.moments # a dict for 1st order moments tensors + return states + + def set_states(self, states): + super().set_states(states) + if 'moments' in states: + self.moments = states['moments'] + self.mom_value = self.momentum(self.step_counter) + + +# Data augmentation +def augmentation(x, batch_size): + xpad = np.pad(x, [[0, 0], [0, 0], [4, 4], [4, 4]], 'symmetric') + for data_num in range(0, batch_size): + offset = np.random.randint(8, size=2) + x[data_num, :, :, :] = xpad[data_num, :, + offset[0]:offset[0] + x.shape[2], + offset[1]:offset[1] + x.shape[2]] + if_flip = np.random.randint(2) + if (if_flip): + x[data_num, :, :, :] = x[data_num, :, :, ::-1] + return x + + +# Calculate accuracy +def accuracy(pred, target): + # y is network output to be compared with ground truth (int) + y = np.argmax(pred, axis=1) + a = y == target + correct = np.array(a, "int").sum() + return correct + + +# Data partition according to the rank +def partition(global_rank, world_size, train_x, train_y, val_x, val_y): + # Partition training data + data_per_rank = train_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + train_x = train_x[idx_start:idx_end] + train_y = train_y[idx_start:idx_end] + + # Partition evaluation data + data_per_rank = val_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + val_x = val_x[idx_start:idx_end] + val_y = val_y[idx_start:idx_end] + return train_x, train_y, val_x, val_y + + +# Function to all reduce NUMPY accuracy and loss from multiple devices +def reduce_variable(variable, dist_opt, reducer): + reducer.copy_from_numpy(variable) + dist_opt.all_reduce(reducer.data) + dist_opt.wait() + output = tensor.to_numpy(reducer) + return output + + +def resize_dataset(x, image_size): + num_data = x.shape[0] + dim = x.shape[1] + X = np.zeros(shape=(num_data, dim, image_size, image_size), + dtype=np.float32) + for n in range(0, num_data): + for d in range(0, dim): + X[n, d, :, :] = np.array(Image.fromarray(x[n, d, :, :]).resize( + (image_size, image_size), Image.BILINEAR), + dtype=np.float32) + return X + + +def run(global_rank, + world_size, + local_rank, + layer_hidden_list, + max_epoch, + batch_size, + model, + data, + mssgd, + graph, + verbosity, + dist_option='plain', + spars=None, + precision='float32'): + # dev = device.create_cuda_gpu_on(local_rank) # need to change to CPU device for CPU-only machines + dev = device.get_default_device() + dev.SetRandSeed(0) + np.random.seed(0) + + if data == 'cifar10': + from data import cifar10 + train_x, train_y, val_x, val_y = cifar10.load() + elif data == 'cifar100': + from data import cifar100 + train_x, train_y, val_x, val_y = cifar100.load() + elif data == 'mnist': + from data import mnist + train_x, train_y, val_x, val_y = mnist.load() + + + num_channels = train_x.shape[1] + image_size = train_x.shape[2] + data_size = np.prod(train_x.shape[1:train_x.ndim]).item() + num_classes = (np.max(train_y) + 1).item() + + if model == 'resnet': + from model import resnet + model = resnet.resnet50(num_channels=num_channels, + num_classes=num_classes) + elif model == 'xceptionnet': + from model import xceptionnet + model = xceptionnet.create_model(num_channels=num_channels, + num_classes=num_classes) + elif model == 'cnn': + from model import cnn + model = cnn.create_model(num_channels=num_channels, + num_classes=num_classes) + elif model == 'alexnet': + from model import alexnet + model = alexnet.create_model(num_channels=num_channels, + num_classes=num_classes) + elif model == 'mlp': + import os, sys, inspect + current = os.path.dirname( + os.path.abspath(inspect.getfile(inspect.currentframe()))) + parent = os.path.dirname(current) + sys.path.insert(0, parent) + from mlp import model + model = model.create_model(data_size=data_size, + num_classes=num_classes) + + elif model == 'msmlp': + import os, sys, inspect + current = os.path.dirname( + os.path.abspath(inspect.getfile(inspect.currentframe()))) + parent = os.path.dirname(current) + sys.path.insert(0, parent) + from msmlp import model + model = model.create_model(data_size=data_size, + num_classes=num_classes) + + elif model == 'ms_model_mlp': + import os, sys, inspect + current = os.path.dirname( + os.path.abspath(inspect.getfile(inspect.currentframe()))) + parent = os.path.dirname(current) + sys.path.insert(0, parent) + from ms_model_mlp import model + model = model.create_model(data_size=data_size, + num_classes=num_classes, + layer_hidden_list=layer_hidden_list) + # print ("model: \n", model) + + # For distributed training, sequential has better performance + if hasattr(mssgd, "communicator"): + DIST = True + sequential = True + else: + DIST = False + sequential = False + + if DIST: + train_x, train_y, val_x, val_y = partition(global_rank, world_size, + train_x, train_y, val_x, + val_y) + + if model.dimension == 4: + tx = tensor.Tensor( + (batch_size, num_channels, model.input_size, model.input_size), dev, + singa_dtype[precision]) + elif model.dimension == 2: + tx = tensor.Tensor((batch_size, data_size), dev, singa_dtype[precision]) + np.reshape(train_x, (train_x.shape[0], -1)) + np.reshape(val_x, (val_x.shape[0], -1)) + + ty = tensor.Tensor((batch_size,), dev, tensor.int32) + num_train_batch = train_x.shape[0] // batch_size + num_val_batch = val_x.shape[0] // batch_size + idx = np.arange(train_x.shape[0], dtype=np.int32) + + # Attach model to graph + model.set_optimizer(mssgd) + model.compile([tx], is_train=True, use_graph=graph, sequential=sequential) + dev.SetVerbosity(verbosity) + + # Training and evaluation loop + for epoch in range(max_epoch): + start_time = time.time() + np.random.shuffle(idx) + + if global_rank == 0: + print('Starting Epoch %d:' % (epoch)) + + # Training phase + train_correct = np.zeros(shape=[1], dtype=np.float32) + test_correct = np.zeros(shape=[1], dtype=np.float32) + train_loss = np.zeros(shape=[1], dtype=np.float32) + + model.train() + print ("num_train_batch: \n", num_train_batch) + print () + for b in range(num_train_batch): + # if b % 200 == 0: + # print ("b: \n", b) + # Generate the patch data in this iteration + x = train_x[idx[b * batch_size:(b + 1) * batch_size]] + if model.dimension == 4: + x = augmentation(x, batch_size) + if (image_size != model.input_size): + x = resize_dataset(x, model.input_size) + x = x.astype(np_dtype[precision]) + y = train_y[idx[b * batch_size:(b + 1) * batch_size]] + + + synflow_flag = False + # Train the model + if epoch == (max_epoch - 1) and b == (num_train_batch - 1): ### synflow calcuation for the last batch + print ("last epoch calculate synflow") + synflow_flag = True + ### step 1: all one input + # Copy the patch data into input tensors + tx.copy_from_numpy(np.ones(x.shape, dtype=np.float32)) + ty.copy_from_numpy(y) + ### step 2: all weights turned to positive (done) + ### step 3: new loss (done) + # print ("before model forward ...") + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + ### step 4: calculate the multiplication of weights + synflow_score = 0.0 + for pn_p_g_item in pn_p_g_list: + print ("calculate weight param * grad parameter name: \n", pn_p_g_item[0]) + if len(pn_p_g_item[1].shape) == 2: # param_value.data is "weight" + print ("pn_p_g_item[1].shape: \n", pn_p_g_item[1].shape) + synflow_score += np.sum(np.absolute(tensor.to_numpy(pn_p_g_item[1]) * tensor.to_numpy(pn_p_g_item[2]))) + print ("layer_hidden_list: \n", layer_hidden_list) + print ("synflow_score: \n", synflow_score) + elif epoch == (max_epoch - 1) and b == (num_train_batch - 2): # all weights turned to positive + # Copy the patch data into input tensors + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + # print ("before model forward ...") + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + train_correct += accuracy(tensor.to_numpy(out), y) + train_loss += tensor.to_numpy(loss)[0] + # all params turned to positive + for pn_p_g_item in pn_p_g_list: + print ("absolute value parameter name: \n", pn_p_g_item[0]) + pn_p_g_item[1] = tensor.abs(pn_p_g_item[1]) # tensor actually ... + else: # normal train steps + # Copy the patch data into input tensors + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + # print ("normal before model(tx, ty, synflow_flag, dist_option, spars)") + # print ("train_cnn tx: \n", tx) + # print ("train_cnn ty: \n", ty) + # print ("before model forward ...") + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + # print ("normal after model(tx, ty, synflow_flag, dist_option, spars)") + train_correct += accuracy(tensor.to_numpy(out), y) + train_loss += tensor.to_numpy(loss)[0] + + if DIST: + # Reduce the evaluation accuracy and loss from multiple devices + reducer = tensor.Tensor((1,), dev, tensor.float32) + train_correct = reduce_variable(train_correct, mssgd, reducer) + train_loss = reduce_variable(train_loss, mssgd, reducer) + + if global_rank == 0: + print('Training loss = %f, training accuracy = %f' % + (train_loss, train_correct / + (num_train_batch * batch_size * world_size)), + flush=True) + + # Evaluation phase + model.eval() + for b in range(num_val_batch): + x = val_x[b * batch_size:(b + 1) * batch_size] + if model.dimension == 4: + if (image_size != model.input_size): + x = resize_dataset(x, model.input_size) + x = x.astype(np_dtype[precision]) + y = val_y[b * batch_size:(b + 1) * batch_size] + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + out_test = model(tx) + test_correct += accuracy(tensor.to_numpy(out_test), y) + + if DIST: + # Reduce the evaulation accuracy from multiple devices + test_correct = reduce_variable(test_correct, mssgd, reducer) + + # Output the evaluation accuracy + if global_rank == 0: + print('Evaluation accuracy = %f, Elapsed Time = %fs' % + (test_correct / (num_val_batch * batch_size * world_size), + time.time() - start_time), + flush=True) + + dev.PrintTimeProfiling() + + +if __name__ == '__main__': + # Use argparse to get command config: max_epoch, model, data, etc., for single gpu training + parser = argparse.ArgumentParser( + description='Training using the autograd and graph.') + parser.add_argument( + 'model', + choices=['cnn', 'resnet', 'xceptionnet', 'mlp', 'msmlp', 'alexnet', 'ms_model_mlp'], + default='cnn') + parser.add_argument('data', + choices=['mnist', 'cifar10', 'cifar100'], + default='mnist') + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-m', + '--max-epoch', + default=2, + type=int, + help='maximum epochs', + dest='max_epoch') + parser.add_argument('-b', + '--batch-size', + default=64, + type=int, + help='batch size', + dest='batch_size') + parser.add_argument('-l', + '--learning-rate', + default=0.005, + type=float, + help='initial learning rate', + dest='lr') + # Determine which gpu to use + parser.add_argument('-i', + '--device-id', + default=0, + type=int, + help='which GPU to use', + dest='device_id') + parser.add_argument('-g', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('-v', + '--log-verbosity', + default=0, + type=int, + help='logging verbosity', + dest='verbosity') + + args = parser.parse_args() + + # mssgd = MSSGD(lr=args.lr, momentum=0.9, weight_decay=1e-5, dtype=singa_dtype[args.precision]) + + DEFAULT_LAYER_CHOICES_4 = [8, 16, 24, 32] + for layer1 in DEFAULT_LAYER_CHOICES_4: + for layer2 in DEFAULT_LAYER_CHOICES_4: + for layer3 in DEFAULT_LAYER_CHOICES_4: + for layer4 in DEFAULT_LAYER_CHOICES_4: + layer_hidden_list = [layer1, layer2+1, layer3+2, layer4+3] + # print ("layer_hidden_list: \n", layer_hidden_list) + mssgd = MSSGD(lr=args.lr, momentum=0.9, weight_decay=1e-5, dtype=singa_dtype[args.precision]) + run(0, + 1, + args.device_id, + layer_hidden_list, + args.max_epoch, + args.batch_size, + args.model, + args.data, + mssgd, + args.graph, + args.verbosity, + precision=args.precision) + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_multiprocess.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_multiprocess.py new file mode 100644 index 0000000000..182dd35eed --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/cnn_ms/train_multiprocess.py @@ -0,0 +1,111 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + + +from singa import singa_wrap as singa +from singa import opt +from singa import tensor +import argparse +import train_cnn +import multiprocessing + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +def run(args, local_rank, world_size, nccl_id): + sgd = opt.SGD(lr=args.lr, momentum=0.9, weight_decay=1e-5, dtype=singa_dtype[args.precision]) + sgd = opt.DistOpt(sgd, nccl_id=nccl_id, local_rank=local_rank, world_size=world_size) + train_cnn.run(sgd.global_rank, sgd.world_size, sgd.local_rank, args.max_epoch, + args.batch_size, args.model, args.data, sgd, args.graph, + args.verbosity, args.dist_option, args.spars, args.precision) + + +if __name__ == '__main__': + # Use argparse to get command config: max_epoch, model, data, etc., for single gpu training + parser = argparse.ArgumentParser( + description='Training using the autograd and graph.') + parser.add_argument('model', + choices=['resnet', 'xceptionnet', 'cnn', 'mlp'], + default='cnn') + parser.add_argument('data', choices=['cifar10', 'cifar100', 'mnist'], default='mnist') + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-m', + '--max-epoch', + default=10, + type=int, + help='maximum epochs', + dest='max_epoch') + parser.add_argument('-b', + '--batch-size', + default=64, + type=int, + help='batch size', + dest='batch_size') + parser.add_argument('-l', + '--learning-rate', + default=0.005, + type=float, + help='initial learning rate', + dest='lr') + parser.add_argument('-w', + '--world-size', + default=2, + type=int, + help='number of gpus to be used', + dest='world_size') + parser.add_argument('-d', + '--dist-option', + default='plain', + choices=['plain','half','partialUpdate','sparseTopK','sparseThreshold'], + help='distibuted training options', + dest='dist_option') # currently partialUpdate support graph=False only + parser.add_argument('-s', + '--sparsification', + default='0.05', + type=float, + help='the sparsity parameter used for sparsification, between 0 to 1', + dest='spars') + parser.add_argument('-g', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('-v', + '--log-verbosity', + default=0, + type=int, + help='logging verbosity', + dest='verbosity') + + args = parser.parse_args() + + # Generate a NCCL ID to be used for collective communication + nccl_id = singa.NcclIdHolder() + + process = [] + for local_rank in range(0, args.world_size): + process.append( + multiprocessing.Process(target=run, + args=(args, local_rank, args.world_size, nccl_id))) + + for p in process: + p.start() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/model.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/model.py new file mode 100644 index 0000000000..454b382d58 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/model.py @@ -0,0 +1,226 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import layer +from singa import model +from singa import tensor +from singa import opt +from singa import device +from singa.autograd import Operator +from singa.layer import Layer +from singa import singa_wrap as singa +import argparse +import numpy as np + +np_dtype = {"float16": np.float16, "float32": np.float32} + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +#### self-defined loss begin + +### from autograd.py +class SumError(Operator): + + def __init__(self): + super(SumError, self).__init__() + # self.t = t.data + + def forward(self, x): + # self.err = singa.__sub__(x, self.t) + self.data_x = x + # sqr = singa.Square(self.err) + # loss = singa.SumAll(sqr) + loss = singa.SumAll(x) + # self.n = 1 + # for s in x.shape(): + # self.n *= s + # loss /= self.n + return loss + + def backward(self, dy=1.0): + # dx = self.err + dev = device.get_default_device() + dx = tensor.Tensor(self.data_x.shape, dev, singa_dtype['float32']) + dx.copy_from_numpy(np.ones(self.data_x.shape)) + # dx *= float(2 / self.n) + dx *= dy + return dx + +def se_loss(x): + # assert x.shape == t.shape, "input and target shape different: %s, %s" % ( + # x.shape, t.shape) + return SumError()(x)[0] + +### from layer.py +class SumErrorLayer(Layer): + """ + Generate a MeanSquareError operator + """ + + def __init__(self): + super(SumErrorLayer, self).__init__() + + def forward(self, x): + return se_loss(x) + +#### self-defined loss end + +class MSMLP(model.Model): + + def __init__(self, data_size=10, perceptron_size=100, num_classes=10, layer_hidden_list=[10,10,10,10]): + super(MSMLP, self).__init__() + self.num_classes = num_classes + self.dimension = 2 + + self.relu = layer.ReLU() + self.linear1 = layer.Linear(layer_hidden_list[0]) + self.linear2 = layer.Linear(layer_hidden_list[1]) + self.linear3 = layer.Linear(layer_hidden_list[2]) + self.linear4 = layer.Linear(layer_hidden_list[3]) + self.linear5 = layer.Linear(num_classes) + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + self.sum_error = SumErrorLayer() + + def forward(self, inputs): + y = self.linear1(inputs) + y = self.relu(y) + y = self.linear2(y) + y = self.relu(y) + y = self.linear3(y) + y = self.relu(y) + y = self.linear4(y) + y = self.relu(y) + y = self.linear5(y) + return y + + def train_one_batch(self, x, y, dist_option, spars, synflow_flag): + # print ("in train_one_batch") + out = self.forward(x) + # print ("train_one_batch x.data: \n", x.data) + # print ("train_one_batch y.data: \n", y.data) + # print ("train_one_batch out.data: \n", out.data) + if synflow_flag: + # print ("sum_error") + loss = self.sum_error(out) + else: # normal training + # print ("softmax_cross_entropy") + loss = self.softmax_cross_entropy(out, y) + # print ("train_one_batch loss.data: \n", loss.data) + + if dist_option == 'plain': + # print ("before pn_p_g_list = self.optimizer(loss)") + pn_p_g_list = self.optimizer(loss) + # print ("after pn_p_g_list = self.optimizer(loss)") + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + # print ("len(pn_p_g_list): \n", len(pn_p_g_list)) + # print ("len(pn_p_g_list[0]): \n", len(pn_p_g_list[0])) + # print ("pn_p_g_list[0][0]: \n", pn_p_g_list[0][0]) + # print ("pn_p_g_list[0][1].data: \n", pn_p_g_list[0][1].data) + # print ("pn_p_g_list[0][2].data: \n", pn_p_g_list[0][2].data) + return pn_p_g_list, out, loss + # return pn_p_g_list[0], pn_p_g_list[1], pn_p_g_list[2], out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def create_model(pretrained=False, **kwargs): + """Constructs a CNN model. + + Args: + pretrained (bool): If True, returns a pre-trained model. + + Returns: + The created CNN model. + """ + model = MSMLP(**kwargs) + + return model + + +__all__ = ['MLP', 'create_model'] + +if __name__ == "__main__": + np.random.seed(0) + + parser = argparse.ArgumentParser() + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-g', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('-m', + '--max-epoch', + default=1001, + type=int, + help='maximum epochs', + dest='max_epoch') + args = parser.parse_args() + + # generate the boundary + f = lambda x: (5 * x + 1) + bd_x = np.linspace(-1.0, 1, 200) + bd_y = f(bd_x) + + # generate the training data + x = np.random.uniform(-1, 1, 400) + y = f(x) + 2 * np.random.randn(len(x)) + + # choose one precision + precision = singa_dtype[args.precision] + np_precision = np_dtype[args.precision] + + # convert training data to 2d space + label = np.asarray([5 * a + 1 > b for (a, b) in zip(x, y)]).astype(np.int32) + data = np.array([[a, b] for (a, b) in zip(x, y)], dtype=np_precision) + + dev = device.create_cuda_gpu_on(0) + sgd = opt.SGD(0.1, 0.9, 1e-5, dtype=singa_dtype[args.precision]) + tx = tensor.Tensor((400, 2), dev, precision) + ty = tensor.Tensor((400,), dev, tensor.int32) + model = MLP(data_size=2, perceptron_size=3, num_classes=2) + + # attach model to graph + model.set_optimizer(sgd) + model.compile([tx], is_train=True, use_graph=args.graph, sequential=True) + model.train() + + for i in range(args.max_epoch): + tx.copy_from_numpy(data) + ty.copy_from_numpy(label) + out, loss = model(tx, ty, 'fp32', spars=None) + + if i % 100 == 0: + print("training loss = ", tensor.to_numpy(loss)[0]) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/native.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/native.py new file mode 100644 index 0000000000..a82ec3b24c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/ms_model_mlp/native.py @@ -0,0 +1,137 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import tensor +from singa.tensor import Tensor +from singa import autograd +from singa import opt +import numpy as np +from singa import device +import argparse + +np_dtype = {"float16": np.float16, "float32": np.float32} + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +if __name__ == "__main__": + parser = argparse.ArgumentParser() + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-m', + '--max-epoch', + default=1001, + type=int, + help='maximum epochs', + dest='max_epoch') + args = parser.parse_args() + + np.random.seed(0) + + autograd.training = True + + # prepare training data in numpy array + + # generate the boundary + f = lambda x: (5 * x + 1) + bd_x = np.linspace(-1.0, 1, 200) + bd_y = f(bd_x) + + # generate the training data + x = np.random.uniform(-1, 1, 400) + y = f(x) + 2 * np.random.randn(len(x)) + + # convert training data to 2d space + label = np.asarray([5 * a + 1 > b for (a, b) in zip(x, y)]) + data = np.array([[a, b] for (a, b) in zip(x, y)], dtype=np.float32) + + def to_categorical(y, num_classes): + """ + Converts a class vector (integers) to binary class matrix. + + Args: + y: class vector to be converted into a matrix + (integers from 0 to num_classes). + num_classes: total number of classes. + + Returns: + A binary matrix representation of the input. + """ + y = np.array(y, dtype="int") + n = y.shape[0] + categorical = np.zeros((n, num_classes)) + categorical[np.arange(n), y] = 1 + return categorical + + label = to_categorical(label, 2).astype(np.float32) + print("train_data_shape:", data.shape) + print("train_label_shape:", label.shape) + + precision = singa_dtype[args.precision] + np_precision = np_dtype[args.precision] + + dev = device.create_cuda_gpu() + + inputs = Tensor(data=data, device=dev) + target = Tensor(data=label, device=dev) + + inputs = inputs.as_type(precision) + target = target.as_type(tensor.int32) + + w0_np = np.random.normal(0, 0.1, (2, 3)).astype(np_precision) + w0 = Tensor(data=w0_np, + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b0 = Tensor(shape=(3,), + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b0.set_value(0.0) + + w1_np = np.random.normal(0, 0.1, (3, 2)).astype(np_precision) + w1 = Tensor(data=w1_np, + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b1 = Tensor(shape=(2,), + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b1.set_value(0.0) + + sgd = opt.SGD(0.05, 0.8) + + # training process + for i in range(args.max_epoch): + x = autograd.matmul(inputs, w0) + x = autograd.add_bias(x, b0) + x = autograd.relu(x) + x = autograd.matmul(x, w1) + x = autograd.add_bias(x, b1) + loss = autograd.softmax_cross_entropy(x, target) + sgd(loss) + + if i % 100 == 0: + print("%d, training loss = " % i, tensor.to_numpy(loss)[0]) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/model.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/model.py new file mode 100644 index 0000000000..c0f0b7b4ee --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/model.py @@ -0,0 +1,217 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import layer +from singa import model +from singa import tensor +from singa import opt +from singa import device +from singa.autograd import Operator +from singa.layer import Layer +from singa import singa_wrap as singa +import argparse +import numpy as np + +np_dtype = {"float16": np.float16, "float32": np.float32} + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +#### self-defined loss begin + +### from autograd.py +class SumError(Operator): + + def __init__(self): + super(SumError, self).__init__() + # self.t = t.data + + def forward(self, x): + # self.err = singa.__sub__(x, self.t) + self.data_x = x + # sqr = singa.Square(self.err) + # loss = singa.SumAll(sqr) + loss = singa.SumAll(x) + # self.n = 1 + # for s in x.shape(): + # self.n *= s + # loss /= self.n + return loss + + def backward(self, dy=1.0): + # dx = self.err + dev = device.get_default_device() + dx = tensor.Tensor(self.data_x.shape, dev, singa_dtype['float32']) + dx.copy_from_numpy(np.ones(self.data_x.shape)) + # dx *= float(2 / self.n) + dx *= dy + return dx + +def se_loss(x): + # assert x.shape == t.shape, "input and target shape different: %s, %s" % ( + # x.shape, t.shape) + return SumError()(x)[0] + +### from layer.py +class SumErrorLayer(Layer): + """ + Generate a MeanSquareError operator + """ + + def __init__(self): + super(SumErrorLayer, self).__init__() + + def forward(self, x): + return se_loss(x) + +#### self-defined loss end + +class MSMLP(model.Model): + + def __init__(self, data_size=10, perceptron_size=100, num_classes=10): + super(MSMLP, self).__init__() + self.num_classes = num_classes + self.dimension = 2 + + self.relu = layer.ReLU() + self.linear1 = layer.Linear(perceptron_size) + self.linear2 = layer.Linear(num_classes) + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + self.sum_error = SumErrorLayer() + + def forward(self, inputs): + y = self.linear1(inputs) + y = self.relu(y) + y = self.linear2(y) + return y + + def train_one_batch(self, x, y, dist_option, spars, synflow_flag): + # print ("in train_one_batch") + out = self.forward(x) + # print ("train_one_batch x.data: \n", x.data) + # print ("train_one_batch y.data: \n", y.data) + # print ("train_one_batch out.data: \n", out.data) + if synflow_flag: + # print ("sum_error") + loss = self.sum_error(out) + else: # normal training + # print ("softmax_cross_entropy") + loss = self.softmax_cross_entropy(out, y) + # print ("train_one_batch loss.data: \n", loss.data) + + if dist_option == 'plain': + # print ("before pn_p_g_list = self.optimizer(loss)") + pn_p_g_list = self.optimizer(loss) + # print ("after pn_p_g_list = self.optimizer(loss)") + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + # print ("len(pn_p_g_list): \n", len(pn_p_g_list)) + # print ("len(pn_p_g_list[0]): \n", len(pn_p_g_list[0])) + # print ("pn_p_g_list[0][0]: \n", pn_p_g_list[0][0]) + # print ("pn_p_g_list[0][1].data: \n", pn_p_g_list[0][1].data) + # print ("pn_p_g_list[0][2].data: \n", pn_p_g_list[0][2].data) + return pn_p_g_list, out, loss + # return pn_p_g_list[0], pn_p_g_list[1], pn_p_g_list[2], out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def create_model(pretrained=False, **kwargs): + """Constructs a CNN model. + + Args: + pretrained (bool): If True, returns a pre-trained model. + + Returns: + The created CNN model. + """ + model = MSMLP(**kwargs) + + return model + + +__all__ = ['MLP', 'create_model'] + +if __name__ == "__main__": + np.random.seed(0) + + parser = argparse.ArgumentParser() + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-g', + '--disable-graph', + default='True', + action='store_false', + help='disable graph', + dest='graph') + parser.add_argument('-m', + '--max-epoch', + default=1001, + type=int, + help='maximum epochs', + dest='max_epoch') + args = parser.parse_args() + + # generate the boundary + f = lambda x: (5 * x + 1) + bd_x = np.linspace(-1.0, 1, 200) + bd_y = f(bd_x) + + # generate the training data + x = np.random.uniform(-1, 1, 400) + y = f(x) + 2 * np.random.randn(len(x)) + + # choose one precision + precision = singa_dtype[args.precision] + np_precision = np_dtype[args.precision] + + # convert training data to 2d space + label = np.asarray([5 * a + 1 > b for (a, b) in zip(x, y)]).astype(np.int32) + data = np.array([[a, b] for (a, b) in zip(x, y)], dtype=np_precision) + + dev = device.create_cuda_gpu_on(0) + sgd = opt.SGD(0.1, 0.9, 1e-5, dtype=singa_dtype[args.precision]) + tx = tensor.Tensor((400, 2), dev, precision) + ty = tensor.Tensor((400,), dev, tensor.int32) + model = MLP(data_size=2, perceptron_size=3, num_classes=2) + + # attach model to graph + model.set_optimizer(sgd) + model.compile([tx], is_train=True, use_graph=args.graph, sequential=True) + model.train() + + for i in range(args.max_epoch): + tx.copy_from_numpy(data) + ty.copy_from_numpy(label) + out, loss = model(tx, ty, 'fp32', spars=None) + + if i % 100 == 0: + print("training loss = ", tensor.to_numpy(loss)[0]) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/native.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/native.py new file mode 100644 index 0000000000..a82ec3b24c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/algo/singa_ms/msmlp/native.py @@ -0,0 +1,137 @@ +# +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# + +from singa import tensor +from singa.tensor import Tensor +from singa import autograd +from singa import opt +import numpy as np +from singa import device +import argparse + +np_dtype = {"float16": np.float16, "float32": np.float32} + +singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} + +if __name__ == "__main__": + parser = argparse.ArgumentParser() + parser.add_argument('-p', + choices=['float32', 'float16'], + default='float32', + dest='precision') + parser.add_argument('-m', + '--max-epoch', + default=1001, + type=int, + help='maximum epochs', + dest='max_epoch') + args = parser.parse_args() + + np.random.seed(0) + + autograd.training = True + + # prepare training data in numpy array + + # generate the boundary + f = lambda x: (5 * x + 1) + bd_x = np.linspace(-1.0, 1, 200) + bd_y = f(bd_x) + + # generate the training data + x = np.random.uniform(-1, 1, 400) + y = f(x) + 2 * np.random.randn(len(x)) + + # convert training data to 2d space + label = np.asarray([5 * a + 1 > b for (a, b) in zip(x, y)]) + data = np.array([[a, b] for (a, b) in zip(x, y)], dtype=np.float32) + + def to_categorical(y, num_classes): + """ + Converts a class vector (integers) to binary class matrix. + + Args: + y: class vector to be converted into a matrix + (integers from 0 to num_classes). + num_classes: total number of classes. + + Returns: + A binary matrix representation of the input. + """ + y = np.array(y, dtype="int") + n = y.shape[0] + categorical = np.zeros((n, num_classes)) + categorical[np.arange(n), y] = 1 + return categorical + + label = to_categorical(label, 2).astype(np.float32) + print("train_data_shape:", data.shape) + print("train_label_shape:", label.shape) + + precision = singa_dtype[args.precision] + np_precision = np_dtype[args.precision] + + dev = device.create_cuda_gpu() + + inputs = Tensor(data=data, device=dev) + target = Tensor(data=label, device=dev) + + inputs = inputs.as_type(precision) + target = target.as_type(tensor.int32) + + w0_np = np.random.normal(0, 0.1, (2, 3)).astype(np_precision) + w0 = Tensor(data=w0_np, + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b0 = Tensor(shape=(3,), + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b0.set_value(0.0) + + w1_np = np.random.normal(0, 0.1, (3, 2)).astype(np_precision) + w1 = Tensor(data=w1_np, + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b1 = Tensor(shape=(2,), + device=dev, + dtype=precision, + requires_grad=True, + stores_grad=True) + b1.set_value(0.0) + + sgd = opt.SGD(0.05, 0.8) + + # training process + for i in range(args.max_epoch): + x = autograd.matmul(inputs, w0) + x = autograd.add_bias(x, b0) + x = autograd.relu(x) + x = autograd.matmul(x, w1) + x = autograd.add_bias(x, b1) + loss = autograd.softmax_cross_entropy(x, target) + sgd(loss) + + if i % 100 == 0: + print("%d, training loss = " % i, tensor.to_numpy(loss)[0]) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/vote.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/vote.py new file mode 100644 index 0000000000..35dfb9d6fc --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase1/vote.py @@ -0,0 +1,115 @@ +from src.eva_engine.phase1.algo.alg_base import Evaluator +from .utils.autograd_hacks import * +from src.common.constant import Config + +class VoteEvaluator(Evaluator): + + def __init__(self): + super().__init__() + + def evaluate(self, arch: nn.Module, device, batch_data: object, batch_labels: torch.Tensor, space_name: str) -> float: + """ + This is simply sum over all weigth's norm to calculate models performance + :param arch: + :param device: CPU or GPU + :param batch_data: + :param batch_labels: + :return: + """ + + pass + + +def vote_between_two_arch(arch1_info: dict, arch2_info: dict, metric: list, space: str): + """ + Return which architecture is better, + :param arch1_info: + :param arch2_info: + :param metric: + :param space: + :return: + """ + left_vote = 0 + right_vote = 0 + for m_name in metric: + # if this metrics vote to left + if vote_to_left[space](m_name, + float(arch1_info["scores"][m_name]["score"]), + float(arch2_info["scores"][m_name]["score"])): + left_vote += 1 + else: + right_vote += 1 + + if left_vote > right_vote: + return arch1_info["architecture_id"] + else: + return arch2_info["architecture_id"] + + +def compare_score_201(m_name: str, s1: float, s2: float) -> bool: + """ + Return if s1 is better than s2, + :param m_name: + :param s1: + :param s2: + :return: if s1 is better than s2 + """ + if m_name == "grad_norm": + return s1 > s2 + if m_name == "grad_plain": + return s1 < s2 + if m_name == "ntk_cond_num": + return s1 < s2 + if m_name == "ntk_trace": + return s1 > s2 + if m_name == "ntk_trace_approx": + return s1 > s2 + if m_name == "fisher": + return s1 > s2 + if m_name == "grasp": + return s1 > s2 + if m_name == "snip": + return s1 > s2 + if m_name == "synflow": + return s1 > s2 + if m_name == "weight_norm": + return s1 > s2 + if m_name == "nas_wot": + return s1 > s2 + + +def compare_score_101(m_name: str, s1: float, s2: float) -> bool: + """ + Return if s1 is better than s2, + :param m_name: + :param s1: + :param s2: + :return: if s1 is better than s2 + """ + if m_name == "grad_norm": + return s1 < s2 + if m_name == "grad_plain": + return s1 < s2 + if m_name == "ntk_cond_num": + return s1 < s2 + if m_name == "ntk_trace": + return s1 < s2 + if m_name == "ntk_trace_approx": + return s1 < s2 + if m_name == "fisher": + return s1 < s2 + if m_name == "grasp": + return s1 > s2 + if m_name == "snip": + return s1 < s2 + if m_name == "synflow": + return s1 > s2 + if m_name == "weight_norm": + return s1 > s2 + if m_name == "nas_wot": + return s1 > s2 + + +vote_to_left = {} +vote_to_left["101"] = compare_score_101 +vote_to_left["201"] = compare_score_201 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/trainer.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/trainer.py new file mode 100644 index 0000000000..39ddf2d7e4 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/algo/trainer.py @@ -0,0 +1,535 @@ +import time + +from src.tools import utils + +from singa import singa_wrap as singa +from singa import device as singa_device +from singa import tensor +from singa import opt +from singa import autograd +from singa.opt import Optimizer +from singa.opt import DecayScheduler +from singa.opt import Constant +import numpy as np +import time +import argparse +from PIL import Image + +np_dtype = {"float16": np.float16, "float32": np.float32} + +# singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} +singa_dtype = {"float32": tensor.float32} + +### MSOptimizer +class MSOptimizer(Optimizer): + def __call__(self, loss): + pn_p_g_list = self.call_with_returns(loss) + # print ("optimizer1 before self.step()") + # print ("optimizer1 before print len(pn_p_g_list): \n", len(pn_p_g_list)) + self.step() + # print ("optimizer1 after print len(pn_p_g_list): \n", len(pn_p_g_list)) + # print ("optimizer1 after self.step()") + return pn_p_g_list + + def call_with_returns(self, loss): + # print ("call_with_returns before apply loss.data: \n", loss.data) + pn_p_g_list = [] + for p, g in autograd.backward(loss): + if p.name is None: + p.name = id(p) + self.apply(p.name, p, g) + # print ("call with returns") + # print ("p.name: \n", p.name) + # print ("p.data: \n", p.data) + # print ("g.data: \n", g.data) + pn_p_g_list.append([p.name, p, g]) # need iterables + # print ("call_with_returns after apply loss.data: \n", loss.data) + return pn_p_g_list + +class MSSGD(MSOptimizer): + """Implements stochastic gradient descent (optionally with momentum). + + Nesterov momentum is based on the formula from `On the importance of initialization and momentum in deep learning`__. + + Args: + lr(float): learning rate + momentum(float, optional): momentum factor(default: 0) + weight_decay(float, optional): weight decay(L2 penalty)(default: 0) + dampening(float, optional): dampening for momentum(default: 0) + nesterov(bool, optional): enables Nesterov momentum(default: False) + + Typical usage example: + >> > from singa import opt + >> > optimizer = opt.SGD(lr=0.1, momentum=0.9) + >> > optimizer.update() + + __ http: // www.cs.toronto.edu / %7Ehinton / absps / momentum.pdf + + .. note:: + The implementation of SGD with Momentum / Nesterov subtly differs from + Sutskever et. al. and implementations in some other frameworks. + + Considering the specific case of Momentum, the update can be written as + + .. math:: + v = \rho * v + g \\ + p = p - lr * v + + where p, g, v and: math: `\rho` denote the parameters, gradient, + velocity, and momentum respectively. + + This is in contrast to Sutskever et. al. and + other frameworks which employ an update of the form + + .. math:: + v = \rho * v + lr * g \\ + p = p - v + + The Nesterov version is analogously modified. + """ + + def __init__(self, + lr=0.1, + momentum=0, + dampening=0, + weight_decay=0, + nesterov=False, + dtype=tensor.float32): + super(MSSGD, self).__init__(lr) + + # init momentum + if type(momentum) == float or type(momentum) == int: + if momentum < 0.0: + raise ValueError("Invalid momentum value: {}".format(momentum)) + self.momentum = Constant(momentum) + elif isinstance(momentum, DecayScheduler): + self.momentum = momentum + momentum = momentum.init_value + else: + raise TypeError("Wrong momentum type") + # self.dtype = dtype + # self.mom_value = self.momentum(self.step_counter).as_type(self.dtype) + self.mom_value = self.momentum(self.step_counter) + + # init dampening + if type(dampening) == float or type(dampening) == int: + self.dampening = Constant(dampening) + elif isinstance(dampening, DecayScheduler): + self.dampening = dampening + dampening = dampening.init_value + else: + raise TypeError("Wrong dampening type") + # self.dam_value = self.dampening(self.step_counter).as_type(self.dtype) + self.dam_value = self.dampening(self.step_counter) + + # init weight_decay + if type(weight_decay) == float or type(weight_decay) == int: + if weight_decay < 0.0: + raise ValueError( + "Invalid weight_decay value: {}".format(weight_decay)) + self.weight_decay = Constant(weight_decay) + elif isinstance(weight_decay, DecayScheduler): + self.weight_decay = weight_decay + else: + raise TypeError("Wrong weight_decay type") + # self.decay_value = self.weight_decay(self.step_counter).as_type(self.dtype) + self.decay_value = self.weight_decay(self.step_counter) + + # init other params + self.nesterov = nesterov + self.moments = dict() + + # check value + if nesterov and (momentum <= 0 or dampening != 0): + raise ValueError( + "Nesterov momentum requires a momentum and zero dampening") + + def apply(self, param_name, param_value, param_grad): + """Performs a single optimization step. + + Args: + param_name(String): the name of the param + param_value(Tensor): param values to be update in-place + grad(Tensor): param gradients; the values may be updated + in this function; cannot use it anymore + """ + assert param_value.shape == param_grad.shape, ("shape mismatch", + param_value.shape, + param_grad.shape) + self.device_check(param_value, self.step_counter, self.lr_value, + self.mom_value, self.dam_value, self.decay_value) + + # derive dtype from input + # assert param_value.dtype == self.dtype + + # TODO add branch operator + # if self.decay_value != 0: + if self.weight_decay.init_value != 0: + singa.Axpy(self.decay_value.data, param_value.data, param_grad.data) + + if self.momentum.init_value != 0: + if param_name not in self.moments: + flag = param_value.device.graph_enabled() + param_value.device.EnableGraph(False) + self.moments[param_name] = tensor.zeros_like(param_value) + param_value.device.EnableGraph(flag) + + buf = self.moments[param_name] + buf *= self.mom_value + alpha = 1.0 - self.dam_value + singa.Axpy(alpha.data, param_grad.data, buf.data) + + if self.nesterov: + singa.Axpy(self.mom_value.data, buf.data, param_grad.data) + else: + param_grad = buf + + minus_lr = 0.0 - self.lr_value + singa.Axpy(minus_lr.data, param_grad.data, param_value.data) + + def step(self): + # increment step counter, lr and moment + # print ("before super step") + super().step() + # print ("after super step") + # print ("before custiomized step") + # mom_value = self.momentum(self.step_counter).as_type(self.dtype) + # dam_value = self.dampening(self.step_counter).as_type(self.dtype) + # decay_value = self.weight_decay(self.step_counter).as_type(self.dtype) + mom_value = self.momentum(self.step_counter) + dam_value = self.dampening(self.step_counter) + decay_value = self.weight_decay(self.step_counter) + self.mom_value.copy_from(mom_value) + self.dam_value.copy_from(dam_value) + self.decay_value.copy_from(decay_value) + # print ("after customized step") + + def get_states(self): + states = super().get_states() + if self.mom_value > 0: + states[ + 'moments'] = self.moments # a dict for 1st order moments tensors + return states + + def set_states(self, states): + super().set_states(states) + if 'moments' in states: + self.moments = states['moments'] + self.mom_value = self.momentum(self.step_counter) + +# Data augmentation +def augmentation(x, batch_size): + xpad = np.pad(x, [[0, 0], [0, 0], [4, 4], [4, 4]], 'symmetric') + for data_num in range(0, batch_size): + offset = np.random.randint(8, size=2) + x[data_num, :, :, :] = xpad[data_num, :, + offset[0]:offset[0] + x.shape[2], + offset[1]:offset[1] + x.shape[2]] + if_flip = np.random.randint(2) + if (if_flip): + x[data_num, :, :, :] = x[data_num, :, :, ::-1] + return x + + +# Calculate accuracy +def accuracy(pred, target): + # y is network output to be compared with ground truth (int) + y = np.argmax(pred, axis=1) + # print ("in accuracy y shape: ", y.shape) + # print ("in accuracy target shape: ", target.shape) + a = y == target + correct = np.array(a, "int").sum() + return correct + + +# Data partition according to the rank +def partition(global_rank, world_size, train_x, train_y, val_x, val_y): + # Partition training data + data_per_rank = train_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + train_x = train_x[idx_start:idx_end] + train_y = train_y[idx_start:idx_end] + + # Partition evaluation data + data_per_rank = val_x.shape[0] // world_size + idx_start = global_rank * data_per_rank + idx_end = (global_rank + 1) * data_per_rank + val_x = val_x[idx_start:idx_end] + val_y = val_y[idx_start:idx_end] + return train_x, train_y, val_x, val_y + + +# Function to all reduce NUMPY accuracy and loss from multiple devices +def reduce_variable(variable, dist_opt, reducer): + reducer.copy_from_numpy(variable) + dist_opt.all_reduce(reducer.data) + dist_opt.wait() + output = tensor.to_numpy(reducer) + return output + + +def resize_dataset(x, image_size): + num_data = x.shape[0] + dim = x.shape[1] + X = np.zeros(shape=(num_data, dim, image_size, image_size), + dtype=np.float32) + for n in range(0, num_data): + for d in range(0, dim): + X[n, d, :, :] = np.array(Image.fromarray(x[n, d, :, :]).resize( + (image_size, image_size), Image.BILINEAR), + dtype=np.float32) + return X + +from torch.utils.data import DataLoader +class ModelTrainer: + + @classmethod + def fully_train_arch(cls, + model, + use_test_acc: bool, + epoch_num, + train_loader: DataLoader, + val_loader: DataLoader, + test_loader: DataLoader, + args, + logger=None + ) -> (float, float, dict): + """ + Args: + model: + use_test_acc: + epoch_num: how many epoch, set by scheduler + train_loader: + val_loader: + test_loader: + args: + Returns: + """ + + if logger is None: + from src.logger import logger + logger = logger + + start_time, best_valid_auc = time.time(), 0. + + num_labels = args.num_labels + lr = args.lr + iter_per_epoch = args.iter_per_epoch + # report_freq = args.report_freq + # given_patience = args.patience + + # assign new values + args.epoch_num = epoch_num + + # for multiple classification + + # optimizer + precision = 'float32' + mssgd = MSSGD(lr=args.lr, momentum=0.9, weight_decay=1e-4, dtype=singa_dtype[precision]) + device_id = 0 + max_epoch = epoch_num + graph = True + verbosity = 0 + dist_option='plain' + spars=None + global_rank = 0 + world_size = 1 + + # training params + if args.device == 'cpu': + dev = singa_device.get_default_device() + else: # GPU + dev = singa_device.create_cuda_gpu_on(args.local_rank) # need to change to CPU device for CPU-only machines + dev.SetRandSeed(0) + + # For distributed training, sequential has better performance + if hasattr(mssgd, "communicator"): + DIST = True + sequential = True + else: + DIST = False + sequential = False + + info_dic = {} + valid_auc = -1 + valid_loss = 0 + + ### singa data + tx = tensor.Tensor((args.batch_size, args.nfeat), dev, singa_dtype[precision]) + ty = tensor.Tensor((args.batch_size,), dev, tensor.int32) + ### singa data + + model.set_optimizer(mssgd) + model.compile([tx], is_train=True, use_graph=graph, sequential=sequential) + dev.SetVerbosity(verbosity) + + # Training and evaluation loop + for epoch in range(max_epoch): + start_time = time.time() + logger.info(f'Epoch [{epoch:3d}/{epoch_num:3d}]') + # np.random.shuffle(idx) + + if global_rank == 0: + print('Starting Epoch %d:' % (epoch)) + + # Training phase + train_correct = np.zeros(shape=[1], dtype=np.float32) + test_correct = np.zeros(shape=[1], dtype=np.float32) + train_loss = np.zeros(shape=[1], dtype=np.float32) + + model.train() + # print ("num_train_batch: \n", num_train_batch) + # print () + batch_idx = 0 + # for b in range(num_train_batch): + for batch_idx, batch in enumerate(train_loader): + if batch_idx % 50 == 0: + print ("trainer.py train batch_idx: \n", batch_idx) + # Generate the batch data in this iteration + # x = train_x[idx[b * batch_size:(b + 1) * batch_size]] + # if model.dimension == 4: + # x = augmentation(x, batch_size) + # if (image_size != model.input_size): + # x = resize_dataset(x, model.input_size) + # x = x.astype(np_dtype[precision]) + # y = train_y[idx[b * batch_size:(b + 1) * batch_size]] + + y = batch['y'].cpu().numpy() + batch['id'] = batch['id'].cpu().numpy().astype(int) + # batch['value'] = batch['value'].to(args.device) + x = np.zeros((batch['id'].shape[0], args.nfeat), dtype=np.float32) + # print ("target shape: ", target.shape) + # print ("target: ", target) + # print ("batch['id'] shape: ", batch['id'].shape) + # print ("batch['id']: ", batch['id']) + # print ("batch['value'] shape: ", batch['value'].shape) + # print ("batch['value']: ", batch['value']) + # print ("batch['id'].cpu().numpy().astype(int): \n", batch['id'].cpu().numpy().astype(int)) + for i in range(batch['id'].shape[0]): + x[i][batch['id'][i]] = (np.float32)(1.0) + x = x.astype(dtype=np.float32) + y = y.astype(dtype=np.int32) + + if x.shape[0] != args.batch_size: # last batch not processing + continue + + synflow_flag = False + # Train the model + # if True: # normal train steps + # Copy the patch data into input tensors + # print ("normal train steps\n") + # print ("x.astype(np.float32): \n", x.astype(np.float32)) + # print ("y: \n", y) + tx = tensor.Tensor(x.shape, dev, singa_dtype[precision]) + ty = tensor.Tensor((y.shape[0],), dev, tensor.int32) + tx.copy_from_numpy(x) # dtype=np.float32 + # print ("tx: \n", tx) + ty.copy_from_numpy(y) + # print ("ty: \n", ty) + # print ("normal before model(tx, ty, synflow_flag, dist_option, spars)") + # print ("train_cnn tx: \n", tx) + # print ("train_cnn ty: \n", ty) + # print ("trainer.py train before model forward ...") + # print ("model: ", model) + pn_p_g_list, out, loss = model(tx, ty, dist_option, spars, synflow_flag) + # print ("trainer.py train normal after model(tx, ty, synflow_flag, dist_option, spars)") + # print ("trainer.py train tx shape: ", tx.shape) + # print ("trainer.py train ty shape: ", ty.shape) + # print ("trainer.py train out.shape: ", out.shape) + # print ("trainer.py train out: ", out) + # print ("trainer.py train y shape: ", y.shape) + train_correct += accuracy(tensor.to_numpy(out), y) + train_loss += tensor.to_numpy(loss)[0] + + if DIST: + # Reduce the evaluation accuracy and loss from multiple devices + reducer = tensor.Tensor((1,), dev, tensor.float32) + train_correct = reduce_variable(train_correct, mssgd, reducer) + train_loss = reduce_variable(train_loss, mssgd, reducer) + + if global_rank == 0: + print('Training loss = %f, training accuracy = %f' % + (train_loss, train_correct / + (batch_idx * args.batch_size * world_size)), + flush=True) + print ("train total batch_idx: ", batch_idx) + train_metric = train_correct / (batch_idx * args.batch_size * world_size) + + # Evaluation phase + model.eval() + batch_idx = 0 + # for b in range(num_val_batch): + # print ("evaluation begins") + for batch_idx, batch in enumerate(test_loader): + # print ("trainer.py test batch_idx: \n", batch_idx) + # x = val_x[b * batch_size:(b + 1) * batch_size] + # if model.dimension == 4: + # if (image_size != model.input_size): + # x = resize_dataset(x, model.input_size) + # x = x.astype(np_dtype[precision]) + # y = val_y[b * batch_size:(b + 1) * batch_size] + # batch['value'] = batch['value'].cpu().numpy().astype(np_dtype[precision]) + # x = batch['value'].cpu().numpy().astype(np_dtype[precision]) + + y = batch['y'].cpu().numpy() + batch['id'] = batch['id'].cpu().numpy().astype(int) + # batch['value'] = batch['value'].to(args.device) + x = np.zeros((batch['id'].shape[0], args.nfeat), dtype=np.float32) + # print ("target shape: ", target.shape) + # print ("target: ", target) + # print ("batch['id'] shape: ", batch['id'].shape) + # print ("batch['id']: ", batch['id']) + # print ("batch['value'] shape: ", batch['value'].shape) + # print ("batch['value']: ", batch['value']) + # print ("batch['id'].cpu().numpy().astype(int): \n", batch['id'].cpu().numpy().astype(int)) + for i in range(batch['id'].shape[0]): + x[i][batch['id'][i]] = (np.float32)(1.0) + # print ("x[1]: \n", x[1]) + x = x.astype(dtype=np.float32) + y = y.astype(dtype=np.int32) + + if x.shape[0] != (args.batch_size * 8): # last batch not processing + # print ("trainer.py test batch_idx: ", batch_idx) + # print ("trainer.py test x.shape: ", x.shape) + continue + + tx = tensor.Tensor(x.shape, dev, singa_dtype[precision]) + ty = tensor.Tensor((y.shape[0],), dev, tensor.int32) + tx.copy_from_numpy(x) + ty.copy_from_numpy(y) + # print ("trainer.py test tx shape: ", tx.shape) + out_test = model(tx) + # print ("trainer.py test out_test shape: ", out_test.shape) + # print ("trainer.py test y shape: ", y.shape) + # print ("trainer.py out_test: ", out_test) + # print ("trainer.py y: ", y) + test_correct += accuracy(tensor.to_numpy(out_test), y) + # print ("test_correct: ", test_correct) + + if DIST: + # Reduce the evaulation accuracy from multiple devices + test_correct = reduce_variable(test_correct, mssgd, reducer) + + # Output the evaluation accuracy + if global_rank == 0: + print('Evaluation accuracy = %f, Elapsed Time = %fs' % + (test_correct / (batch_idx * args.batch_size * 8 * world_size), + time.time() - start_time), + flush=True) + # print ("test all batch_idx: ", batch_idx) + test_metric = test_correct / (batch_idx * args.batch_size * 8 * world_size) + + + info_dic[epoch] = { + "train_metric": str(train_metric[0]), + "test_metric": str(test_metric[0]), + "train_loss": str(train_loss[0]), + # "valid_loss": valid_loss, + "train_test_total_time": str(time.time() - start_time)} + + dev.PrintTimeProfiling() + + # return valid_auc, time.time() - start_time, info_dic + print ("info_dic: ", info_dic) + return test_metric, time.time() - start_time, info_dic diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_sr.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_sr.py new file mode 100644 index 0000000000..d723fe7b18 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_sr.py @@ -0,0 +1,126 @@ +from copy import copy +from src.common.constant import Config + + +class BudgetAwareControllerSR: + def __init__(self, evaluator, time_per_epoch, max_unit=200): + """ + :param evaluator: + :param max_unit: for 201, it's 200, for 101 it's 108 + """ + self._evaluator = evaluator + self.max_unit_per_model = max_unit + self.time_per_epoch = time_per_epoch + self.name = "SUCCREJCT" + + def schedule_budget_per_model_based_on_T(self, space_name, fixed_time_budget, K_): + # for benchmarking only phase 2 + + # try different K and U combinations + # only consider 15625 arches in this paper + # min_budget_required: when K = 1, N = min_budget_required * 1 + if space_name == Config.NB101: + U_options = [4, 12, 16, 108] + else: + U_options = list(range(1, 200)) + + history = [] + + for U in U_options: + expected_time_used = self.pre_calculate_epoch_required(K_, U) * self.time_per_epoch + if expected_time_used > fixed_time_budget: + break + else: + history.append(U) + if len(history) == 0: + raise f"{fixed_time_budget} is too small for current config" + return history[-1] + + def pre_calculate_epoch_required(self, K, U): + """ + :param K: candidates lists + :param U: min resource each candidate needs + :return: + """ + total_epoch_each_rounds = K * U + min_budget_required = 0 + + previous_epoch = None + while True: + cur_cand_num = K + if cur_cand_num == 1: + break + # number of each res given to each cand, pick lower bound + epoch_per_model = int(total_epoch_each_rounds / cur_cand_num) + if previous_epoch is None: + previous_epoch = epoch_per_model + elif previous_epoch == epoch_per_model: + # which means the epoch don't increase, no need to re-evaluate each component + K = cur_cand_num - 1 + continue + + if epoch_per_model >= self.max_unit_per_model: + epoch_per_model = self.max_unit_per_model + # evaluate each arch + min_budget_required += epoch_per_model * cur_cand_num + # sort from min to max + if epoch_per_model == self.max_unit_per_model: + # each model is fully evaluated, just return top 1 + K = 1 + else: + # only keep 1/eta, pick lower bound + K = cur_cand_num - 1 + return min_budget_required + + def run_phase2(self, U: int, candidates_m: list): + """ + :param candidates_m: candidates lists + :param U: min resource each candidate needs + :return: + """ + # print(f" *********** begin BudgetAwareControllerSR with U={U}, K={len(candidates_m)} ***********") + candidates = copy(candidates_m) + total_epoch_each_rounds = len(candidates) * U + min_budget_required = 0 + previous_epoch = None + scored_cand = None + while True: + cur_cand_num = len(candidates) + if cur_cand_num == 1: + break + total_score = [] + # number of each res given to each cand, pick lower bound + epoch_per_model = int(total_epoch_each_rounds / cur_cand_num) + + if previous_epoch is None: + previous_epoch = epoch_per_model + elif previous_epoch == epoch_per_model: + # which means the epoch don't increase, no need to re-evaluate each component + num_keep = cur_cand_num - 1 + candidates = [ele[0] for ele in scored_cand[-num_keep:]] + continue + + if epoch_per_model >= self.max_unit_per_model: + epoch_per_model = self.max_unit_per_model + + # print(f"[successive_reject]: {cur_cand_num} model left, " + # f"and evaluate each model with {epoch_per_model} epoch") + # evaluate each arch + for cand in candidates: + score = self._evaluator.p2_evaluate(cand, epoch_per_model) + total_score.append((cand, score)) + min_budget_required += epoch_per_model + # sort from min to max + scored_cand = sorted(total_score, key=lambda x: x[1]) + + if epoch_per_model == self.max_unit_per_model: + # each model is fully evaluated, just return top 1 + candidates = [scored_cand[-1][0]] + else: + # only keep m-1, remove the worst one + num_keep = cur_cand_num - 1 + candidates = [ele[0] for ele in scored_cand[-num_keep:]] + + return candidates[0], None, min_budget_required + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_uniform.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_uniform.py new file mode 100644 index 0000000000..c770c42806 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/eva_engine/phase2/run_uniform.py @@ -0,0 +1,78 @@ + + +from copy import copy +from random import randint +from src.common.constant import Config + + +# UniformAllocation +class UniformAllocation: + + def __init__(self, evaluator, time_per_epoch, max_unit=200): + """ + :param evaluator: + :param max_unit: for 201, it's 200, for 101 it's 108 + """ + self._evaluator = evaluator + self.max_unit_per_model = max_unit + self.time_per_epoch = time_per_epoch + self.name = "UNIFORM" + + def schedule_budget_per_model_based_on_T(self, space_name, fixed_time_budget, K_): + # for benchmarking only phase 2 + + # try different K and U combinations + # only consider 15625 arches in this paper + # min_budget_required: when K = 1, N = min_budget_required * 1 + if space_name == Config.NB101: + U_options = [4, 12, 16, 108] + else: + U_options = list(range(1, 200)) + + history = [] + + for U in U_options: + expected_time_used = self.pre_calculate_epoch_required(K_, U) * self.time_per_epoch + if expected_time_used > fixed_time_budget: + break + else: + history.append(U) + return history[-1] + + def pre_calculate_epoch_required(self, K, U): + """ + :param B: total budget for phase 2 + :param U: mini unit computation for each modle + :param candidates_m: + :return: + """ + return K*U + + def run_phase2(self, U: int, candidates_m: list): + """ + :param U: mini unit computation for each modle + :param candidates_m: + :return: + """ + + # print(f" *********** begin uniformly_allocate with U={U}, K={len(candidates_m)} ***********") + + candidates = copy(candidates_m) + min_budget_required = 0 + + if U >= self.max_unit_per_model: + U = self.max_unit_per_model + + # print(f"[uniformly_allocate]: uniformly allocate {U} epoch to each model") + + total_score = [] + for cand in candidates: + score = self._evaluator.p2_evaluate(cand, U) + total_score.append((cand, score)) + min_budget_required += U + # sort from min to max + scored_cand = sorted(total_score, key=lambda x: x[1]) + candidate = scored_cand[-1][0] + return candidate, None, min_budget_required + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/logger/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/logger/__init__.py new file mode 100644 index 0000000000..8d2e16d4ec --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/logger/__init__.py @@ -0,0 +1,23 @@ +import logging +import os + +if os.environ.get("log_logger_folder_name") == None: + log_logger_folder_name = "logs_default" + if not os.path.exists(f"./{log_logger_folder_name}"): + os.makedirs(f"./{log_logger_folder_name}") +else: + log_logger_folder_name = os.environ.get("log_logger_folder_name") + if not os.path.exists(log_logger_folder_name): + os.makedirs(log_logger_folder_name) + +logger = logging.getLogger(__name__) + +if os.environ.get("log_file_name") == None: + log_name = f"{log_logger_folder_name}/test.log" +else: + log_name = f"{log_logger_folder_name}/" + os.environ.get("log_file_name") + +logging.basicConfig(level=logging.INFO, + format='%(asctime)s %(levelname)-8s %(message)s', + datefmt='%d %b %Y %H:%M:%S', + filename=log_name, filemode='w') diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/README.md b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/README.md new file mode 100644 index 0000000000..a22a70e9ab --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/README.md @@ -0,0 +1,2 @@ + +This is used to parse the local json file which is the result of the all experiments \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_explore_ea.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_explore_ea.py new file mode 100644 index 0000000000..1b074a1e73 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_explore_ea.py @@ -0,0 +1,78 @@ +import json +import os +import sqlite3 +import traceback + +from src.common.constant import Config + +base_folder_dir = os.environ.get("base_dir") +if base_folder_dir is None: base_folder_dir = os.getcwd() +base_dir = os.path.join(base_folder_dir, "img_data", "ground_truth") +print("local api running at {}".format(base_dir)) + +# sum score is better +tf_smt_file_NB101C10 = os.path.join(base_dir, "TFMEM_101_c10_100run_8k_models_score_sum") +tf_smt_file_NB201C10 = os.path.join(base_dir, "TFMEM_201_c10_100run_score_sum") +tf_smt_file_NB201C100 = os.path.join(base_dir, "TFMEM_201_c100_100run_score_sum") +tf_smt_file_NB201Img = os.path.join(base_dir, "TFMEM_201_imgNet_100run_score_sum") + +# rank is not as good as sum +# tf_smt_file_NB201C10 = os.path.join(base_dir, "TFMEM_201_c10_100run_rank_bugs") +# tf_smt_file_NB201C100 = os.path.join(base_dir, "TFMEM_201_c100_200run_rank") +# tf_smt_file_NB201Img = os.path.join(base_dir, "TFMEM_201_imgNet_200run_rank") + +con = None +cur = None + + +# fetch result from simulated result +def fetch_from_db(space_name, dataset, run_id_m, N_m): + """ + :param run_id_m: run_id 100 max + :param B1_m: number of models evaluted + :return: + """ + global con + global cur + if con is None: + if space_name == Config.NB201: + if dataset == Config.c10: + tf_smt_used = tf_smt_file_NB201C10 + elif dataset == Config.c100: + tf_smt_used = tf_smt_file_NB201C100 + elif dataset == Config.imgNet: + tf_smt_used = tf_smt_file_NB201Img + else: + print(f"{dataset} is Not implemented") + raise + elif space_name == Config.NB101: + if dataset == Config.c10: + tf_smt_used = tf_smt_file_NB101C10 + else: + print(f"{dataset}Not implemented") + raise + else: + print(f"{space_name} is Not implemented") + raise + + print(tf_smt_used) + con = sqlite3.connect(tf_smt_used) + cur = con.cursor() + + res = cur.execute( + "SELECT * FROM simulateExp WHERE run_num = {} and model_explored = {}".format(run_id_m, N_m)) + fetch_res = res.fetchone() + + try: + arch_id = fetch_res[2] + candidates = json.loads(fetch_res[3]) + current_time = float(fetch_res[4]) + except: + print(traceback.format_exc()) + raise f"res is None when using run_id ={run_id_m} and bm = {N_m}" + + return arch_id, candidates, current_time + + +if __name__ == '__main__': + print(fetch_from_db(Config.NB201, Config.c10, 3, 10)) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_train_baseline.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_train_baseline.py new file mode 100644 index 0000000000..54848af6fa --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/img_train_baseline.py @@ -0,0 +1,113 @@ + + +import os +import numpy as np +from src.common.constant import Config +from src.tools.io_tools import read_json + +base_dir_folder = os.environ.get("base_dir") +if base_dir_folder is None:base_dir_folder = os.getcwd() +base_dir = os.path.join(base_dir_folder, "img_data") + +print("gt_api running at {}".format(base_dir)) +train_base201_c10 = os.path.join(base_dir, "train_based_201_c10.json") +train_base201_c100 = os.path.join(base_dir, "train_based_201_c100.json") +train_base201_img = os.path.join(base_dir, "train_based_201_img.json") + +train_base101_c10 = os.path.join(base_dir, "train_based_101_c10_100run_24k_models.json") + + +def post_processing_train_base_result(search_space, dataset, x_max_value: int = None): + + if search_space == Config.NB201 and dataset == Config.c10: + data = read_json(train_base201_c10) + + elif search_space == Config.NB201 and dataset == Config.c100: + data = read_json(train_base201_c100) + elif search_space == Config.NB201 and dataset == Config.imgNet: + data = read_json(train_base201_img) + + elif search_space == Config.NB101 and dataset == Config.c10: + data = read_json(train_base101_c10) + else: + print(f"Cannot read dataset {dataset} of file") + raise + + # data is in form of + """ + data[run_id] = {} + data[run_id]["arch_id_list"] + data[run_id]["current_best_acc"] + data[run_id]["x_axis_time"] + """ + + acc_got_row = [] + time_used_row = [] + min_arch_across_all_run = 15625 + for run_id in data: + acc_got_row.append(data[run_id]["current_best_acc"]) + time_used_row.append(data[run_id]["x_axis_time"]) + if len(data[run_id]["current_best_acc"]) < min_arch_across_all_run: + min_arch_across_all_run = len(data[run_id]["current_best_acc"]) + + # for each run, only use min_arch_across_all_run + for i in range(len(acc_got_row)): + acc_got_row[i] = acc_got_row[i][:min_arch_across_all_run] + time_used_row[i] = time_used_row[i][:min_arch_across_all_run] + + acc_got = np.array(acc_got_row) + time_used = np.array(time_used_row) + + if data['0']["current_best_acc"][-1] < 1: + acc_got = acc_got * 100 + + acc_l = np.quantile(acc_got, 0.25, axis=0) + acc_m = np.quantile(acc_got, 0.5, axis=0) + acc_h = np.quantile(acc_got, 0.75, axis=0) + + time_l = np.quantile(time_used, 0.25, axis=0) + time_m = np.quantile(time_used, 0.5, axis=0).tolist() + time_h = np.quantile(time_used, 0.75, axis=0) + + x_list = [ele/60 for ele in time_m] + y_list_low = acc_l[:len(x_list)] + y_list_m = acc_m[:len(x_list)] + y_list_high = acc_h[:len(x_list)] + + # if the x array max value is provided. + if x_max_value is not None: + final_x_list = [] + final_x_list_low = [] + final_x_list_m = [] + final_x_list_high = [] + for i in range(len(x_list)): + if x_list[i] <= x_max_value: + final_x_list.append(x_list[i]) + final_x_list_low.append(y_list_low[i]) + final_x_list_m.append(y_list_m[i]) + final_x_list_high.append(y_list_high[i]) + else: + break + return final_x_list, final_x_list_low, final_x_list_m, final_x_list_high + else: + return x_list, y_list_low.tolist(), y_list_m.tolist(), y_list_high.tolist() + + +if __name__ == "__main__": + search_space = Config.NB201 + dataset = Config.c100 + x_list, y_list_low, y_list_m, y_list_high = post_processing_train_base_result(search_space, dataset) + + from matplotlib import pyplot as plt + + plt.fill_between(x_list, y_list_low, y_list_high, alpha=0.1) + plt.plot(x_list, y_list_m, "-*", label="Training-based") + + plt.xscale("symlog") + plt.grid() + plt.xlabel("Time Budget given by user (mins)") + plt.ylabel("Test Accuracy") + plt.legend() + plt.show() + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/interface.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/interface.py new file mode 100644 index 0000000000..20cccabe90 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/interface.py @@ -0,0 +1,123 @@ +# query ground truth +from src.common.constant import Config, CommonVars +from src.query_api.query_api_img import Gt201, Gt101 +from src.query_api.query_api_mlp import GTMLP +from src.query_api.query_api_img import ImgScoreQueryApi +from typing import * + + +def profile_NK_trade_off(dataset): + """ + This is get from the profling result. + We try various N/K combinations, and find this is better. + """ + if dataset == Config.c10: + return 85 + elif dataset == Config.c100: + return 85 + elif dataset == Config.imgNet: + return 130 + else: + return 30 + + +class SimulateTrain: + + def __init__(self, space_name: str): + """ + :param space_name: NB101 or NB201, MLP + """ + self.space_name = space_name + self.api = None + + # get the test_acc and time usage to train of this arch_id + def get_ground_truth(self, arch_id: str, dataset: str, epoch_num: int = None, total_epoch: int = 200): + """ + :param arch_id: + :param dataset: + :param epoch_num: which epoch's performance to return + :param total_epoch: + """ + if self.space_name == Config.NB101: + self.api = Gt101() + acc, time_usage = self.api.get_c10_test_info(arch_id, dataset, epoch_num) + return acc, time_usage + + elif self.space_name == Config.NB201: + self.api = Gt201() + if total_epoch == 200: + acc, time_usage = self.api.query_200_epoch(arch_id, dataset, epoch_num) + else: # 12 + acc, time_usage = self.api.query_12_epoch(arch_id, dataset, epoch_num) + return acc, time_usage + + elif self.space_name == Config.MLPSP: + self.api = GTMLP(dataset) + acc, time_usage = self.api.get_valid_auc(arch_id, epoch_num) + return acc, time_usage + + else: + raise NotImplementedError + + # get the high acc of k arch with highest score + def get_high_acc_top_10(self, top10): + all_top10_acc = [] + time_usage = 0 + for arch_id in top10: + score_, time_usage_ = self.get_ground_truth(arch_id) + all_top10_acc.append(score_) + time_usage += time_usage_ + return max(all_top10_acc), time_usage + + def get_best_arch_id(self, top10): + cur_best = 0 + res = None + for arch_id in top10: + acc, _ = self.get_ground_truth(arch_id) + if acc > cur_best: + cur_best = acc + res = arch_id + return res + + def query_all_model_ids(self, dataset): + if self.space_name == Config.NB101: + self.api = Gt101() + elif self.space_name == Config.NB201: + self.api = Gt201() + elif self.space_name == Config.MLPSP: + self.api = GTMLP(dataset) + return self.api.get_all_trained_model_ids() + + +class SimulateScore: + def __init__(self, space_name: str, dataset_name: str): + """ + :param space_name: NB101 or NB201, MLP + :param dataset_name: NB101 or NB201, MLP + """ + self.space_name = space_name + if self.space_name == Config.MLPSP: + self.api = GTMLP(dataset_name) + else: + self.api = ImgScoreQueryApi(self.space_name, dataset_name) + + # get the test_acc and time usage to train of this arch_id + def query_tfmem_rank_score(self, arch_id) -> Dict: + # todo: here we use the global rank, other than dymalically update the rank + # todo: so, we directly return the rank_score, instead of the mutilpel_algs score + # return {"nas_wot": self.api.get_metrics_score(arch_id, dataset)["nas_wot"], + # "synflow": self.api.get_metrics_score(arch_id, dataset)["synflow"], + # } + return self.api.get_global_rank_score(arch_id) + + def query_all_tfmem_score(self, arch_id) -> Dict: + """ + return {alg_name: score} + """ + return self.api.api_get_score(arch_id) + + def query_all_model_ids(self, dataset) -> List: + """ + return all models_ids as a list + """ + return self.api.get_all_scored_model_ids() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_img.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_img.py new file mode 100644 index 0000000000..87b466e6b9 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_img.py @@ -0,0 +1,277 @@ +import os +import random +from src.common.constant import Config +from src.tools.io_tools import read_json, write_json +from src.query_api.singleton import Singleton +from src.tools.io_tools import read_pickle +from src.tools.compute import generate_global_rank + + +base_dir_folder = os.environ.get("base_dir") +if base_dir_folder is None: base_dir_folder = os.getcwd() +base_dir = os.path.join(base_dir_folder, "img_data") +print("local api running at {}".format(base_dir)) + +# todo: move all those to a config file +# score result +pre_score_path_101C10 = os.path.join(base_dir, "score_101_15k_c10_128.json") +pre_score_path_201C10 = os.path.join(base_dir, "score_201_15k_c10_bs32_ic16.json") +pre_score_path_201C100 = os.path.join(base_dir, "score_201_15k_c100_bs32_ic16.json") +pre_score_path_201IMG = os.path.join(base_dir, "score_201_15k_imgNet_bs32_ic16.json") + +# expreflow +expreflow_score_path_101C10 = os.path.join(base_dir, "score_nasbench101_cifar10_batch_size_32_cpu.json") +# expreflow_score_path_201C10 = os.path.join(base_dir, "score_nasbench201_cifar10_batch_size_32_cpu.json") +# expreflow_score_path_201C100 = os.path.join(base_dir, "score_nasbench201_cifar100_batch_size_32_cpu.json") +# expreflow_score_path_201IMG = os.path.join(base_dir, "score_nasbench201_ImageNet16-120_batch_size_32_cpu.json") + +expreflow_score_path_201C10 = os.path.join(base_dir_folder, "score_scale_traj_width/score_nasbench201_cifar10_batch_size_32_cpu.json") +expreflow_score_path_201C100 = os.path.join(base_dir_folder, "score_scale_traj_width/score_nasbench201_cifar100_batch_size_32_cpu.json") +expreflow_score_path_201IMG = os.path.join(base_dir_folder, "score_scale_traj_width/score_nasbench201_ImageNet16-120_batch_size_32_cpu.json") + +# training accuracy result. +gt201 = os.path.join(base_dir, "ground_truth/201_allEpoch_info") +gt101 = os.path.join(base_dir, "ground_truth/101_allEpoch_info_json") +gt101P = os.path.join(base_dir, "ground_truth/nasbench1_accuracy.p") +id_to_hash_path = os.path.join(base_dir, "ground_truth/nb101_id_to_hash.json") + + +# We pre-compute the time usage, and get a range, +# Then we randomly pick one value from the range each time +def guess_score_time(search_space_m, dataset): + if search_space_m == Config.NB101: + return Gt101.guess_score_time() + if search_space_m == Config.NB201: + return Gt201.guess_score_time(dataset) + + +def guess_train_one_epoch_time(search_space_m, dataset): + if search_space_m == Config.NB101: + return Gt101().guess_train_one_epoch_time() + if search_space_m == Config.NB201: + return Gt201().guess_train_one_epoch_time(dataset) + + +class ImgScoreQueryApi: + # Multiton pattern + # use those algoroithm => new tfmem + default_alg_name_list = ["nas_wot", "synflow"] + _instances = {} + + def __new__(cls, search_space_name: str, dataset: str): + if (search_space_name, dataset) not in cls._instances: + instance = super(ImgScoreQueryApi, cls).__new__(cls) + instance.search_space_name, instance.dataset = search_space_name, dataset + + # read pre-scored file path + if search_space_name == Config.NB201: + if dataset == Config.c10: + instance.pre_score_path = pre_score_path_201C10 + instance.express_score_path = expreflow_score_path_201C10 + elif dataset == Config.c100: + instance.pre_score_path = pre_score_path_201C100 + instance.express_score_path = expreflow_score_path_201C100 + elif dataset == Config.imgNet: + instance.pre_score_path = pre_score_path_201IMG + instance.express_score_path = expreflow_score_path_201IMG + if search_space_name == Config.NB101: + instance.pre_score_path = pre_score_path_101C10 + instance.express_score_path = expreflow_score_path_101C10 + + instance.data = read_json(instance.pre_score_path) + express_score_data = read_json(instance.express_score_path) + for arch_id in express_score_data: + if arch_id in instance.data: + instance.data[arch_id].update(express_score_data[arch_id]) + else: + instance.data[arch_id] = express_score_data[arch_id] + + instance.global_rank = generate_global_rank( + instance.data, instance.default_alg_name_list) + + cls._instances[(search_space_name, dataset)] = instance + return cls._instances[(search_space_name, dataset)] + + def api_get_score(self, arch_id: str, tfmem: str = None): + # retrieve score from pre-scored file + if tfmem is None: + return self.data[arch_id] + else: + return {tfmem: float(self.data[arch_id][tfmem])} + + def update_existing_data(self, arch_id, alg_name, score_str): + """ + Add new arch's inf into data + :param arch_id: + :param alg_name: + :param score_str: + :return: + """ + if str(arch_id) not in self.data: + self.data[str(arch_id)] = {} + else: + self.data[str(arch_id)] = self.data[str(arch_id)] + self.data[str(arch_id)][alg_name] = '{:f}'.format(score_str) + + def is_arch_and_alg_inside_data(self, arch_id, alg_name): + if arch_id in self.data and alg_name in self.data[arch_id]: + return True + else: + return False + + def is_arch_inside_data(self, arch_id): + if arch_id in self.data: + return True + else: + return False + + def get_len_data(self): + return len(self.data) + + def save_latest_data(self): + """ + update the latest score data + """ + write_json(self.pre_score_path, self.data) + + def get_all_scored_model_ids(self): + return list(self.data.keys()) + + def get_global_rank_score(self, arch_id): + return self.global_rank[arch_id] + + +class Gt201(metaclass=Singleton): + + @classmethod + def guess_score_time(cls, dataset=Config.c10): + return random.randint(3315, 4502) * 0.0001 + + def __init__(self): + self.data201 = read_json(gt201) + + def get_c10valid_200epoch_test_info(self, arch_id: int): + """ + cifar10-valid means train with train set, valid with validation dataset + Thus, acc is lower than train with train+valid. + :param arch_id: + :return: + """ + return self.query_200_epoch(str(arch_id), Config.c10_valid) + + def get_c10_200epoch_test_info(self, arch_id: int): + """ + cifar10-valid means train with train set, valid with validation dataset + Thus, acc is lower than train with train+valid. + :param arch_id: + :return: + """ + return self.query_200_epoch(str(arch_id), Config.c10) + + def get_c100_200epoch_test_info(self, arch_id: int): + return self.query_200_epoch(str(arch_id), Config.c100) + + def get_imgNet_200epoch_test_info(self, arch_id: int): + return self.query_200_epoch(str(arch_id), Config.imgNet) + + def query_200_epoch(self, arch_id: str, dataset, epoch_num: int = 199): + if epoch_num is None or epoch_num > 199: + epoch_num = 199 + arch_id = str(arch_id) + t_acc = self.data201[arch_id]["200"][dataset][str(epoch_num)]["test_accuracy"] + time_usage = self.data201[arch_id]["200"][dataset][str(epoch_num)]["time_usage"] + return t_acc, time_usage + + def query_12_epoch(self, arch_id: str, dataset, epoch_num: int = 11): + if epoch_num is None or epoch_num > 11: + epoch_num = 11 + arch_id = str(arch_id) + t_acc = self.data201[arch_id]["12"][dataset][str(epoch_num)]["test_accuracy"] + time_usage = self.data201[arch_id]["12"][dataset][str(epoch_num)]["time_usage"] + return t_acc, time_usage + + def count_models(self): + return len(self.data201) + + def guess_train_one_epoch_time(self, dataset): + if dataset == Config.c10: + dataset = Config.c10_valid + # pick the max value over 5k arch training time, it's 40 + # res = 0 + # for arch_id in range(15624): + # _, time_usage = self.query_200_epoch(str(arch_id), dataset, 1) + # if time_usage > res: + # res = time_usage + # return res + return 40 + + def get_all_trained_model_ids(self): + # 201 all data has the same model set. + return list(self.data201.keys()) + + +class Gt101(metaclass=Singleton): + + @classmethod + def guess_score_time(cls): + return random.randint(1169, 1372) * 0.0001 + + def __init__(self): + self.data101_from_zerocost = read_pickle(gt101P) + self.id_to_hash_map = read_json(id_to_hash_path) + self.data101_full = read_json(gt101) + + def get_c10_test_info(self, arch_id: str, dataset: str = Config.c10, epoch_num: int = 108): + """ + Default use 108 epoch for c10, this is the largest epoch number. + :param dataset: + :param arch_id: architecture id + :param epoch_num: query the result of the specific epoch number + :return: + """ + if dataset != Config.c10: + raise "NB101 only have c10 results" + + if epoch_num is None or epoch_num > 108: + epoch_num = 108 + elif epoch_num > 36: + epoch_num = 36 + elif epoch_num > 12: + epoch_num = 12 + elif epoch_num > 4: + epoch_num = 4 + else: + epoch_num = 4 + arch_id = str(arch_id) + # this is acc from zero-cost paper, which only record 108 epoch' result [test, valid, train] + # t_acc = self.data101_from_zerocost[self.id_to_hash_map[arch_id]][0] + # this is acc from parse_testacc_101.py, + t_acc = self.data101_full[arch_id][Config.c10][str(epoch_num)]["test-accuracy"] + time_usage = self.data101_full[arch_id][Config.c10][str(epoch_num)]["time_usage"] + # print(f"[Debug]: Acc different = {t_acc_usage - t_acc}") + return t_acc, time_usage + + def count_models(self): + return len(self.data101_from_zerocost) + + def guess_train_one_epoch_time(self): + # only have information for 4 epoch + d = dict.fromkeys(self.data101_full) + keys = random.sample(list(d), 15000) + + # pick the max value over 5k arch training time + res = 0 + for rep_time in range(15000): + arch_id = keys[rep_time] + _, time_usage = self.get_c10_test_info(arch_id=arch_id, dataset=Config.c10, epoch_num=4) + if time_usage > res: + res = time_usage + return res + + def get_all_trained_model_ids(self): + return list(self.data101_full.keys()) + + +if __name__ == "__main__": + lapi = ImgScoreQueryApi(Config.NB101, Config.c10) + lapi.get_len_data() diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_mlp.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_mlp.py new file mode 100644 index 0000000000..08854c5f83 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/query_api_mlp.py @@ -0,0 +1,145 @@ +import os +from src.common.constant import Config +from src.tools.compute import generate_global_rank +from src.tools.io_tools import read_json + +base_dir = os.environ.get("base_dir") +if base_dir is None: base_dir = os.getcwd() +print("base_dir is {}".format(base_dir)) + +# todo: move all those to a config file +# MLP related ground truth +mlp_train_frappe = os.path.join(base_dir, "tab_data/frappe/all_train_baseline_frappe.json") +mlp_train_uci_diabetes = os.path.join(base_dir, "tab_data/uci_diabetes/all_train_baseline_uci_160k_40epoch.json") +mlp_train_criteo = os.path.join(base_dir, "tab_data/criteo/all_train_baseline_criteo.json") + +# score result +mlp_score_frappe = os.path.join(base_dir, "tab_data/frappe/score_frappe_batch_size_32_local_finish_all_models.json") +# mlp_score_frappe = os.path.join(base_dir, "tab_data/frappe/score_frappe_batch_size_32_nawot_synflow.json") +mlp_score_uci = os.path.join(base_dir, "tab_data/uci_diabetes/score_uci_diabetes_batch_size_32_all_metrics.json") +mlp_score_criteo = os.path.join(base_dir, "tab_data/criteo/score_criteo_batch_size_32.json") + +# 0.8028456677612497 +# todo: here is for debug expressFlow only +exp_mlp_score_frappe = os.path.join(base_dir, "score_scale_traj_width/score_mlp_sp_frappe_batch_size_32_cpu.json") +exp_mlp_score_uci = os.path.join(base_dir, "score_scale_traj_width/score_mlp_sp_uci_diabetes_batch_size_32_cpu.json") +exp_mlp_score_criteo = os.path.join(base_dir, "score_scale_traj_width/score_mlp_sp_criteo_batch_size_32_cpu.json") + +# todo here we use weigth sharing. +mlp_score_frappe_weight_share = os.path.join(base_dir, "tab_data/weight_share_nas_frappe.json") + +# pre computed result +score_one_model_time_dict = { + "cpu": { + Config.Frappe: 0.0211558125, + Config.UCIDataset: 0.015039052631578948, + Config.Criteo: 0.6824370454545454 + }, + "gpu": { + Config.Frappe: 0.013744457142857143, + Config.UCIDataset: 0.008209692307692308, + Config.Criteo: 0.6095493157894737 + } +} + +train_one_epoch_time_dict = { + "cpu": { + Config.Frappe: 5.122203075885773, + Config.UCIDataset: 4.16297769, + Config.Criteo: 422 + }, + "gpu": { + Config.Frappe: 2.8, + Config.UCIDataset: 1.4, + Config.Criteo: 125 + } +} + + +class GTMLP: + _instances = {} + # use those algoroithm => new tfmem + default_alg_name_list = ["nas_wot", "synflow"] + device = "cpu" + + def __new__(cls, dataset: str): + if dataset not in cls._instances: + instance = super(GTMLP, cls).__new__(cls) + instance.dataset = dataset + if dataset == Config.Frappe: + instance.mlp_train_path = mlp_train_frappe + instance.mlp_score_path = mlp_score_frappe + instance.mlp_score_path_expressflow = exp_mlp_score_frappe + instance.mlp_score_path_weight_share = mlp_score_frappe_weight_share + elif dataset == Config.Criteo: + instance.mlp_train_path = mlp_train_criteo + instance.mlp_score_path = mlp_score_criteo + instance.mlp_score_path_expressflow = exp_mlp_score_criteo + instance.mlp_score_path_weight_share = "./not_exist" + elif dataset == Config.UCIDataset: + instance.mlp_train_path = mlp_train_uci_diabetes + instance.mlp_score_path = mlp_score_uci + instance.mlp_score_path_expressflow = exp_mlp_score_uci + instance.mlp_score_path_weight_share = "./not_exist" + instance.mlp_train = read_json(instance.mlp_train_path) + instance.mlp_score = read_json(instance.mlp_score_path) + + # todo: here we combine two json dict, remove later + mlp_score_expressflow = read_json(instance.mlp_score_path_expressflow) + for arch_id in mlp_score_expressflow: + if arch_id in instance.mlp_score: + instance.mlp_score[arch_id].update(mlp_score_expressflow[arch_id]) + + mlp_score_weight_share = read_json(instance.mlp_score_path_weight_share) + for arch_id in mlp_score_weight_share: + if arch_id in instance.mlp_score: + instance.mlp_score[arch_id].update({"weight_share": mlp_score_weight_share[arch_id]}) + + instance.mlp_global_rank = generate_global_rank( + instance.mlp_score, instance.default_alg_name_list) + + cls._instances[dataset] = instance + return cls._instances[dataset] + + def get_all_trained_model_ids(self): + return list(self.mlp_train[self.dataset].keys()) + + def get_all_scored_model_ids(self): + return list(self.mlp_score.keys()) + + def get_score_one_model_time(self, device: str): + _train_time_per_epoch = score_one_model_time_dict[device].get(self.dataset) + if _train_time_per_epoch is None: + raise NotImplementedError + return _train_time_per_epoch + + def get_train_one_epoch_time(self, device: str): + _train_time_per_epoch = train_one_epoch_time_dict[device].get(self.dataset) + if _train_time_per_epoch is None: + raise NotImplementedError + return _train_time_per_epoch + + def get_valid_auc(self, arch_id: str, epoch_num: int): + # todo: due to the too many job contention on server, the time usage may not valid. + time_usage = (int(epoch_num) + 1) * self.get_train_one_epoch_time(self.device) + if self.dataset == Config.Frappe: + if epoch_num is None or epoch_num >= 20: epoch_num = 19 + t_acc = self.mlp_train[self.dataset][arch_id][str(epoch_num)]["valid_auc"] + return t_acc, time_usage + elif self.dataset == Config.Criteo: + if epoch_num is None or epoch_num >= 10: epoch_num = 9 + t_acc = self.mlp_train[self.dataset][arch_id][str(epoch_num)]["valid_auc"] + return t_acc, time_usage + elif self.dataset == Config.UCIDataset: + if epoch_num is None or epoch_num >= 40: epoch_num = 39 + t_acc = self.mlp_train[self.dataset][arch_id][str(epoch_num)]["valid_auc"] + return t_acc, time_usage + else: + raise NotImplementedError + + def api_get_score(self, arch_id: str) -> dict: + score_dic = self.mlp_score[arch_id] + return score_dic + + def get_global_rank_score(self, arch_id): + return self.mlp_global_rank[arch_id] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/singleton.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/singleton.py new file mode 100644 index 0000000000..eab587122e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/query_api/singleton.py @@ -0,0 +1,13 @@ +import threading + + +class Singleton(type): + _instances = {} + _lock = threading.Lock() + + def __call__(cls, *args, **kwargs): + if cls not in cls._instances: + with cls._lock: + if cls not in cls._instances: + cls._instances[cls] = super(Singleton, cls).__call__(*args, **kwargs) + return cls._instances[cls] diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/model_params.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/model_params.py new file mode 100644 index 0000000000..b3f2150e81 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/core/model_params.py @@ -0,0 +1,23 @@ +class ModelMacroCfg: + """ + Macro search space config + Search Space basic init, use bn or not, input features, output labels, etc. + """ + + def __init__(self, num_labels): + """ + Args: + num_labels: output labels. + """ + self.num_labels = num_labels + + +class ModelMicroCfg: + """ + Micro space cfg + Identifier for each model, connection patter, operations etc. + encoding = serialized(ModelMicroCfg) + """ + + def __init__(self): + pass diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/init_search_space.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/init_search_space.py new file mode 100644 index 0000000000..037ca12e79 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/init_search_space.py @@ -0,0 +1,37 @@ +import os +from src.common.constant import Config +from src.search_space.core.space import SpaceWrapper +from src.query_api.query_api_img import ImgScoreQueryApi + +def init_search_space(args) -> SpaceWrapper: + """ + :param args: + :param loapi: Local score API, records all scored arch, 101 use it to detect which arch is scored. + :return: + """ + # elif args.search_space == Config.MLPSP: + if args.search_space == Config.MLPSP: + from .mlp_api.space import MlpSpace + from .mlp_api.model_params import MlpMacroCfg + from .mlp_api.space import DEFAULT_LAYER_CHOICES_20, DEFAULT_LAYER_CHOICES_10 + print ("src/search_space/init_search_space.py config.MLPSP") + if args.hidden_choice_len == 10: + model_cfg = MlpMacroCfg( + args.nfield, + args.nfeat, + args.nemb, + args.num_layers, + args.num_labels, + DEFAULT_LAYER_CHOICES_10) + else: + model_cfg = MlpMacroCfg( + args.nfield, + args.nfeat, + args.nemb, + args.num_layers, + args.num_labels, + DEFAULT_LAYER_CHOICES_20) + + return MlpSpace(model_cfg) + else: + raise Exception diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/model_params.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/model_params.py new file mode 100644 index 0000000000..2c6b9482d9 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/model_params.py @@ -0,0 +1,16 @@ +from src.search_space.core.model_params import ModelMacroCfg + + +class MlpMacroCfg(ModelMacroCfg): + + def __init__(self, nfield: int, nfeat: int, nemb: int, + num_layers: int, + num_labels: int, + layer_choices: list): + super().__init__(num_labels) + + self.nfield = nfield + self.nfeat = nfeat + self.nemb = nemb + self.layer_choices = layer_choices + self.num_layers = num_layers diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/rl_policy.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/rl_policy.py new file mode 100644 index 0000000000..207773d056 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/rl_policy.py @@ -0,0 +1,18 @@ +from src.search_space.core.rl_policy import RLPolicyBase + + +class RLMlpSPaceTopology(RLPolicyBase): + def __init__(self, search_space, rl_learning_rate, max_nodes=4): + super().__init__() + + def generate_arch(self, config): + pass + + def select_action(self): + pass + + def _sample_new_cfg(self): + pass + + def update_policy(self, reward, baseline_values, log_prob): + pass diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/space.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/space.py new file mode 100644 index 0000000000..6668c0265d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/mlp_api/space.py @@ -0,0 +1,625 @@ +import copy +import itertools +import random +import time +from copy import deepcopy +from typing import Generator + +from src.common.constant import Config, CommonVars +from src.eva_engine import evaluator_register +from src.eva_engine.phase2.algo.trainer import ModelTrainer +from src.logger import logger +from src.search_space.core.model_params import ModelMicroCfg, ModelMacroCfg +from src.search_space.core.space import SpaceWrapper +from src.search_space.mlp_api.model_params import MlpMacroCfg +from src.query_api.interface import profile_NK_trade_off +from src.query_api.query_api_mlp import GTMLP + +from singa import layer +from singa import model +from singa import tensor +from singa import opt +from singa import device +from singa.autograd import Operator +from singa.layer import Layer +from singa import singa_wrap as singa +import argparse +import numpy as np + +# Useful constants + +DEFAULT_LAYER_CHOICES_20 = [8, 16, 24, 32, # 8 + 48, 64, 80, 96, 112, 128, 144, 160, 176, 192, 208, 224, 240, 256, # 16 + 384, 512] +DEFAULT_LAYER_CHOICES_10 = [8, 16, 32, + 48, 96, 112, 144, 176, 240, + 384] + + +np_dtype = {"float16": np.float16, "float32": np.float32} + +# singa_dtype = {"float16": tensor.float16, "float32": tensor.float32} +singa_dtype = {"float32": tensor.float32} + +class MlpMicroCfg(ModelMicroCfg): + + @classmethod + def builder(cls, encoding: str): + return MlpMicroCfg([int(ele) for ele in encoding.split("-")]) + + def __init__(self, hidden_layer_list: list): + super().__init__() + self.hidden_layer_list = hidden_layer_list + + def __str__(self): + return "-".join(str(x) for x in self.hidden_layer_list) + +#### self-defined loss begin + +### from autograd.py +class SumError(Operator): + + def __init__(self): + super(SumError, self).__init__() + # self.t = t.data + + def forward(self, x): + # self.err = singa.__sub__(x, self.t) + self.data_x = x + # print ("SumError forward x: ", x) + # print ("SumError forward x.L2(): ", x.L2()) + # print ("SumError forward x shape(): ", x.shape()) + # sqr = singa.Square(self.err) + # loss = singa.SumAll(sqr) + loss = singa.SumAll(x) + # self.n = 1 + # for s in x.shape(): + # self.n *= s + # loss /= self.n + return loss + + def backward(self, dy=1.0): + # dx = self.err + dev = device.get_default_device() + # print ("backward self.data_x.shape(): ", self.data_x.shape()) + dx = tensor.Tensor(self.data_x.shape(), dev, singa_dtype['float32']) + dx.copy_from_numpy(np.ones(self.data_x.shape(), dtype=np.float32)) + # print ("SumError backward dx data: ", dx.data) + # dx *= float(2 / self.n) + dx.data *= float(dy) + return dx.data + +def se_loss(x): + # assert x.shape == t.shape, "input and target shape different: %s, %s" % ( + # x.shape, t.shape) + return SumError()(x)[0] + +### from layer.py +class SumErrorLayer(Layer): + """ + Generate a MeanSquareError operator + """ + + def __init__(self): + super(SumErrorLayer, self).__init__() + + def forward(self, x): + return se_loss(x) + +#### self-defined loss end + +class SINGADNNModel(model.Model): + + def __init__(self, nfield: int, nfeat: int, nemb: int, + hidden_layer_list: list, dropout_rate: float, + noutput: int, use_bn: bool = True): + # def __init__(self, data_size=10, perceptron_size=100, num_classes=10, layer_hidden_list=[10,10,10,10]): + super(SINGADNNModel, self).__init__() + # self.num_classes = num_classes + self.dimension = 2 # data dimension = 2 + + self.mlp_ninput = nfield * nemb + self.nfeat = nfeat + + layer_hidden_list = [] + for index, layer_size in enumerate(hidden_layer_list): + layer_hidden_list.append(layer_size) + self.relu = layer.ReLU() + self.linear1 = layer.Linear(layer_hidden_list[0]) + # print ("linear1.in_features: ", self.linear1.in_features) + # print ("linear1.out_features: ", self.linear1.out_features) + self.linear2 = layer.Linear(layer_hidden_list[1]) + # print ("linear2.in_features: ", self.linear2.in_features) + # print ("linear2.out_features: ", self.linear2.out_features) + self.linear3 = layer.Linear(layer_hidden_list[2]) + # print ("linear3.in_features: ", self.linear3.in_features) + # print ("linear3.out_features: ", self.linear3.out_features) + self.linear4 = layer.Linear(layer_hidden_list[3]) + # print ("linear4.in_features: ", self.linear4.in_features) + # print ("linear4.out_features: ", self.linear4.out_features) + self.linear5 = layer.Linear(noutput) + # print ("linear5.in_features: ", self.linear5.in_features) + # print ("linear5.out_features: ", self.linear5.out_features) + self.softmax_cross_entropy = layer.SoftMaxCrossEntropy() + self.sum_error = SumErrorLayer() + # for weight-sharing + self.is_masked_subnet = False + self.hidden_layer_list = hidden_layer_list + # Initialize subnet mask with ones + self.subnet_mask = [np.ones(size) for size in hidden_layer_list] + + def forward(self, inputs): + # print ("in space.py forward") + # print ("in space.py inputs shape: ", inputs.shape) + y = self.linear1(inputs) + y = self.relu(y) + y = self.linear2(y) + y = self.relu(y) + y = self.linear3(y) + y = self.relu(y) + y = self.linear4(y) + y = self.relu(y) + y = self.linear5(y) + return y + + def generate_all_ones_embedding(self): + """ + Only for the MLP + Returns: + """ + import torch + # batch_data = torch.ones(1, self.mlp_ninput).double() # embedding + batch_data = torch.ones(1, self.nfeat).double() # one-hot + # print ("batch_data shape: ", batch_data.shape) + return batch_data + + def sample_subnet(self, arch_id: str, device: str): + # arch_id e.g., '128-128-128-128' + sizes = list(map(int, arch_id.split('-'))) + self.is_masked_subnet = True + # randomly mask neurons in the layers. + + for idx, size in enumerate(sizes): + # Create a mask of ones and zeros with the required length + mask = np.concatenate([ + np.ones(size), + np.zeros(self.hidden_layer_list[idx] - size)], + dim=0) + # Shuffle the mask to randomize which neurons are active + mask = mask[np.random.permutation(mask.size(0))] + self.subnet_mask[idx] = mask + + def train_one_batch(self, x, y, dist_option, spars, synflow_flag): + # print ("space.py in train_one_batch") + out = self.forward(x) + # print ("train_one_batch out shape: ", out.shape) + # print ("train_one_batch tensor.to_numpy(out): ", tensor.to_numpy(out)) + # print ("space.py train_one_batch x.shape: \n", x.shape) + # print ("train_one_batch y.data: \n", y.data) + # print ("space.py train_one_batch out.shape: \n", out.shape) + if synflow_flag: + # print ("train_one_batch sum_error") + loss = self.sum_error(out) + # print ("sum_error loss data: ", loss.data) + else: # normal training + # print ("train_one_batch softmax_cross_entropy") + loss = self.softmax_cross_entropy(out, y) + # print ("softmax_cross_entropy loss.data: ", loss.data) + # print ("train_one_batch loss.data: \n", loss.data) + + if dist_option == 'plain': + # print ("before pn_p_g_list = self.optimizer(loss)") + pn_p_g_list = self.optimizer(loss) + # print ("after pn_p_g_list = self.optimizer(loss)") + elif dist_option == 'half': + self.optimizer.backward_and_update_half(loss) + elif dist_option == 'partialUpdate': + self.optimizer.backward_and_partial_update(loss) + elif dist_option == 'sparseTopK': + self.optimizer.backward_and_sparse_update(loss, + topK=True, + spars=spars) + elif dist_option == 'sparseThreshold': + self.optimizer.backward_and_sparse_update(loss, + topK=False, + spars=spars) + # print ("len(pn_p_g_list): \n", len(pn_p_g_list)) + # print ("len(pn_p_g_list[0]): \n", len(pn_p_g_list[0])) + # print ("pn_p_g_list[0][0]: \n", pn_p_g_list[0][0]) + # print ("pn_p_g_list[0][1].data: \n", pn_p_g_list[0][1].data) + # print ("pn_p_g_list[0][2].data: \n", pn_p_g_list[0][2].data) + return pn_p_g_list, out, loss + # return pn_p_g_list[0], pn_p_g_list[1], pn_p_g_list[2], out, loss + + def set_optimizer(self, optimizer): + self.optimizer = optimizer + + +def create_model(pretrained=False, **kwargs): + """Constructs a CNN model. + + Args: + pretrained (bool): If True, returns a pre-trained model. + + Returns: + The created CNN model. + """ + model = SINGADNNModel(**kwargs) + + return model + + +__all__ = ['SINGADNNModel', 'create_model'] + +from torch.utils.data import DataLoader +class MlpSpace(SpaceWrapper): + def __init__(self, modelCfg: MlpMacroCfg): + super().__init__(modelCfg, Config.MLPSP) + + def load(self): + pass + + @classmethod + def serialize_model_encoding(cls, arch_micro: ModelMicroCfg) -> str: + assert isinstance(arch_micro, MlpMicroCfg) + return str(arch_micro) + + @classmethod + def deserialize_model_encoding(cls, model_encoding: str) -> ModelMicroCfg: + return MlpMicroCfg.builder(model_encoding) + + @classmethod + def new_arch_scratch(cls, arch_macro: ModelMacroCfg, arch_micro: ModelMicroCfg, bn: bool = True): + assert isinstance(arch_micro, MlpMicroCfg) + assert isinstance(arch_macro, MlpMacroCfg) + # mlp = DNNModel( + mlp = SINGADNNModel( + nfield=arch_macro.nfield, + nfeat=arch_macro.nfeat, + nemb=arch_macro.nemb, + hidden_layer_list=arch_micro.hidden_layer_list, + dropout_rate=0, + noutput=arch_macro.num_labels, + use_bn=bn, + ) + return mlp + + def new_arch_scratch_with_default_setting(self, model_encoding: str, bn: bool): + model_micro = MlpSpace.deserialize_model_encoding(model_encoding) + return MlpSpace.new_arch_scratch(self.model_cfg, model_micro, bn) + + def new_architecture(self, arch_id: str): + assert isinstance(self.model_cfg, MlpMacroCfg) + """ + Args: + arch_id: arch id is the same as encoding. + Returns: + """ + arch_micro = MlpSpace.deserialize_model_encoding(arch_id) + assert isinstance(arch_micro, MlpMicroCfg) + # print ("src/search_space/mlp_api/space.py new_architecture") + # print ("src/search_space/mlp_api/space.py arch_micro:\n", arch_micro) + # mlp = DNNModel( + mlp = SINGADNNModel( + nfield=self.model_cfg.nfield, + nfeat=self.model_cfg.nfeat, + nemb=self.model_cfg.nemb, + hidden_layer_list=arch_micro.hidden_layer_list, + dropout_rate=0, + noutput=self.model_cfg.num_labels) + return mlp + + def new_architecture_with_micro_cfg(self, arch_micro: ModelMicroCfg): + assert isinstance(arch_micro, MlpMicroCfg) + assert isinstance(self.model_cfg, MlpMacroCfg) + # mlp = DNNModel( + mlp = SINGADNNModel( + nfield=self.model_cfg.nfield, + nfeat=self.model_cfg.nfeat, + nemb=self.model_cfg.nemb, + hidden_layer_list=arch_micro.hidden_layer_list, + dropout_rate=0, + noutput=self.model_cfg.num_labels) + return mlp + + def profiling_score_time( + self, dataset: str, + train_loader: DataLoader = None, val_loader: DataLoader = None, + args=None, is_simulate: bool = False): + assert isinstance(self.model_cfg, MlpMacroCfg) + + device = "cpu" + if is_simulate: + gtmlp = GTMLP(dataset) + # todo, we use hybird here. + # those are from the pre-calculator + _train_time_per_epoch = gtmlp.get_score_one_model_time("cpu") + score_time = _train_time_per_epoch + else: + + # get a random batch. + import torch + batch = iter(train_loader).__next__() + target = batch['y'].type(torch.LongTensor) + batch['id'] = batch['id'].to(device) + batch['value'] = batch['value'].to(device) + target = target.to(device) + # .reshape(target.shape[0], self.model_cfg.num_labels). + + # pick the largest net to train + # super_net = DNNModel( + super_net = SINGADNNModel( + nfield=args.nfield, + nfeat=args.nfeat, + nemb=args.nemb, + hidden_layer_list=[DEFAULT_LAYER_CHOICES_20[-1]] * self.model_cfg.num_layers, + dropout_rate=0, + noutput=self.model_cfg.num_labels) + super_net.init_embedding(requires_grad=False) + super_net.to(device) + # measure score time, + score_time_begin = time.time() + naswot_score, _ = evaluator_register[CommonVars.NAS_WOT].evaluate_wrapper( + arch=super_net, + device=device, + batch_data=batch, + batch_labels=target) + + # re-init hte net + del super_net + # super_net = DNNModel( + super_net = SINGADNNModel( + nfield=args.nfield, + nfeat=args.nfeat, + nemb=args.nemb, + hidden_layer_list=[DEFAULT_LAYER_CHOICES_20[-1]] * self.model_cfg.num_layers, + dropout_rate=0, + noutput=self.model_cfg.num_labels, + use_bn=False) + super_net.init_embedding(requires_grad=False) + super_net.to(device) + synflow_score, _ = evaluator_register[CommonVars.PRUNE_SYNFLOW].evaluate_wrapper( + arch=super_net, + device=device, + batch_data=batch, + batch_labels=target) + + score_time = time.time() - score_time_begin + + # re-init hte net + del super_net + return score_time + + def profiling_train_time(self, dataset: str, + train_loader: DataLoader = None, val_loader: DataLoader = None, + args=None, is_simulate: bool = False): + + device = args.device + + if is_simulate: + gtmlp = GTMLP(dataset) + # todo, find a ideal server, and use 512 model to profile. + # those are from the pre-calculator + _train_time_per_epoch = gtmlp.get_train_one_epoch_time(device) + else: + # super_net = DNNModel( + super_net = SINGADNNModel( + nfield=args.nfield, + nfeat=args.nfeat, + nemb=args.nemb, + hidden_layer_list=[DEFAULT_LAYER_CHOICES_20[-1]] * self.model_cfg.num_layers, + dropout_rate=0, + noutput=self.model_cfg.num_labels) + super_net.init_embedding(requires_grad=True) + super_net.to(device) + # only train for ony iteratin to evaluat the time usage. + targs = copy.deepcopy(args) + valid_auc, train_time_epoch, train_log = ModelTrainer.fully_train_arch( + model=super_net, + use_test_acc=False, + epoch_num=1, + train_loader=train_loader, + val_loader=val_loader, + test_loader=val_loader, + args=targs) + del super_net + _train_time_per_epoch = train_time_epoch + + return _train_time_per_epoch + + def profiling(self, dataset: str, + train_loader: DataLoader = None, val_loader: DataLoader = None, + args=None, is_simulate: bool = False) -> (float, float, int): + + assert isinstance(self.model_cfg, MlpMacroCfg) + device = args.device + + if is_simulate: + gtmlp = GTMLP(dataset) + # todo, we use hybird here. + # those are from the pre-calculator + _train_time_per_epoch = gtmlp.get_score_one_model_time("cpu") + score_time = _train_time_per_epoch + else: + import torch + # get a random batch. + batch = iter(train_loader).__next__() + target = batch['y'].type(torch.LongTensor) + batch['id'] = batch['id'].to(device) + batch['value'] = batch['value'].to(device) + target = target.to(device) + # .reshape(target.shape[0], self.model_cfg.num_labels). + + # pick the largest net to train + # super_net = DNNModel( + super_net = SINGADNNModel( + nfield=args.nfield, + nfeat=args.nfeat, + nemb=args.nemb, + hidden_layer_list=[DEFAULT_LAYER_CHOICES_20[-1]] * self.model_cfg.num_layers, + dropout_rate=0, + noutput=self.model_cfg.num_labels) + super_net.init_embedding(requires_grad=False) + super_net.to(device) + + # measure score time, + score_time_begin = time.time() + naswot_score, _ = evaluator_register[CommonVars.NAS_WOT].evaluate_wrapper( + arch=super_net, + device=device, + batch_data=batch, + batch_labels=target) + + # re-init hte net + del super_net + # super_net = DNNModel( + super_net = SINGADNNModel( + nfield=args.nfield, + nfeat=args.nfeat, + nemb=args.nemb, + hidden_layer_list=[DEFAULT_LAYER_CHOICES_20[-1]] * self.model_cfg.num_layers, + dropout_rate=0, + noutput=self.model_cfg.num_labels, + use_bn=False) + super_net.init_embedding(requires_grad=False) + super_net.to(device) + + synflow_score, _ = evaluator_register[CommonVars.PRUNE_SYNFLOW].evaluate_wrapper( + arch=super_net, + device=device, + batch_data=batch, + batch_labels=target) + + score_time = time.time() - score_time_begin + + # re-init hte net + del super_net + + if is_simulate: + gtmlp = GTMLP(dataset) + # todo, find a ideal server, and use 512 model to profile. + # those are from the pre-calculator + _train_time_per_epoch = gtmlp.get_train_one_epoch_time(device) + else: + # super_net = DNNModel( + super_net = SINGADNNModel( + nfield=args.nfield, + nfeat=args.nfeat, + nemb=args.nemb, + hidden_layer_list=[DEFAULT_LAYER_CHOICES_20[-1]] * self.model_cfg.num_layers, + dropout_rate=0, + noutput=self.model_cfg.num_labels) + super_net.init_embedding(requires_grad=True) + super_net.to(device) + + # only train for ony iteratin to evaluat the time usage. + targs = copy.deepcopy(args) + valid_auc, train_time_epoch, train_log = ModelTrainer.fully_train_arch( + model=super_net, + use_test_acc=False, + epoch_num=1, + train_loader=train_loader, + val_loader=val_loader, + test_loader=val_loader, + args=targs) + del super_net + _train_time_per_epoch = train_time_epoch + + # todo: this is pre-defined by using img Dataset, suppose each epoch only train 200 iterations + score_time_per_model = score_time + train_time_per_epoch = _train_time_per_epoch + if args.kn_rate != -1: + n_k_ratio = args.kn_rate + else: + n_k_ratio = profile_NK_trade_off(dataset) + print(f"Profiling results: score_time_per_model={score_time_per_model}," + f" train_time_per_epoch={train_time_per_epoch}") + logger.info(f"Profiling results: score_time_per_model={score_time_per_model}," + f" train_time_per_epoch={train_time_per_epoch}") + return score_time_per_model, train_time_per_epoch, n_k_ratio + + def micro_to_id(self, arch_struct: ModelMicroCfg) -> str: + assert isinstance(arch_struct, MlpMicroCfg) + return str(arch_struct.hidden_layer_list) + + def __len__(self): + assert isinstance(self.model_cfg, MlpMacroCfg) + return len(self.model_cfg.layer_choices) ** self.model_cfg.num_layers + + def get_arch_size(self, arch_micro: ModelMicroCfg) -> int: + assert isinstance(arch_micro, MlpMicroCfg) + result = 1 + for ele in arch_micro.hidden_layer_list: + result = result * ele + return result + + def sample_all_models(self) -> Generator[str, ModelMicroCfg, None]: + assert isinstance(self.model_cfg, MlpMacroCfg) + # 2-dimensional matrix for the search spcae + space = [] + for _ in range(self.model_cfg.num_layers): + space.append(self.model_cfg.layer_choices) + + # generate all possible combinations + combinations = itertools.product(*space) + + # encoding each of them + while True: + # debug only + # yield "8-16-32-64", MlpMicroCfg([8, 16, 32, 64]) + ele = combinations.__next__() + model_micro = MlpMicroCfg(list(ele)) + model_encoding = str(model_micro) + yield model_encoding, model_micro + + def random_architecture_id(self) -> (str, ModelMicroCfg): + assert isinstance(self.model_cfg, MlpMacroCfg) + arch_encod = [] + for _ in range(self.model_cfg.num_layers): + layer_size = random.choice(self.model_cfg.layer_choices) + arch_encod.append(layer_size) + + model_micro = MlpMicroCfg(arch_encod) + # this is the model id == str(model micro) + model_encoding = str(model_micro) + return model_encoding, model_micro + + '''Below is for EA''' + + def mutate_architecture(self, parent_arch: ModelMicroCfg) -> (str, ModelMicroCfg): + assert isinstance(parent_arch, MlpMicroCfg) + assert isinstance(self.model_cfg, MlpMacroCfg) + child_layer_list = deepcopy(parent_arch.hidden_layer_list) + + # 1. choose layer index + chosen_hidden_layer_index = random.choice(list(range(len(child_layer_list)))) + + # 2. choose size of the layer index, increase the randomness + while True: + cur_layer_size = child_layer_list[chosen_hidden_layer_index] + mutated_layer_size = random.choice(self.model_cfg.layer_choices) + if mutated_layer_size != cur_layer_size: + child_layer_list[chosen_hidden_layer_index] = mutated_layer_size + new_model = MlpMicroCfg(child_layer_list) + return str(new_model), new_model + + def mutate_architecture_move_proposal(self, parent_arch: ModelMicroCfg): + assert isinstance(parent_arch, MlpMicroCfg) + assert isinstance(self.model_cfg, MlpMacroCfg) + child_layer_list = deepcopy(parent_arch.hidden_layer_list) + + all_combs = set() + # 1. choose layer index + for chosen_hidden_layer_index in list(range(len(child_layer_list))): + + # 2. choose size of the layer index, increase the randomness + while True: + cur_layer_size = child_layer_list[chosen_hidden_layer_index] + mutated_layer_size = random.choice(self.model_cfg.layer_choices) + if mutated_layer_size != cur_layer_size: + child_layer_list[chosen_hidden_layer_index] = mutated_layer_size + new_model = MlpMicroCfg(child_layer_list) + all_combs.add((str(new_model), new_model)) + break + + return list(all_combs) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/weight_initializers.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/weight_initializers.py new file mode 100644 index 0000000000..de1c544423 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/search_space/utils/weight_initializers.py @@ -0,0 +1,78 @@ +# Copyright 2021 Samsung Electronics Co., Ltd. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at + +# http://www.apache.org/licenses/LICENSE-2.0 + +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# ============================================================================= + +def init_net(net, w_type, b_type): + """ + Init network with various algorithms + :param net: + :param w_type: + :param b_type: + :return: + """ + if w_type == 'none': + pass + elif w_type == 'xavier': + net.apply(_init_weights_vs) + elif w_type == 'kaiming': + net.apply(_init_weights_he) + elif w_type == 'zero': + net.apply(_init_weights_zero) + else: + raise NotImplementedError(f'init_type={w_type} is not supported.') + + if b_type == 'none': + pass + elif b_type == 'xavier': + net.apply(_init_bias_vs) + elif b_type == 'kaiming': + net.apply(_init_bias_he) + elif b_type == 'zero': + net.apply(_init_bias_zero) + else: + raise NotImplementedError(f'init_type={b_type} is not supported.') + +import torch.nn as nn + +def _init_weights_vs(m): + if type(m) == nn.Linear or type(m) == nn.Conv2d: + nn.init.xavier_normal_(m.weight) + + +def _init_bias_vs(m): + if type(m) == nn.Linear or type(m) == nn.Conv2d: + if m.bias is not None: + nn.init.xavier_normal_(m.bias) + + +def _init_weights_he(m): + if type(m) == nn.Linear or type(m) == nn.Conv2d: + nn.init.kaiming_normal_(m.weight) + + +def _init_bias_he(m): + if type(m) == nn.Linear or type(m) == nn.Conv2d: + if m.bias is not None: + nn.init.kaiming_normal_(m.bias) + + +def _init_weights_zero(m): + if type(m) == nn.Linear or type(m) == nn.Conv2d: + m.weight.data.fill_(.0) + + +def _init_bias_zero(m): + if type(m) == nn.Linear or type(m) == nn.Conv2d: + if m.bias is not None: + m.bias.data.fill_(.0) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/genotypes.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/genotypes.py new file mode 100644 index 0000000000..6b53055cba --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/genotypes.py @@ -0,0 +1,18 @@ +from collections import namedtuple + + +Genotype = namedtuple('Genotype', 'normal normal_concat reduce reduce_concat') + +PRIMITIVES = [ + 'none', + 'max_pool_3x3', + 'avg_pool_3x3', + 'skip_connect', + 'sep_conv_3x3', + 'sep_conv_5x5', + 'dil_conv_3x3', + 'dil_conv_5x5' +] + +NUM_VERTICES = 4 +NUM_OPS = 7 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/model.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/model.py new file mode 100644 index 0000000000..02081a6b85 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/model.py @@ -0,0 +1,291 @@ + +from operations import * +from .utils import drop_path + + +class Cell(nn.Module): + + def __init__(self, genotype, C_prev_prev, C_prev, C, reduction, reduction_prev): + super(Cell, self).__init__() + # print(C_prev_prev, C_prev, C) + + if reduction_prev: + self.preprocess0 = FactorizedReduce(C_prev_prev, C) + else: + self.preprocess0 = ReLUConvBN(C_prev_prev, C, 1, 1, 0) + self.preprocess1 = ReLUConvBN(C_prev, C, 1, 1, 0) + + if reduction: + op_names, indices = zip(*genotype.reduce) + concat = genotype.reduce_concat + else: + op_names, indices = zip(*genotype.normal) + concat = genotype.normal_concat + self._compile(C, op_names, indices, concat, reduction) + + def _compile(self, C, op_names, indices, concat, reduction): + assert len(op_names) == len(indices) + self._steps = len(op_names) // 2 + self._concat = concat + self.multiplier = len(concat) + + self._ops = nn.ModuleList() + for name, index in zip(op_names, indices): + stride = 2 if reduction and index < 2 else 1 + op = OPS[name](C, stride, True) + self._ops += [op] + self._indices = indices + + def forward(self, s0, s1, drop_prob): + s0 = self.preprocess0(s0) + s1 = self.preprocess1(s1) + + states = [s0, s1] + for i in range(self._steps): + h1 = states[self._indices[2 * i]] + h2 = states[self._indices[2 * i + 1]] + op1 = self._ops[2 * i] + op2 = self._ops[2 * i + 1] + h1 = op1(h1) + h2 = op2(h2) + if self.training and drop_prob > 0.: + if not isinstance(op1, Identity): + h1 = drop_path(h1, drop_prob) + if not isinstance(op2, Identity): + h2 = drop_path(h2, drop_prob) + s = h1 + h2 + states += [s] + return torch.cat([states[i] for i in self._concat], dim=1) + + +class AuxiliaryHeadCIFAR(nn.Module): + + def __init__(self, C, num_classes): + """assuming input size 8x8""" + super(AuxiliaryHeadCIFAR, self).__init__() + self.features = nn.Sequential( + nn.ReLU(inplace=True), + # image size = 2 x 2 + nn.AvgPool2d(5, stride=3, padding=0, count_include_pad=False), + nn.Conv2d(C, 128, 1, bias=False), + nn.BatchNorm2d(128), + nn.ReLU(inplace=True), + nn.Conv2d(128, 768, 2, bias=False), + nn.BatchNorm2d(768), + nn.ReLU(inplace=True) + ) + self.classifier = nn.Linear(768, num_classes) + + def forward(self, x): + x = self.features(x) + x = self.classifier(x.view(x.size(0), -1)) + return x + + +class AuxiliaryHeadTinyImageNet(nn.Module): + + def __init__(self, C, num_classes): + """assuming input size 8x8""" + super(AuxiliaryHeadTinyImageNet, self).__init__() + self.features = nn.Sequential( + nn.ReLU(inplace=False), + # image size = 2 x 2 + nn.AvgPool2d(5, stride=3, padding=0, count_include_pad=False), + nn.Conv2d(C, 128, 1, bias=False), + nn.BatchNorm2d(128), + nn.ReLU(inplace=False), + nn.Conv2d(128, 768, 2, bias=False), + nn.BatchNorm2d(768), + nn.ReLU(inplace=False) + ) + self.classifier = nn.Linear(768, num_classes) + + def forward(self, x): + x = self.features(x) + x = self.classifier(x.view(x.size(0), -1)) + return x + + +class AuxiliaryHeadImageNet(nn.Module): + + def __init__(self, C, num_classes): + """assuming input size 14x14""" + super(AuxiliaryHeadImageNet, self).__init__() + self.features = nn.Sequential( + nn.ReLU(inplace=True), + nn.AvgPool2d(5, stride=2, padding=0, count_include_pad=False), + nn.Conv2d(C, 128, 1, bias=False), + nn.BatchNorm2d(128), + nn.ReLU(inplace=True), + nn.Conv2d(128, 768, 2, bias=False), + # NOTE: This batchnorm was omitted in my earlier implementation due to a typo. + # Commenting it out for consistency with the experiments in the paper. + # nn.BatchNorm2d(768), + nn.ReLU(inplace=True) + ) + self.classifier = nn.Linear(768, num_classes) + + def forward(self, x): + x = self.features(x) + x = self.classifier(x.view(x.size(0), -1)) + return x + + +class NetworkCIFAR(nn.Module): + + def __init__(self, C, num_classes, layers, auxiliary, genotype): + super(NetworkCIFAR, self).__init__() + self._layers = layers + self._auxiliary = auxiliary + + stem_multiplier = 3 + C_curr = stem_multiplier * C + self.stem = nn.Sequential( + nn.Conv2d(3, C_curr, 3, padding=1, bias=False), + nn.BatchNorm2d(C_curr) + ) + + C_prev_prev, C_prev, C_curr = C_curr, C_curr, C + self.cells = nn.ModuleList() + reduction_prev = False + for i in range(layers): + if i in [layers // 3, 2 * layers // 3]: + C_curr *= 2 + reduction = True + else: + reduction = False + cell = Cell(genotype, C_prev_prev, C_prev, + C_curr, reduction, reduction_prev) + reduction_prev = reduction + self.cells += [cell] + C_prev_prev, C_prev = C_prev, cell.multiplier * C_curr + if i == 2 * layers // 3: + C_to_auxiliary = C_prev + + if auxiliary: + self.auxiliary_head = AuxiliaryHeadCIFAR( + C_to_auxiliary, num_classes) + self.global_pooling = nn.AdaptiveAvgPool2d(1) + self.classifier = nn.Linear(C_prev, num_classes) + + def forward(self, input): + logits_aux = None + s0 = s1 = self.stem(input) + for i, cell in enumerate(self.cells): + s0, s1 = s1, cell(s0, s1, self.drop_path_prob) + if i == 2 * self._layers // 3: + if self._auxiliary and self.training: + logits_aux = self.auxiliary_head(s1) + out = self.global_pooling(s1) + logits = self.classifier(out.view(out.size(0), -1)) + return logits, logits_aux + + +class NetworkTinyImageNet(nn.Module): + + def __init__(self, C, num_classes, layers, auxiliary, genotype): + super(NetworkTinyImageNet, self).__init__() + self._layers = layers + self._auxiliary = auxiliary + + stem_multiplier = 3 + C_curr = stem_multiplier * C + self.stem = nn.Sequential( + nn.Conv2d(3, C_curr, 3, stride=2, padding=1, bias=False), + nn.BatchNorm2d(C_curr) + ) + + C_prev_prev, C_prev, C_curr = C_curr, C_curr, C + self.cells = nn.ModuleList() + reduction_prev = False + for i in range(layers): + if i in [layers // 3, 2 * layers // 3]: + C_curr *= 2 + reduction = True + else: + reduction = False + cell = Cell(genotype, C_prev_prev, C_prev, + C_curr, reduction, reduction_prev) + reduction_prev = reduction + self.cells += [cell] + C_prev_prev, C_prev = C_prev, cell.multiplier * C_curr + if i == 2 * layers // 3: + C_to_auxiliary = C_prev + + if auxiliary: + self.auxiliary_head = AuxiliaryHeadCIFAR( + C_to_auxiliary, num_classes) + self.global_pooling = nn.AdaptiveAvgPool2d(1) + self.classifier = nn.Linear(C_prev, num_classes) + + def forward(self, input): + logits_aux = None + s0 = s1 = self.stem(input) + for i, cell in enumerate(self.cells): + s0, s1 = s1, cell(s0, s1, self.drop_path_prob) + if i == 2 * self._layers // 3: + if self._auxiliary and self.training: + logits_aux = self.auxiliary_head(s1) + out = self.global_pooling(s1) + logits = self.classifier(out.view(out.size(0), -1)) + return logits, logits_aux + + +class NetworkImageNet(nn.Module): + + def __init__(self, C, num_classes, layers, auxiliary, genotype): + super(NetworkImageNet, self).__init__() + self._layers = layers + self._auxiliary = auxiliary + + self.stem0 = nn.Sequential( + nn.Conv2d(3, C // 2, kernel_size=3, + stride=2, padding=1, bias=False), + nn.BatchNorm2d(C // 2), + nn.ReLU(inplace=True), + nn.Conv2d(C // 2, C, 3, stride=2, padding=1, bias=False), + nn.BatchNorm2d(C), + ) + + self.stem1 = nn.Sequential( + nn.ReLU(inplace=True), + nn.Conv2d(C, C, 3, stride=2, padding=1, bias=False), + nn.BatchNorm2d(C), + ) + + C_prev_prev, C_prev, C_curr = C, C, C + + self.cells = nn.ModuleList() + reduction_prev = True + for i in range(layers): + if i in [layers // 3, 2 * layers // 3]: + C_curr *= 2 + reduction = True + else: + reduction = False + cell = Cell(genotype, C_prev_prev, C_prev, + C_curr, reduction, reduction_prev) + reduction_prev = reduction + self.cells += [cell] + C_prev_prev, C_prev = C_prev, cell.multiplier * C_curr + if i == 2 * layers // 3: + C_to_auxiliary = C_prev + + if auxiliary: + self.auxiliary_head = AuxiliaryHeadImageNet( + C_to_auxiliary, num_classes) + self.global_pooling = nn.AvgPool2d(7) + self.classifier = nn.Linear(C_prev, num_classes) + + def forward(self, input): + logits_aux = None + s0 = self.stem0(input) + s1 = self.stem1(s0) + for i, cell in enumerate(self.cells): + s0, s1 = s1, cell(s0, s1, self.drop_path_prob) + if i == 2 * self._layers // 3: + if self._auxiliary and self.training: + logits_aux = self.auxiliary_head(s1) + out = self.global_pooling(s1) + logits = self.classifier(out.view(out.size(0), -1)) + return logits, logits_aux diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/util_convert.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/util_convert.py new file mode 100644 index 0000000000..bd4f2ddf09 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/darts_lib/util_convert.py @@ -0,0 +1,109 @@ + +from scipy.special import softmax +from .genotypes import * + + +def genotype(weights, steps=4, multiplier=4): + def _parse(weights): + gene = [] + n = 2 + start = 0 + for i in range(steps): + end = start + n + W = weights[start:end].copy() + edges = sorted(range(i + 2), key=lambda x: -max( + W[x][k] for k in range(len(W[x])) if k != PRIMITIVES.index('none')))[:2] + for j in edges: + k_best = None + for k in range(len(W[j])): + if k != PRIMITIVES.index('none'): + if k_best is None or W[j][k] > W[j][k_best]: + k_best = k + gene.append((PRIMITIVES[k_best], j)) + start = end + n += 1 + return gene + + gene_normal = _parse(softmax(weights[0], axis=-1)) + gene_reduce = _parse(softmax(weights[1], axis=-1)) + + concat = range(2 + steps - multiplier, steps + 2) + genotype = Genotype( + normal=gene_normal, normal_concat=concat, + reduce=gene_reduce, reduce_concat=concat + ) + return genotype + + +# from naslib +def convert_genotype_to_compact(genotype): + """Converts Genotype to the compact representation""" + OPS = [ + "max_pool_3x3", + "avg_pool_3x3", + "skip_connect", + "sep_conv_3x3", + "sep_conv_5x5", + "dil_conv_3x3", + "dil_conv_5x5", + ] + compact = [] + + for i, cell_type in enumerate(["normal", "reduce"]): + cell = eval("genotype." + cell_type) + compact.append([]) + + for j in range(8): + compact[i].append((cell[j][1], OPS.index(cell[j][0]))) + + compact_tuple = (tuple(compact[0]), tuple(compact[1])) + return compact_tuple + + +# from naslib +def convert_compact_to_genotype(compact): + """Converts the compact representation to a Genotype""" + OPS = [ + "max_pool_3x3", + "avg_pool_3x3", + "skip_connect", + "sep_conv_3x3", + "sep_conv_5x5", + "dil_conv_3x3", + "dil_conv_5x5", + ] + genotype = [] + + for i in range(2): + cell = compact[i] + genotype.append([]) + + for j in range(8): + genotype[i].append((OPS[cell[j][1]], cell[j][0])) + + return Genotype( + normal=genotype[0], + normal_concat=[2, 3, 4, 5], + reduce=genotype[1], + reduce_concat=[2, 3, 4, 5], + ) + # TODO: need to check with Colin and/or Arber + # return Genotype( + # normal = genotype[0], + # normal_concat = [2, 3, 4, 5, 6], + # reduce = genotype[1], + # reduce_concat = [4, 5, 6] + # ) + + +# from naslib +def make_compact_mutable(compact): + # convert tuple to list so that it is mutable + arch_list = [] + for cell in compact: + arch_list.append([]) + for pair in cell: + arch_list[-1].append([]) + for num in pair: + arch_list[-1][-1].append(num) + return arch_list diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_infers/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_infers/__init__.py new file mode 100644 index 0000000000..ac1a183d96 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_infers/__init__.py @@ -0,0 +1,5 @@ +##################################################### +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2019.01 # +##################################################### +from .tiny_network import TinyNetwork +from .nasnet_cifar import NASNetonCIFAR diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/__init__.py new file mode 100644 index 0000000000..0d770cb80d --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/__init__.py @@ -0,0 +1,33 @@ +################################################## +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2019 # +################################################## +# The macro structure is defined in NAS-Bench-201 +from .search_model_darts import TinyNetworkDarts +from .search_model_gdas import TinyNetworkGDAS +from .search_model_setn import TinyNetworkSETN +from .search_model_enas import TinyNetworkENAS +from .search_model_random import TinyNetworkRANDOM +from .generic_model import GenericNAS201Model +from .genotypes import Structure as CellStructure, architectures as CellArchitectures + +# NASNet-based macro structure +from .search_model_gdas_nasnet import NASNetworkGDAS +from .search_model_gdas_frc_nasnet import NASNetworkGDAS_FRC +from .search_model_darts_nasnet import NASNetworkDARTS + + +nas201_super_nets = { + "DARTS-V1": TinyNetworkDarts, + "DARTS-V2": TinyNetworkDarts, + "GDAS": TinyNetworkGDAS, + "SETN": TinyNetworkSETN, + "ENAS": TinyNetworkENAS, + "RANDOM": TinyNetworkRANDOM, + "generic": GenericNAS201Model, +} + +nasnet_super_nets = { + "GDAS": NASNetworkGDAS, + "GDAS_FRC": NASNetworkGDAS_FRC, + "DARTS": NASNetworkDARTS, +} diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/genotypes.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/genotypes.py new file mode 100644 index 0000000000..eaf085470f --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/cell_searchs/genotypes.py @@ -0,0 +1,280 @@ +################################################## +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2019 # +################################################## +from copy import deepcopy + + +def get_combination(space, num): + combs = [] + for i in range(num): + if i == 0: + for func in space: + combs.append([(func, i)]) + else: + new_combs = [] + for string in combs: + for func in space: + xstring = string + [(func, i)] + new_combs.append(xstring) + combs = new_combs + return combs + + +class Structure: + def __init__(self, genotype): + assert isinstance(genotype, list) or isinstance( + genotype, tuple + ), "invalid class of genotype : {:}".format(type(genotype)) + self.node_num = len(genotype) + 1 + self.nodes = [] + self.node_N = [] + for idx, node_info in enumerate(genotype): + assert isinstance(node_info, list) or isinstance( + node_info, tuple + ), "invalid class of node_info : {:}".format(type(node_info)) + assert len(node_info) >= 1, "invalid length : {:}".format(len(node_info)) + for node_in in node_info: + assert isinstance(node_in, list) or isinstance( + node_in, tuple + ), "invalid class of in-node : {:}".format(type(node_in)) + assert ( + len(node_in) == 2 and node_in[1] <= idx + ), "invalid in-node : {:}".format(node_in) + self.node_N.append(len(node_info)) + self.nodes.append(tuple(deepcopy(node_info))) + + def tolist(self, remove_str): + # convert this class to the list, if remove_str is 'none', then remove the 'none' operation. + # note that we re-order the input node in this function + # return the-genotype-list and success [if unsuccess, it is not a connectivity] + genotypes = [] + for node_info in self.nodes: + node_info = list(node_info) + node_info = sorted(node_info, key=lambda x: (x[1], x[0])) + node_info = tuple(filter(lambda x: x[0] != remove_str, node_info)) + if len(node_info) == 0: + return None, False + genotypes.append(node_info) + return genotypes, True + + def node(self, index): + assert index > 0 and index <= len(self), "invalid index={:} < {:}".format( + index, len(self) + ) + return self.nodes[index] + + def tostr(self): + strings = [] + for node_info in self.nodes: + string = "|".join([x[0] + "~{:}".format(x[1]) for x in node_info]) + string = "|{:}|".format(string) + strings.append(string) + return "+".join(strings) + + def check_valid(self): + nodes = {0: True} + for i, node_info in enumerate(self.nodes): + sums = [] + for op, xin in node_info: + if op == "none" or nodes[xin] is False: + x = False + else: + x = True + sums.append(x) + nodes[i + 1] = sum(sums) > 0 + return nodes[len(self.nodes)] + + def to_unique_str(self, consider_zero=False): + # this is used to identify the isomorphic cell, which rerquires the prior knowledge of operation + # two operations are special, i.e., none and skip_connect + nodes = {0: "0"} + for i_node, node_info in enumerate(self.nodes): + cur_node = [] + for op, xin in node_info: + if consider_zero is None: + x = "(" + nodes[xin] + ")" + "@{:}".format(op) + elif consider_zero: + if op == "none" or nodes[xin] == "#": + x = "#" # zero + elif op == "skip_connect": + x = nodes[xin] + else: + x = "(" + nodes[xin] + ")" + "@{:}".format(op) + else: + if op == "skip_connect": + x = nodes[xin] + else: + x = "(" + nodes[xin] + ")" + "@{:}".format(op) + cur_node.append(x) + nodes[i_node + 1] = "+".join(sorted(cur_node)) + return nodes[len(self.nodes)] + + def check_valid_op(self, op_names): + for node_info in self.nodes: + for inode_edge in node_info: + # assert inode_edge[0] in op_names, 'invalid op-name : {:}'.format(inode_edge[0]) + if inode_edge[0] not in op_names: + return False + return True + + def __repr__(self): + return "{name}({node_num} nodes with {node_info})".format( + name=self.__class__.__name__, node_info=self.tostr(), **self.__dict__ + ) + + def __len__(self): + return len(self.nodes) + 1 + + def __getitem__(self, index): + return self.nodes[index] + + @staticmethod + def str2structure(xstr): + if isinstance(xstr, Structure): + return xstr + assert isinstance(xstr, str), "must take string (not {:}) as input".format( + type(xstr) + ) + nodestrs = xstr.split("+") + genotypes = [] + for i, node_str in enumerate(nodestrs): + inputs = list(filter(lambda x: x != "", node_str.split("|"))) + for xinput in inputs: + assert len(xinput.split("~")) == 2, "invalid input length : {:}".format( + xinput + ) + inputs = (xi.split("~") for xi in inputs) + input_infos = tuple((op, int(IDX)) for (op, IDX) in inputs) + genotypes.append(input_infos) + return Structure(genotypes) + + @staticmethod + def str2fullstructure(xstr, default_name="none"): + assert isinstance(xstr, str), "must take string (not {:}) as input".format( + type(xstr) + ) + nodestrs = xstr.split("+") + genotypes = [] + for i, node_str in enumerate(nodestrs): + inputs = list(filter(lambda x: x != "", node_str.split("|"))) + for xinput in inputs: + assert len(xinput.split("~")) == 2, "invalid input length : {:}".format( + xinput + ) + inputs = (xi.split("~") for xi in inputs) + input_infos = list((op, int(IDX)) for (op, IDX) in inputs) + all_in_nodes = list(x[1] for x in input_infos) + for j in range(i): + if j not in all_in_nodes: + input_infos.append((default_name, j)) + node_info = sorted(input_infos, key=lambda x: (x[1], x[0])) + genotypes.append(tuple(node_info)) + return Structure(genotypes) + + @staticmethod + def gen_all(search_space, num, return_ori): + assert isinstance(search_space, list) or isinstance( + search_space, tuple + ), "invalid class of search-space : {:}".format(type(search_space)) + assert ( + num >= 2 + ), "There should be at least two nodes in a neural cell instead of {:}".format( + num + ) + all_archs = get_combination(search_space, 1) + for i, arch in enumerate(all_archs): + all_archs[i] = [tuple(arch)] + + for inode in range(2, num): + cur_nodes = get_combination(search_space, inode) + new_all_archs = [] + for previous_arch in all_archs: + for cur_node in cur_nodes: + new_all_archs.append(previous_arch + [tuple(cur_node)]) + all_archs = new_all_archs + if return_ori: + return all_archs + else: + return [Structure(x) for x in all_archs] + + +class Struct101(Structure): + def __init__(self, arch_id, genotype): + super().__init__(genotype) + self.arch_id = arch_id + + +ResNet_CODE = Structure( + [ + (("nor_conv_3x3", 0),), # node-1 + (("nor_conv_3x3", 1),), # node-2 + (("skip_connect", 0), ("skip_connect", 2)), + ] # node-3 +) + +AllConv3x3_CODE = Structure( + [ + (("nor_conv_3x3", 0),), # node-1 + (("nor_conv_3x3", 0), ("nor_conv_3x3", 1)), # node-2 + (("nor_conv_3x3", 0), ("nor_conv_3x3", 1), ("nor_conv_3x3", 2)), + ] # node-3 +) + +AllFull_CODE = Structure( + [ + ( + ("skip_connect", 0), + ("nor_conv_1x1", 0), + ("nor_conv_3x3", 0), + ("avg_pool_3x3", 0), + ), # node-1 + ( + ("skip_connect", 0), + ("nor_conv_1x1", 0), + ("nor_conv_3x3", 0), + ("avg_pool_3x3", 0), + ("skip_connect", 1), + ("nor_conv_1x1", 1), + ("nor_conv_3x3", 1), + ("avg_pool_3x3", 1), + ), # node-2 + ( + ("skip_connect", 0), + ("nor_conv_1x1", 0), + ("nor_conv_3x3", 0), + ("avg_pool_3x3", 0), + ("skip_connect", 1), + ("nor_conv_1x1", 1), + ("nor_conv_3x3", 1), + ("avg_pool_3x3", 1), + ("skip_connect", 2), + ("nor_conv_1x1", 2), + ("nor_conv_3x3", 2), + ("avg_pool_3x3", 2), + ), + ] # node-3 +) + +AllConv1x1_CODE = Structure( + [ + (("nor_conv_1x1", 0),), # node-1 + (("nor_conv_1x1", 0), ("nor_conv_1x1", 1)), # node-2 + (("nor_conv_1x1", 0), ("nor_conv_1x1", 1), ("nor_conv_1x1", 2)), + ] # node-3 +) + +AllIdentity_CODE = Structure( + [ + (("skip_connect", 0),), # node-1 + (("skip_connect", 0), ("skip_connect", 1)), # node-2 + (("skip_connect", 0), ("skip_connect", 1), ("skip_connect", 2)), + ] # node-3 +) + +architectures = { + "resnet": ResNet_CODE, + "all_c3x3": AllConv3x3_CODE, + "all_c1x1": AllConv1x1_CODE, + "all_idnt": AllIdentity_CODE, + "all_full": AllFull_CODE, +} diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/__init__.py new file mode 100644 index 0000000000..9c305ffdab --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/__init__.py @@ -0,0 +1,9 @@ +##################################################### +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2019.01 # +##################################################### +from .InferCifarResNet_width import InferWidthCifarResNet +from .InferImagenetResNet import InferImagenetResNet +from .InferCifarResNet_depth import InferDepthCifarResNet +from .InferCifarResNet import InferCifarResNet +from .InferMobileNetV2 import InferMobileNetV2 +from .InferTinyCellNet import DynamicShapeTinyNet diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/shared_utils.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/shared_utils.py new file mode 100644 index 0000000000..86ab949240 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_infers/shared_utils.py @@ -0,0 +1,5 @@ +def parse_channel_info(xstring): + blocks = xstring.split(" ") + blocks = [x.split("-") for x in blocks] + blocks = [[int(_) for _ in x] for x in blocks] + return blocks diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_searchs/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_searchs/__init__.py new file mode 100644 index 0000000000..15e2260870 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/models/shape_searchs/__init__.py @@ -0,0 +1,9 @@ +################################################## +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2019 # +################################################## +from .SearchCifarResNet_width import SearchWidthCifarResNet +from .SearchCifarResNet_depth import SearchDepthCifarResNet +from .SearchCifarResNet import SearchShapeCifarResNet +from .SearchSimResNet_width import SearchWidthSimResNet +from .SearchImagenetResNet import SearchShapeImagenetResNet +from .generic_size_tiny_cell_model import GenericNAS301Model diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/__init__.py new file mode 100644 index 0000000000..fd40910d9e --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/__init__.py @@ -0,0 +1,4 @@ + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/graph_util.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/graph_util.py new file mode 100644 index 0000000000..b3e8194f0c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/graph_util.py @@ -0,0 +1,168 @@ +# Copyright 2019 The Google Research Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +"""Utility functions used by generate_graph.py.""" +from __future__ import absolute_import +from __future__ import division +from __future__ import print_function + +import hashlib +import itertools + +import numpy as np + + +def gen_is_edge_fn(bits): + """Generate a boolean function for the edge connectivity. + + Given a bitstring FEDCBA and a 4x4 matrix, the generated matrix is + [[0, A, B, D], + [0, 0, C, E], + [0, 0, 0, F], + [0, 0, 0, 0]] + + Note that this function is agnostic to the actual matrix dimension due to + order in which elements are filled out (column-major, starting from least + significant bit). For example, the same FEDCBA bitstring (0-padded) on a 5x5 + matrix is + [[0, A, B, D, 0], + [0, 0, C, E, 0], + [0, 0, 0, F, 0], + [0, 0, 0, 0, 0], + [0, 0, 0, 0, 0]] + + Args: + bits: integer which will be interpreted as a bit mask. + + Returns: + vectorized function that returns True when an edge is present. + """ + + def is_edge(x, y): + """Is there an edge from x to y (0-indexed)?""" + if x >= y: + return 0 + # Map x, y to index into bit string + index = x + (y * (y - 1) // 2) + return (bits >> index) % 2 == 1 + + return np.vectorize(is_edge) + + +def is_full_dag(matrix): + """Full DAG == all vertices on a path from vert 0 to (V-1). + + i.e. no disconnected or "hanging" vertices. + + It is sufficient to check for: + 1) no rows of 0 except for row V-1 (only output vertex has no out-edges) + 2) no cols of 0 except for col 0 (only input vertex has no in-edges) + + Args: + matrix: V x V upper-triangular adjacency matrix + + Returns: + True if the there are no dangling vertices. + """ + shape = np.shape(matrix) + + rows = matrix[:shape[0] - 1, :] == 0 + rows = np.all(rows, axis=1) # Any row with all 0 will be True + rows_bad = np.any(rows) + + cols = matrix[:, 1:] == 0 + cols = np.all(cols, axis=0) # Any col with all 0 will be True + cols_bad = np.any(cols) + + return (not rows_bad) and (not cols_bad) + + +def num_edges(matrix): + """Computes number of edges in adjacency matrix.""" + return np.sum(matrix) + + +def hash_module(matrix, labeling): + """Computes a graph-invariance MD5 hash of the matrix and label pair. + + Args: + matrix: np.ndarray square upper-triangular adjacency matrix. + labeling: list of int labels of length equal to both dimensions of + matrix. + + Returns: + MD5 hash of the matrix and labeling. + """ + vertices = np.shape(matrix)[0] + in_edges = np.sum(matrix, axis=0).tolist() + out_edges = np.sum(matrix, axis=1).tolist() + + assert len(in_edges) == len(out_edges) == len(labeling) + hashes = list(zip(out_edges, in_edges, labeling)) + hashes = [hashlib.md5(str(h).encode('utf-8')).hexdigest() for h in hashes] + # Computing this up to the diameter is probably sufficient but since the + # operation is fast, it is okay to repeat more times. + for _ in range(vertices): + new_hashes = [] + for v in range(vertices): + in_neighbors = [hashes[w] for w in range(vertices) if matrix[w, v]] + out_neighbors = [hashes[w] for w in range(vertices) if matrix[v, w]] + new_hashes.append(hashlib.md5( + (''.join(sorted(in_neighbors)) + '|' + + ''.join(sorted(out_neighbors)) + '|' + + hashes[v]).encode('utf-8')).hexdigest()) + hashes = new_hashes + fingerprint = hashlib.md5(str(sorted(hashes)).encode('utf-8')).hexdigest() + + return fingerprint + + +def permute_graph(graph, label, permutation): + """Permutes the graph and labels based on permutation. + + Args: + graph: np.ndarray adjacency matrix. + label: list of labels of same length as graph dimensions. + permutation: a permutation list of ints of same length as graph dimensions. + + Returns: + np.ndarray where vertex permutation[v] is vertex v from the original graph + """ + # vertex permutation[v] in new graph is vertex v in the old graph + forward_perm = zip(permutation, list(range(len(permutation)))) + inverse_perm = [x[1] for x in sorted(forward_perm)] + edge_fn = lambda x, y: graph[inverse_perm[x], inverse_perm[y]] == 1 + new_matrix = np.fromfunction(np.vectorize(edge_fn), + (len(label), len(label)), + dtype=np.int8) + new_label = [label[inverse_perm[i]] for i in range(len(label))] + return new_matrix, new_label + + +def is_isomorphic(graph1, graph2): + """Exhaustively checks if 2 graphs are isomorphic.""" + matrix1, label1 = np.array(graph1[0]), graph1[1] + matrix2, label2 = np.array(graph2[0]), graph2[1] + assert np.shape(matrix1) == np.shape(matrix2) + assert len(label1) == len(label2) + + vertices = np.shape(matrix1)[0] + # Note: input and output in our constrained graphs always map to themselves + # but this script does not enforce that. + for perm in itertools.permutations(range(0, vertices)): + pmatrix1, plabel1 = permute_graph(matrix1, label1, perm) + if np.array_equal(pmatrix1, matrix2) and plabel1 == label2: + return True + + return False diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/model_spec.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/model_spec.py new file mode 100644 index 0000000000..5713fb5848 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/model_spec.py @@ -0,0 +1,325 @@ +import copy +import hashlib +import itertools + +import numpy as np + +# Graphviz is optional and only required for visualization. +try: + import graphviz # pylint: disable=g-import-not-at-top +except ImportError: + pass + +INPUT = "input" +OUTPUT = "output" +CONV3X3 = "conv3x3-bn-relu" +CONV1X1 = "conv1x1-bn-relu" +MAXPOOL3X3 = "maxpool3x3" +OPS = [CONV3X3, CONV1X1, MAXPOOL3X3] + +NUM_VERTICES = 7 +OP_SPOTS = NUM_VERTICES - 2 +MAX_EDGES = 9 + + +class NASBench101ModelSpec(object): + """Model specification given adjacency matrix and labeling.""" + + def __init__(self, matrix, ops, data_format='channels_last'): + """Initialize the module spec. + + Args: + matrix: ndarray or nested list with shape [V, V] for the adjacency matrix. + ops: V-length list of labels for the base ops used. The first and last + elements are ignored because they are the input and output vertices + which have no operations. The elements are retained to keep consistent + indexing. + data_format: channels_last or channels_first. + + Raises: + ValueError: invalid matrix or ops + """ + if not isinstance(matrix, np.ndarray): + matrix = np.array(matrix) + shape = np.shape(matrix) + if len(shape) != 2 or shape[0] != shape[1]: + raise ValueError('matrix must be square') + if shape[0] != len(ops): + raise ValueError('length of ops must match matrix dimensions') + if not is_upper_triangular(matrix): + raise ValueError('matrix must be upper triangular') + + # Both the original and pruned matrices are deep copies of the matrix and + # ops so any changes to those after initialization are not recognized by the + # spec. + self.original_matrix = copy.deepcopy(matrix) + self.original_ops = copy.deepcopy(ops) + + self.matrix = copy.deepcopy(matrix) + self.ops = copy.deepcopy(ops) + self.valid_spec = True + self._prune() + + self.data_format = data_format + + def _prune(self): + """Prune the extraneous parts of the graph. + + General procedure: + 1) Remove parts of graph not connected to input. + 2) Remove parts of graph not connected to output. + 3) Reorder the vertices so that they are consecutive after steps 1 and 2. + + These 3 steps can be combined by deleting the rows and columns of the + vertices that are not reachable from both the input and output (in reverse). + """ + num_vertices = np.shape(self.original_matrix)[0] + + # DFS forward from input + visited_from_input = set([0]) + frontier = [0] + while frontier: + top = frontier.pop() + for v in range(top + 1, num_vertices): + if self.original_matrix[top, v] and v not in visited_from_input: + visited_from_input.add(v) + frontier.append(v) + + # DFS backward from output + visited_from_output = set([num_vertices - 1]) + frontier = [num_vertices - 1] + while frontier: + top = frontier.pop() + for v in range(0, top): + if self.original_matrix[v, top] and v not in visited_from_output: + visited_from_output.add(v) + frontier.append(v) + + # Any vertex that isn't connected to both input and output is extraneous to + # the computation graph. + extraneous = set(range(num_vertices)).difference( + visited_from_input.intersection(visited_from_output)) + + # If the non-extraneous graph is less than 2 vertices, the input is not + # connected to the output and the spec is invalid. + if len(extraneous) > num_vertices - 2: + self.matrix = None + self.ops = None + self.valid_spec = False + return + + self.matrix = np.delete(self.matrix, list(extraneous), axis=0) + self.matrix = np.delete(self.matrix, list(extraneous), axis=1) + for index in sorted(extraneous, reverse=True): + del self.ops[index] + + def hash_spec(self, canonical_ops): + """Computes the isomorphism-invariant graph hash of this spec. + + Args: + canonical_ops: list of operations in the canonical ordering which they + were assigned (i.e. the order provided in the config['available_ops']). + + Returns: + MD5 hash of this spec which can be used to query the dataset. + """ + # Invert the operations back to integer label indices used in graph gen. + labeling = [-1] + [canonical_ops.index(op) for op in self.ops[1:-1]] + [-2] + return hash_module(self.matrix, labeling) + + def visualize(self): + """Creates a dot graph. Can be visualized in colab directly.""" + num_vertices = np.shape(self.matrix)[0] + g = graphviz.Digraph() + g.node(str(0), 'input') + for v in range(1, num_vertices - 1): + g.node(str(v), self.ops[v]) + g.node(str(num_vertices - 1), 'output') + + for src in range(num_vertices - 1): + for dst in range(src + 1, num_vertices): + if self.matrix[src, dst]: + g.edge(str(src), str(dst)) + + return g + + @classmethod + def random_sample_one_architecture(cls, dataset_api: dict, min_size=7): + """ + This will sample a random architecture and update the edges in the + naslib object accordingly. + From the NASBench repository: + one-hot adjacency matrix + draw [0,1] for each slot in the adjacency matrix + """ + while True: + matrix = np.random.choice([0, 1], size=(NUM_VERTICES, NUM_VERTICES)) + matrix = np.triu(matrix, 1) + ops = np.random.choice(OPS, size=min_size).tolist() + ops[0] = INPUT + ops[-1] = OUTPUT + spec = dataset_api["api"].ModelSpec(matrix=matrix, ops=ops) + if not dataset_api["nb101_data"].is_valid(spec): + continue + + spec = NASBench101ModelSpec(matrix, ops) + # only sample model with 7 nodes. + if len(spec.matrix) == min_size: + break + + return spec + + +def is_upper_triangular(matrix): + """True if matrix is 0 on diagonal and below.""" + for src in range(np.shape(matrix)[0]): + for dst in range(0, src + 1): + if matrix[src, dst] != 0: + return False + + return True + + +def gen_is_edge_fn(bits): + """Generate a boolean function for the edge connectivity. + + Given a bitstring FEDCBA and a 4x4 matrix, the generated matrix is + [[0, A, B, D], + [0, 0, C, E], + [0, 0, 0, F], + [0, 0, 0, 0]] + + Note that this function is agnostic to the actual matrix dimension due to + order in which elements are filled out (column-major, starting from least + significant bit). For example, the same FEDCBA bitstring (0-padded) on a 5x5 + matrix is + [[0, A, B, D, 0], + [0, 0, C, E, 0], + [0, 0, 0, F, 0], + [0, 0, 0, 0, 0], + [0, 0, 0, 0, 0]] + + Args: + bits: integer which will be interpreted as a bit mask. + + Returns: + vectorized function that returns True when an edge is present. + """ + + def is_edge(x, y): + """Is there an edge from x to y (0-indexed)?""" + if x >= y: + return 0 + # Map x, y to index into bit string + index = x + (y * (y - 1) // 2) + return (bits >> index) % 2 == 1 + + return np.vectorize(is_edge) + + +def is_full_dag(matrix): + """Full DAG == all vertices on a path from vert 0 to (V-1). + + i.e. no disconnected or "hanging" vertices. + + It is sufficient to check for: + 1) no rows of 0 except for row V-1 (only output vertex has no out-edges) + 2) no cols of 0 except for col 0 (only input vertex has no in-edges) + + Args: + matrix: V x V upper-triangular adjacency matrix + + Returns: + True if the there are no dangling vertices. + """ + shape = np.shape(matrix) + + rows = matrix[:shape[0] - 1, :] == 0 + rows = np.all(rows, axis=1) # Any row with all 0 will be True + rows_bad = np.any(rows) + + cols = matrix[:, 1:] == 0 + cols = np.all(cols, axis=0) # Any col with all 0 will be True + cols_bad = np.any(cols) + + return (not rows_bad) and (not cols_bad) + + +def num_edges(matrix): + """Computes number of edges in adjacency matrix.""" + return np.sum(matrix) + + +def hash_module(matrix, labeling): + """Computes a graph-invariance MD5 hash of the matrix and label pair. + + Args: + matrix: np.ndarray square upper-triangular adjacency matrix. + labeling: list of int labels of length equal to both dimensions of + matrix. + + Returns: + MD5 hash of the matrix and labeling. + """ + vertices = np.shape(matrix)[0] + in_edges = np.sum(matrix, axis=0).tolist() + out_edges = np.sum(matrix, axis=1).tolist() + + assert len(in_edges) == len(out_edges) == len(labeling) + hashes = list(zip(out_edges, in_edges, labeling)) + hashes = [hashlib.md5(str(h).encode('utf-8')).hexdigest() for h in hashes] + # Computing this up to the diameter is probably sufficient but since the + # operation is fast, it is okay to repeat more times. + for _ in range(vertices): + new_hashes = [] + for v in range(vertices): + in_neighbors = [hashes[w] for w in range(vertices) if matrix[w, v]] + out_neighbors = [hashes[w] for w in range(vertices) if matrix[v, w]] + new_hashes.append(hashlib.md5( + (''.join(sorted(in_neighbors)) + '|' + + ''.join(sorted(out_neighbors)) + '|' + + hashes[v]).encode('utf-8')).hexdigest()) + hashes = new_hashes + fingerprint = hashlib.md5(str(sorted(hashes)).encode('utf-8')).hexdigest() + + return fingerprint + + +def permute_graph(graph, label, permutation): + """Permutes the graph and labels based on permutation. + + Args: + graph: np.ndarray adjacency matrix. + label: list of labels of same length as graph dimensions. + permutation: a permutation list of ints of same length as graph dimensions. + + Returns: + np.ndarray where vertex permutation[v] is vertex v from the original graph + """ + # vertex permutation[v] in new graph is vertex v in the old graph + forward_perm = zip(permutation, list(range(len(permutation)))) + inverse_perm = [x[1] for x in sorted(forward_perm)] + edge_fn = lambda x, y: graph[inverse_perm[x], inverse_perm[y]] == 1 + new_matrix = np.fromfunction(np.vectorize(edge_fn), + (len(label), len(label)), + dtype=np.int8) + new_label = [label[inverse_perm[i]] for i in range(len(label))] + return new_matrix, new_label + + +def is_isomorphic(graph1, graph2): + """Exhaustively checks if 2 graphs are isomorphic.""" + matrix1, label1 = np.array(graph1[0]), graph1[1] + matrix2, label2 = np.array(graph2[0]), graph2[1] + assert np.shape(matrix1) == np.shape(matrix2) + assert len(label1) == len(label2) + + vertices = np.shape(matrix1)[0] + # Note: input and output in our constrained graphs always map to themselves + # but this script does not enforce that. + for perm in itertools.permutations(range(0, vertices)): + pmatrix1, plabel1 = permute_graph(matrix1, label1, perm) + if np.array_equal(pmatrix1, matrix2) and plabel1 == label2: + return True + + return False diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/nb101_api.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/nb101_api.py new file mode 100644 index 0000000000..a0a09f0539 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp101_lib/nb101_api.py @@ -0,0 +1,465 @@ + + + +"""This is a NAS-Bench-101 version which is tensorflow independent. + +Before using this API, download the data files from the links in the README. + +Usage: + # Load the data from file (this will take some time) + nasbench = api.NASBench('/path/to/pickle/or/shelve') + + # Create an Inception-like module (5x5 convolution replaced with two 3x3 + # convolutions). + model_spec = api.ModelSpec( + # Adjacency matrix of the module + matrix=[[0, 1, 1, 1, 0, 1, 0], # input layer + [0, 0, 0, 0, 0, 0, 1], # 1x1 conv + [0, 0, 0, 0, 0, 0, 1], # 3x3 conv + [0, 0, 0, 0, 1, 0, 0], # 5x5 conv (replaced by two 3x3's) + [0, 0, 0, 0, 0, 0, 1], # 5x5 conv (replaced by two 3x3's) + [0, 0, 0, 0, 0, 0, 1], # 3x3 max-pool + [0, 0, 0, 0, 0, 0, 0]], # output layer + # Operations at the vertices of the module, matches order of matrix + ops=[INPUT, CONV1X1, CONV3X3, CONV3X3, CONV3X3, MAXPOOL3X3, OUTPUT]) + + + # Query this model from dataset + data = nasbench.query(model_spec) + +Adjacency matrices are expected to be upper-triangular 0-1 matrices within the +defined search space (7 vertices, 9 edges, 3 allowed ops). The first and last +operations must be 'input' and 'output'. The other operations should be from +config['available_ops']. Currently, the available operations are: + CONV3X3 = "conv3x3-bn-relu" + CONV1X1 = "conv1x1-bn-relu" + MAXPOOL3X3 = "maxpool3x3" + +When querying a spec, the spec will first be automatically pruned (removing +unused vertices and edges along with ops). If the pruned spec is still out of +the search space, an OutOfDomainError will be raised, otherwise the data is +returned. + +The returned data object is a dictionary with the following keys: + - module_adjacency: numpy array for the adjacency matrix + - module_operations: list of operation labels + - trainable_parameters: number of trainable parameters in the model + - training_time: the total training time in seconds up to this point + - train_accuracy: training accuracy + - validation_accuracy: validation_accuracy + - test_accuracy: testing accuracy + +Instead of querying the dataset for a single run of a model, it is also possible +to retrieve all metrics for a given spec, using: + + fixed_stats, computed_stats = nasbench.get_metrics_from_spec(model_spec) + +The fixed_stats is a dictionary with the keys: + - module_adjacency + - module_operations + - trainable_parameters + +The computed_stats is a dictionary from epoch count to a list of metric +dicts. For example, computed_stats[108][0] contains the metrics for the first +repeat of the provided model trained to 108 epochs. The available keys are: + - halfway_training_time + - halfway_train_accuracy + - halfway_validation_accuracy + - halfway_test_accuracy + - final_training_time + - final_train_accuracy + - final_validation_accuracy + - final_test_accuracy +""" + +from __future__ import absolute_import +from __future__ import division +from __future__ import print_function + +import copy +import random +import time +import shelve +import hashlib +import _pickle as pickle +import numpy as np + + +class OutOfDomainError(Exception): + """Indicates that the requested graph is outside of the search domain.""" + + +class NASBench(object): + """User-facing API for accessing the NASBench dataset.""" + + def __init__(self, dataset_file, seed=None, data_format='pickle'): + """Initialize dataset, this should only be done once per experiment. + + Args: + dataset_file: path to .tfrecord file containing the dataset. + seed: random seed used for sampling queried models. Two NASBench objects + created with the same seed will return the same data points when queried + with the same models in the same order. By default, the seed is randomly + generated. + """ + self.config = { + 'module_vertices': 7, + 'max_edges': 9, + 'num_repeats': 3, + 'available_ops': ['conv3x3-bn-relu', 'conv1x1-bn-relu', 'maxpool3x3'], + } + random.seed(seed) + + print('Loading dataset from file... This may take a few minutes...') + start = time.time() + + # Stores the fixed statistics that are independent of evaluation (i.e., + # adjacency matrix, operations, and number of parameters). + # hash --> metric name --> scalar + self.fixed_statistics = {} + + # Stores the statistics that are computed via training and evaluating the + # model on CIFAR-10. Statistics are computed for multiple repeats of each + # model at each max epoch length. + # hash --> epochs --> repeat index --> metric name --> scalar + self.computed_statistics = {} + + # Valid queriable epoch lengths. {4, 12, 36, 108} for the full dataset or + # {108} for the smaller dataset with only the 108 epochs. + self.valid_epochs = set() + + # open the database + if data_format == 'shelve': + with shelve.open(dataset_file, 'r') as shelf: + for module_hash in shelf: + # Parse the data from the data file. + fixed_statistics, computed_statistics = shelf[module_hash] + + self.fixed_statistics[module_hash] = fixed_statistics + self.computed_statistics[module_hash] = computed_statistics + + self.valid_epochs.update(set(computed_statistics.keys())) + elif data_format == 'pickle': + with open(dataset_file, 'rb') as f: + data = pickle.load(f) + for module_hash, stats in data.items(): + self.fixed_statistics[module_hash] = stats[0] + self.computed_statistics[module_hash] = stats[1] + + self.valid_epochs.update(set(stats[1].keys())) + else: + raise Exception('Data format not supported') + + elapsed = time.time() - start + print('Loaded dataset in %d seconds' % elapsed) + + self.history = {} + self.training_time_spent = 0.0 + self.total_epochs_spent = 0 + + def query(self, model_spec, epochs=108, stop_halfway=False): + """Fetch one of the evaluations for this model spec. + + Each call will sample one of the config['num_repeats'] evaluations of the + model. This means that repeated queries of the same model (or isomorphic + models) may return identical metrics. + + This function will increment the budget counters for benchmarking purposes. + See self.training_time_spent, and self.total_epochs_spent. + + This function also allows querying the evaluation metrics at the halfway + point of training using stop_halfway. Using this option will increment the + budget counters only up to the halfway point. + + Args: + model_spec: ModelSpec object. + epochs: number of epochs trained. Must be one of the evaluated number of + epochs, [4, 12, 36, 108] for the full dataset. + stop_halfway: if True, returned dict will only contain the training time + and accuracies at the halfway point of training (num_epochs/2). + Otherwise, returns the time and accuracies at the end of training + (num_epochs). + + Returns: + dict containing the evaluated data for this object. + + Raises: + OutOfDomainError: if model_spec or num_epochs is outside the search space. + """ + if epochs not in self.valid_epochs: + raise OutOfDomainError('invalid number of epochs, must be one of %s' + % self.valid_epochs) + + fixed_stat, computed_stat = self.get_metrics_from_spec(model_spec) + sampled_index = random.randint(0, self.config['num_repeats'] - 1) + computed_stat = computed_stat[epochs][sampled_index] + + data = {} + data['module_adjacency'] = fixed_stat['module_adjacency'] + data['module_operations'] = fixed_stat['module_operations'] + data['trainable_parameters'] = fixed_stat['trainable_parameters'] + + if stop_halfway: + data['training_time'] = computed_stat['halfway_training_time'] + data['train_accuracy'] = computed_stat['halfway_train_accuracy'] + data['validation_accuracy'] = computed_stat['halfway_validation_accuracy'] + data['test_accuracy'] = computed_stat['halfway_test_accuracy'] + else: + data['training_time'] = computed_stat['final_training_time'] + data['train_accuracy'] = computed_stat['final_train_accuracy'] + data['validation_accuracy'] = computed_stat['final_validation_accuracy'] + data['test_accuracy'] = computed_stat['final_test_accuracy'] + + self.training_time_spent += data['training_time'] + if stop_halfway: + self.total_epochs_spent += epochs // 2 + else: + self.total_epochs_spent += epochs + + return data + + def is_valid(self, model_spec): + """Checks the validity of the model_spec. + + For the purposes of benchmarking, this does not increment the budget + counters. + + Args: + model_spec: ModelSpec object. + + Returns: + True if model is within space. + """ + try: + self._check_spec(model_spec) + except OutOfDomainError: + return False + + return True + + def get_budget_counters(self): + """Returns the time and budget counters.""" + return self.training_time_spent, self.total_epochs_spent + + def reset_budget_counters(self): + """Reset the time and epoch budget counters.""" + self.training_time_spent = 0.0 + self.total_epochs_spent = 0 + + def hash_iterator(self): + """Returns iterator over all unique model hashes.""" + return self.fixed_statistics.keys() + + def get_metrics_from_hash(self, module_hash): + """Returns the metrics for all epochs and all repeats of a hash. + + This method is for dataset analysis and should not be used for benchmarking. + As such, it does not increment any of the budget counters. + + Args: + module_hash: MD5 hash, i.e., the values yielded by hash_iterator(). + + Returns: + fixed stats and computed stats of the model spec provided. + """ + fixed_stat = copy.deepcopy(self.fixed_statistics[module_hash]) + computed_stat = copy.deepcopy(self.computed_statistics[module_hash]) + return fixed_stat, computed_stat + + def get_metrics_from_spec(self, model_spec): + """Returns the metrics for all epochs and all repeats of a model. + + This method is for dataset analysis and should not be used for benchmarking. + As such, it does not increment any of the budget counters. + + Args: + model_spec: ModelSpec object. + + Returns: + fixed stats and computed stats of the model spec provided. + """ + self._check_spec(model_spec) + module_hash = self._hash_spec(model_spec) + return self.get_metrics_from_hash(module_hash) + + def _check_spec(self, model_spec): + """Checks that the model spec is within the dataset.""" + if not model_spec.valid_spec: + raise OutOfDomainError('invalid spec, provided graph is disconnected.') + + num_vertices = len(model_spec.ops) + num_edges = np.sum(model_spec.matrix) + + if num_vertices > self.config['module_vertices']: + raise OutOfDomainError('too many vertices, got %d (max vertices = %d)' + % (num_vertices, config['module_vertices'])) + + if num_edges > self.config['max_edges']: + raise OutOfDomainError('too many edges, got %d (max edges = %d)' + % (num_edges, self.config['max_edges'])) + + if model_spec.ops[0] != 'input': + raise OutOfDomainError('first operation should be \'input\'') + if model_spec.ops[-1] != 'output': + raise OutOfDomainError('last operation should be \'output\'') + for op in model_spec.ops[1:-1]: + if op not in self.config['available_ops']: + raise OutOfDomainError('unsupported op %s (available ops = %s)' + % (op, self.config['available_ops'])) + + def _hash_spec(self, model_spec): + """Returns the MD5 hash for a provided model_spec.""" + return model_spec.hash_spec(self.config['available_ops']) + + +class ModelSpec(object): + """Model specification given adjacency matrix and labeling.""" + + def __init__(self, matrix, ops, data_format='channels_last'): + """Initialize the module spec. + + Args: + matrix: ndarray or nested list with shape [V, V] for the adjacency matrix. + ops: V-length list of labels for the base ops used. The first and last + elements are ignored because they are the input and output vertices + which have no operations. The elements are retained to keep consistent + indexing. + data_format: channels_last or channels_first. + + Raises: + ValueError: invalid matrix or ops + """ + if not isinstance(matrix, np.ndarray): + matrix = np.array(matrix) + shape = np.shape(matrix) + if len(shape) != 2 or shape[0] != shape[1]: + raise ValueError('matrix must be square') + if shape[0] != len(ops): + raise ValueError('length of ops must match matrix dimensions') + if not is_upper_triangular(matrix): + raise ValueError('matrix must be upper triangular') + + # Both the original and pruned matrices are deep copies of the matrix and + # ops so any changes to those after initialization are not recognized by the + # spec. + self.original_matrix = copy.deepcopy(matrix) + self.original_ops = copy.deepcopy(ops) + + self.matrix = copy.deepcopy(matrix) + self.ops = copy.deepcopy(ops) + self.valid_spec = True + self._prune() + + self.data_format = data_format + + def _prune(self): + """Prune the extraneous parts of the graph. + + General procedure: + 1) Remove parts of graph not connected to input. + 2) Remove parts of graph not connected to output. + 3) Reorder the vertices so that they are consecutive after steps 1 and 2. + + These 3 steps can be combined by deleting the rows and columns of the + vertices that are not reachable from both the input and output (in reverse). + """ + num_vertices = np.shape(self.original_matrix)[0] + + # DFS forward from input + visited_from_input = set([0]) + frontier = [0] + while frontier: + top = frontier.pop() + for v in range(top + 1, num_vertices): + if self.original_matrix[top, v] and v not in visited_from_input: + visited_from_input.add(v) + frontier.append(v) + + # DFS backward from output + visited_from_output = set([num_vertices - 1]) + frontier = [num_vertices - 1] + while frontier: + top = frontier.pop() + for v in range(0, top): + if self.original_matrix[v, top] and v not in visited_from_output: + visited_from_output.add(v) + frontier.append(v) + + # Any vertex that isn't connected to both input and output is extraneous to + # the computation graph. + extraneous = set(range(num_vertices)).difference( + visited_from_input.intersection(visited_from_output)) + + # If the non-extraneous graph is less than 2 vertices, the input is not + # connected to the output and the spec is invalid. + if len(extraneous) > num_vertices - 2: + self.matrix = None + self.ops = None + self.valid_spec = False + return + + self.matrix = np.delete(self.matrix, list(extraneous), axis=0) + self.matrix = np.delete(self.matrix, list(extraneous), axis=1) + for index in sorted(extraneous, reverse=True): + del self.ops[index] + + def hash_spec(self, canonical_ops): + """Computes the isomorphism-invariant graph hash of this spec. + + Args: + canonical_ops: list of operations in the canonical ordering which they + were assigned (i.e. the order provided in the config['available_ops']). + + Returns: + MD5 hash of this spec which can be used to query the dataset. + """ + # Invert the operations back to integer label indices used in graph gen. + labeling = [-1] + [canonical_ops.index(op) for op in self.ops[1:-1]] + [-2] + return hash_module(self.matrix, labeling) + + +def is_upper_triangular(matrix): + """True if matrix is 0 on diagonal and below.""" + for src in range(np.shape(matrix)[0]): + for dst in range(0, src + 1): + if matrix[src, dst] != 0: + return False + + return True + + +def hash_module(matrix, labeling): + """Computes a graph-invariance MD5 hash of the matrix and label pair. + + Args: + matrix: np.ndarray square upper-triangular adjacency matrix. + labeling: list of int labels of length equal to both dimensions of + matrix. + + Returns: + MD5 hash of the matrix and labeling. + """ + vertices = np.shape(matrix)[0] + in_edges = np.sum(matrix, axis=0).tolist() + out_edges = np.sum(matrix, axis=1).tolist() + + assert len(in_edges) == len(out_edges) == len(labeling) + hashes = list(zip(out_edges, in_edges, labeling)) + hashes = [hashlib.md5(str(h).encode('utf-8')).hexdigest() for h in hashes] + # Computing this up to the diameter is probably sufficient but since the + # operation is fast, it is okay to repeat more times. + for _ in range(vertices): + new_hashes = [] + for v in range(vertices): + in_neighbors = [hashes[w] for w in range(vertices) if matrix[w, v]] + out_neighbors = [hashes[w] for w in range(vertices) if matrix[v, w]] + new_hashes.append(hashlib.md5( + (''.join(sorted(in_neighbors)) + '|' + + ''.join(sorted(out_neighbors)) + '|' + + hashes[v]).encode('utf-8')).hexdigest()) + hashes = new_hashes + fingerprint = hashlib.md5(str(sorted(hashes)).encode('utf-8')).hexdigest() + + return fingerprint + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/__init__.py new file mode 100644 index 0000000000..15d6940094 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/__init__.py @@ -0,0 +1,42 @@ +##################################################### +# Copyright (c) Xuanyi Dong [GitHub D-X-Y], 2019.08 # +##################################################### +from .api_utils import ArchResults, ResultsCount +from .api_201 import NASBench201API + +# NAS_BENCH_201_API_VERSION="v1.1" # [2020.02.25] +# NAS_BENCH_201_API_VERSION="v1.2" # [2020.03.09] +# NAS_BENCH_201_API_VERSION="v1.3" # [2020.03.16] +NAS_BENCH_201_API_VERSION="v2.0" # [2020.06.30] + + +def test_api(path): + """This is used to test the API of NAS-Bench-201.""" + api = NASBench201API(path) + num = len(api) + for i, arch_str in enumerate(api): + print ('{:5d}/{:5d} : {:}'.format(i, len(api), arch_str)) + indexes = [1, 2, 11, 301] + for index in indexes: + print('\n--- index={:} ---'.format(index)) + api.show(index) + # show the mean loss and accuracy of an architecture + info = api.query_meta_info_by_index(index) # This is an instance of `ArchResults` + res_metrics = info.get_metrics('cifar10', 'train') # This is a dict with metric names as keys + cost_metrics = info.get_compute_costs('cifar100') # This is a dict with metric names as keys, e.g., flops, params, latency + + # get the detailed information + results = api.query_by_index(index, 'cifar100') # a dict of all trials for 1st net on cifar100, where the key is the seed + print ('There are {:} trials for this architecture [{:}] on cifar100'.format(len(results), api[1])) + for seed, result in results.items(): + print ('Latency : {:}'.format(result.get_latency())) + print ('Train Info : {:}'.format(result.get_train())) + print ('Valid Info : {:}'.format(result.get_eval('x-valid'))) + print ('Test Info : {:}'.format(result.get_eval('x-test'))) + # for the metric after a specific epoch + print ('Train Info [10-th epoch] : {:}'.format(result.get_train(10))) + config = api.get_net_config(index, 'cifar10') + print ('config={:}'.format(config)) + index = api.query_index_by_arch('|nor_conv_3x3~0|+|nor_conv_3x3~0|avg_pool_3x3~1|+|skip_connect~0|nor_conv_3x3~1|skip_connect~2|') + api.show(index) + print('TEST NAS-BENCH-201 DONE.') diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2.py new file mode 100644 index 0000000000..e845b6442f --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2.py @@ -0,0 +1,117 @@ +# Copyright 2021 Samsung Electronics Co., Ltd. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at + +# http://www.apache.org/licenses/LICENSE-2.0 + +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# ============================================================================= + +from .nasbench2_ops import * + + +def gen_searchcell_mask_from_arch_str(arch_str): + nodes = arch_str.split('+') + nodes = [node[1:-1].split('|') for node in nodes] + nodes = [[op_and_input.split('~') for op_and_input in node] for node in nodes] + + keep_mask = [] + for curr_node_idx in range(len(nodes)): + for prev_node_idx in range(curr_node_idx+1): + _op = [edge[0] for edge in nodes[curr_node_idx] if int(edge[1]) == prev_node_idx] + assert len(_op) == 1, 'The arch string does not follow the assumption of 1 connection between two nodes.' + for _op_name in OPS.keys(): + keep_mask.append(_op[0] == _op_name) + return keep_mask + + +def get_model_from_arch_str(arch_str, num_classes, bn=True, init_channels=16): + keep_mask = gen_searchcell_mask_from_arch_str(arch_str) + net = NAS201Model(arch_str=arch_str, num_classes=num_classes, use_bn=bn, keep_mask=keep_mask, stem_ch=init_channels) + return net + + +def get_super_model(num_classes, use_bn=True): + net = NAS201Model(arch_str=arch_str, num_classes=num_classes, use_bn=use_bn) + return net + + +class NAS201Model(nn.Module): + + def __init__(self, arch_str, num_classes, use_bn=True, keep_mask=None, stem_ch=16): + super(NAS201Model, self).__init__() + self.arch_str=arch_str + self.num_classes=num_classes + self.use_bn= use_bn + self.stem_ch = stem_ch + + self.stem = stem(out_channels=stem_ch, use_bn=use_bn) + self.stack_cell1 = nn.Sequential(*[SearchCell(in_channels=stem_ch, out_channels=stem_ch, stride=1, affine=False, track_running_stats=False, use_bn=use_bn, keep_mask=keep_mask) for i in range(5)]) + self.reduction1 = reduction(in_channels=stem_ch, out_channels=stem_ch*2) + self.stack_cell2 = nn.Sequential(*[SearchCell(in_channels=stem_ch*2, out_channels=stem_ch*2, stride=1, affine=False, track_running_stats=False, use_bn=use_bn, keep_mask=keep_mask) for i in range(5)]) + self.reduction2 = reduction(in_channels=stem_ch*2, out_channels=stem_ch*4) + self.stack_cell3 = nn.Sequential(*[SearchCell(in_channels=stem_ch*4, out_channels=stem_ch*4, stride=1, affine=False, track_running_stats=False, use_bn=use_bn, keep_mask=keep_mask) for i in range(5)]) + self.top = top(in_dims=stem_ch*4, num_classes=num_classes, use_bn=use_bn) + + def forward(self, x): + x = self.stem(x) + + x = self.stack_cell1(x) + x = self.reduction1(x) + + x = self.stack_cell2(x) + x = self.reduction2(x) + + x = self.stack_cell3(x) + + x = self.top(x) + return x + + def get_prunable_copy(self, bn=False): + model_new = get_model_from_arch_str(self.arch_str, self.num_classes, use_bn=bn, init_channels=self.stem_ch) + + #TODO this is quite brittle and doesn't work with nn.Sequential when bn is different + # it is only required to maintain initialization -- maybe init after get_punable_copy? + model_new.load_state_dict(self.state_dict(), strict=False) + model_new.train() + + return model_new + + +def get_arch_str_from_model(net): + search_cell = net.stack_cell1[0].options + keep_mask = net.stack_cell1[0].keep_mask + num_nodes = net.stack_cell1[0].num_nodes + + nodes = [] + idx = 0 + for curr_node in range(num_nodes -1): + edges = [] + for prev_node in range(curr_node+1): # n-1 prev nodes + for _op_name in OPS.keys(): + if keep_mask[idx]: + edges.append(f'{_op_name}~{prev_node}') + idx += 1 + node_str = '|'.join(edges) + node_str = f'|{node_str}|' + nodes.append(node_str) + arch_str = '+'.join(nodes) + return arch_str + + +if __name__ == "__main__": + arch_str = '|nor_conv_3x3~0|+|none~0|none~1|+|avg_pool_3x3~0|nor_conv_3x3~1|nor_conv_3x3~2|' + + n = get_model_from_arch_str(arch_str=arch_str, num_classes=10) + print(n.stack_cell1[0]) + + arch_str2 = get_arch_str_from_model(n) + print(arch_str) + print(arch_str2) + print(f'Are the two arch strings same? {arch_str == arch_str2}') diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2_ops.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2_ops.py new file mode 100644 index 0000000000..efcdba3224 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/third_pkg/sp201_lib/nasbench2_ops.py @@ -0,0 +1,160 @@ +# Copyright 2021 Samsung Electronics Co., Ltd. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at + +# http://www.apache.org/licenses/LICENSE-2.0 + +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# ============================================================================= + +import torch.nn as nn + +class ReLUConvBN(nn.Module): + + def __init__(self, in_channels, out_channels, kernel_size, stride, padding, dilation, affine, track_running_stats=True, use_bn=True, name='ReLUConvBN'): + super(ReLUConvBN, self).__init__() + self.name = name + if use_bn: + self.op = nn.Sequential( + nn.ReLU(inplace=False), + nn.Conv2d(in_channels, out_channels, kernel_size, stride=stride, padding=padding, dilation=dilation, bias=not affine), + nn.BatchNorm2d(out_channels, affine=affine, track_running_stats=track_running_stats) + ) + else: + self.op = nn.Sequential( + nn.ReLU(inplace=False), + nn.Conv2d(in_channels, out_channels, kernel_size, stride=stride, padding=padding, dilation=dilation, bias=not affine) + ) + + def forward(self, x): + return self.op(x) + +class Identity(nn.Module): + def __init__(self, name='Identity'): + self.name = name + super(Identity, self).__init__() + + def forward(self, x): + return x + +class Zero(nn.Module): + + def __init__(self, stride, name='Zero'): + self.name = name + super(Zero, self).__init__() + self.stride = stride + + def forward(self, x): + if self.stride == 1: + return x.mul(0.) + return x[:,:,::self.stride,::self.stride].mul(0.) + +class POOLING(nn.Module): + def __init__(self, kernel_size, stride, padding, name='POOLING'): + super(POOLING, self).__init__() + self.name = name + self.avgpool = nn.AvgPool2d(kernel_size=kernel_size, stride=1, padding=1, count_include_pad=False) + + def forward(self, x): + return self.avgpool(x) + + +class reduction(nn.Module): + def __init__(self, in_channels, out_channels): + super(reduction, self).__init__() + self.residual = nn.Sequential( + nn.AvgPool2d(kernel_size=2, stride=2, padding=0), + nn.Conv2d(in_channels=in_channels, out_channels=out_channels, kernel_size=1, stride=1, padding=0, bias=False)) + + self.conv_a = ReLUConvBN(in_channels=in_channels, out_channels=out_channels, kernel_size=3, stride=2, padding=1, dilation=1, affine=True, track_running_stats=True) + self.conv_b = ReLUConvBN(in_channels=out_channels, out_channels=out_channels, kernel_size=3, stride=1, padding=1, dilation=1, affine=True, track_running_stats=True) + + def forward(self, x): + basicblock = self.conv_a(x) + basicblock = self.conv_b(basicblock) + residual = self.residual(x) + return residual + basicblock + +class stem(nn.Module): + def __init__(self, out_channels, use_bn=True): + super(stem, self).__init__() + if use_bn: + self.net = nn.Sequential( + nn.Conv2d(in_channels=3, out_channels=out_channels, kernel_size=3, padding=1, bias=False), + nn.BatchNorm2d(out_channels)) + else: + self.net = nn.Sequential( + nn.Conv2d(in_channels=3, out_channels=out_channels, kernel_size=3, padding=1, bias=False) + ) + + def forward(self, x): + return self.net(x) + +class top(nn.Module): + def __init__(self, in_dims, num_classes, use_bn=True): + super(top, self).__init__() + if use_bn: + self.lastact = nn.Sequential(nn.BatchNorm2d(in_dims), nn.ReLU(inplace=True)) + else: + self.lastact = nn.ReLU(inplace=True) + self.global_pooling = nn.AdaptiveAvgPool2d(1) + self.classifier = nn.Linear(in_dims, num_classes) + + def forward(self, x): + x = self.lastact(x) + x = self.global_pooling(x) + x = x.view(x.size(0), -1) + logits = self.classifier(x) + return logits + + +class SearchCell(nn.Module): + + def __init__(self, in_channels, out_channels, stride, affine, track_running_stats, use_bn=True, num_nodes=4, keep_mask=None): + super(SearchCell, self).__init__() + self.num_nodes = num_nodes + self.options = nn.ModuleList() + for curr_node in range(self.num_nodes-1): + for prev_node in range(curr_node+1): + for _op_name in OPS.keys(): + op = OPS[_op_name](in_channels, out_channels, stride, affine, track_running_stats, use_bn) + self.options.append(op) + + if keep_mask is not None: + self.keep_mask = keep_mask + else: + self.keep_mask = [True]*len(self.options) + + def forward(self, x): + outs = [x] + + idx = 0 + for curr_node in range(self.num_nodes-1): + edges_in = [] + for prev_node in range(curr_node+1): # n-1 prev nodes + for op_idx in range(len(OPS.keys())): + if self.keep_mask[idx]: + edges_in.append(self.options[idx](outs[prev_node])) + idx += 1 + node_output = sum(edges_in) + outs.append(node_output) + + return outs[-1] + + + +OPS = { + 'none' : lambda in_channels, out_channels, stride, affine, track_running_stats, use_bn: Zero(stride, name='none'), + 'avg_pool_3x3' : lambda in_channels, out_channels, stride, affine, track_running_stats, use_bn: POOLING(3, 1, 1, name='avg_pool_3x3'), + 'nor_conv_3x3' : lambda in_channels, out_channels, stride, affine, track_running_stats, use_bn: ReLUConvBN(in_channels, out_channels, 3, 1, 1, 1, affine, track_running_stats, use_bn, name='nor_conv_3x3'), + 'nor_conv_1x1' : lambda in_channels, out_channels, stride, affine, track_running_stats, use_bn: ReLUConvBN(in_channels, out_channels, 1, 1, 0, 1, affine, track_running_stats, use_bn, name='nor_conv_1x1'), + 'skip_connect' : lambda in_channels, out_channels, stride, affine, track_running_stats, use_bn: Identity(name='skip_connect'), +} + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/__init__.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/__init__.py new file mode 100644 index 0000000000..139597f9cb --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/__init__.py @@ -0,0 +1,2 @@ + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/compute.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/compute.py new file mode 100644 index 0000000000..98169c9851 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/compute.py @@ -0,0 +1,121 @@ + + +# for binary insert +from typing import List +import numpy as np + + +def binary_insert_get_rank(rank_list: list, new_item: List) -> int: + """ + Insert the new_item to rank_list, then get the rank of it. + :param rank_list: 0: id, 1: score + :param new_item: + :return: + """ + index = search_position(rank_list, new_item) + # search the position to insert into + rank_list.insert(index, new_item) + return index + + +# O(logN) search the position to insert into +def search_position(rank_list_m: list, new_item: List): + if len(rank_list_m) == 0: + return 0 + left = 0 + right = len(rank_list_m) - 1 + while left + 1 < right: + mid = int((left + right) / 2) + if rank_list_m[mid][1] <= new_item[1]: + left = mid + else: + right = mid + + # consider the time. + if rank_list_m[right][1] <= new_item[1]: + return right + 1 + elif rank_list_m[left][1] <= new_item[1]: + return left + 1 + else: + return left + + +def generate_global_rank(ml_data_score_dic: dict, alg_name_list: List) -> dict: + """ + ml_data_score_dic: { model_id: {alg: score1, alg2: score2} } + return: { model_id: {alg1_alg2: rank_score} } + """ + + history = {} + for alg in alg_name_list: + history[alg] = [] + + for arch_id, arch_score in ml_data_score_dic.items(): + # add model and score to local list + for alg, score in arch_score.items(): + if alg in alg_name_list: + binary_insert_get_rank(history[alg], [str(arch_id), float(score)]) + + # convert multiple scores into rank value + model_new_rank_score = {} + current_explored_models = 0 + for alg in alg_name_list: + current_explored_models = len(history[alg]) + for rank_index in range(len(history[alg])): + ms_ins = history[alg][rank_index] + # rank = index + 1, since index can be 0 + if ms_ins[0] in model_new_rank_score: + model_new_rank_score[ms_ins[0]] += rank_index + 1 + else: + model_new_rank_score[ms_ins[0]] = rank_index + 1 + + for ele in model_new_rank_score.keys(): + model_new_rank_score[ele] = \ + {"_".join(list(alg_name_list)): model_new_rank_score[ele] / current_explored_models} + + return model_new_rank_score + + +def log_scale_x_array(num_points, max_minute, base=10) -> list: + """ + return a list of mins in log scale distance. + """ + # Set the minimum and maximum values for the log scale + min_val = 1 # 1 second + max_val = max_minute * 60 # 1440 minutes converted to seconds + + # Generate the log scale values + log_vals = np.logspace(np.log10(min_val), np.log10(max_val), num=num_points, base=base) + + # Convert the log scale values to minutes + log_vals_min = log_vals / 60 + + # Print the log scale values in minutes + + return log_vals_min.tolist() + + +def sample_in_log_scale(lst: List, num_points: int) -> List: + indices = np.logspace(0, np.log10(len(lst) - 1), num_points + num_points // 2, dtype=int) + # Remove any duplicate indices + indices = np.unique(indices) + return list(indices) + + +def sample_in_log_scale_new(lstM: List, num_points: int) -> List: + lst = np.array(lstM) + # Create an evenly spaced array in the log scale domain + evenly_spaced_log_x = np.linspace(np.log10(lst.min()), np.log10(lst.max()), num_points) + # Convert the new array back to the original scale + evenly_spaced_x = 10 ** evenly_spaced_log_x + # Find the indices of the sampled points in the original x-array + indices = [np.abs(lst - point).argmin() for point in evenly_spaced_x] + return indices + + +def sample_in_line_scale(lst: List, num_points: int) -> List: + indices = np.linspace(0, len(lst) - 1, num_points, dtype=int) + # Remove any duplicate indices + indices = np.unique(indices) + return list(indices) + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/correlation.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/correlation.py new file mode 100644 index 0000000000..038a593875 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/correlation.py @@ -0,0 +1,115 @@ +from scipy import stats +from src.common.constant import CommonVars +import numpy as np +from src.logger import logger +from sklearn import metrics + + +class CorCoefficient: + + @staticmethod + def measure(x1: list, x2: list, measure_metrics: str = CommonVars.AllCorrelation) -> dict: + """ + Measure the correlation coefficient between x1 and x2 + It requires that each dataset be normally distributed. + :param x1: list1 + :param x2: list2 + :param measure_metrics: str + :return: correlation, + Like other correlation coefficients, this one varies between -1 and +1 with 0 implying no correlation. + Correlations of -1 or +1 imply an exact linear relationship + """ + + result = {} + if measure_metrics == CommonVars.KendallTau: + correlation, p_value = stats.kendalltau(x1, x2, nan_policy='omit') + result[CommonVars.KendallTau] = correlation + elif measure_metrics == CommonVars.Spearman: + correlation, p_value = stats.spearmanr(x1, x2, nan_policy='omit') + result[CommonVars.Spearman] = correlation + elif measure_metrics == CommonVars.Pearson: + correlation, p_value = stats.pearsonr(x1, x2) + result[CommonVars.Pearson] = correlation + elif measure_metrics == CommonVars.AvgCorrelation: + # calculate average over all + correlation1, p_value = stats.kendalltau(x1, x2, nan_policy='omit') + correlation2, p_value = stats.spearmanr(x1, x2, nan_policy='omit') + correlation3, p_value = stats.pearsonr(x1, x2) + correlation = (correlation1 + correlation2 + correlation3) / 3 + result[CommonVars.AvgCorrelation] = correlation + elif measure_metrics == CommonVars.AllCorrelation: + correlation1, p_value = stats.kendalltau(x1, x2, nan_policy='omit') + correlation2, p_value = stats.spearmanr(x1, x2, nan_policy='omit') + correlation3, p_value = stats.pearsonr(x1, x2) + correlation4 = (correlation1 + correlation2 + correlation3) / 3 + result[CommonVars.KendallTau] = correlation1 + result[CommonVars.Spearman] = correlation2 + result[CommonVars.Pearson] = correlation3 + result[CommonVars.AvgCorrelation] = correlation4 + else: + raise NotImplementedError(measure_metrics + " is not implemented") + + return result + + @staticmethod + def compare(ytest, test_pred): + ytest = np.array(ytest) + test_pred = np.array(test_pred) + METRICS = [ + "mae", + "rmse", + "pearson", + "spearman", + "kendalltau", + "kt_2dec", + "kt_1dec", + "precision_10", + "precision_20", + "full_ytest", + "full_testpred", + ] + metrics_dict = {} + + try: + metrics_dict["mae"] = np.mean(abs(test_pred - ytest)) + metrics_dict["rmse"] = metrics.mean_squared_error( + ytest, test_pred, squared=False + ) + metrics_dict["pearson"] = np.abs(np.corrcoef(ytest, test_pred)[1, 0]) + metrics_dict["spearman"] = stats.spearmanr(ytest, test_pred)[0] + metrics_dict["kendalltau"] = stats.kendalltau(ytest, test_pred)[0] + metrics_dict["kt_2dec"] = stats.kendalltau( + ytest, np.round(test_pred, decimals=2) + )[0] + metrics_dict["kt_1dec"] = stats.kendalltau( + ytest, np.round(test_pred, decimals=1) + )[0] + print("ytest = ", ytest) + print("test_pred = ", test_pred) + for k in [10, 20]: + top_ytest = np.array( + [y > sorted(ytest)[max(-len(ytest), -k - 1)] for y in ytest] + ) + top_test_pred = np.array( + [ + y > sorted(test_pred)[max(-len(test_pred), -k - 1)] + for y in test_pred + ] + ) + metrics_dict["precision_{}".format(k)] = ( + sum(top_ytest & top_test_pred) / k + ) + metrics_dict["full_ytest"] = ytest.tolist() + metrics_dict["full_testpred"] = test_pred.tolist() + + except: + for metric in METRICS: + metrics_dict[metric] = float("nan") + if np.isnan(metrics_dict["pearson"]) or not np.isfinite( + metrics_dict["pearson"] + ): + logger.info("Error when computing metrics. ytest and test_pred are:") + logger.info(ytest) + logger.info(test_pred) + + return metrics_dict diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/io_tools.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/io_tools.py new file mode 100644 index 0000000000..09cb8d9e64 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/io_tools.py @@ -0,0 +1,42 @@ +import json +import pickle +import os.path + + +def read_json(file_name): + print(f"Loading {file_name}...") + is_exist = os.path.exists(file_name) + if is_exist: + with open(file_name, 'r') as readfile: + data = json.load(readfile) + return data + else: + print(f"{file_name} is not exist") + return {} + + +def write_json(file_name, data): + print(f"writting {file_name}...") + with open(file_name, 'w') as outfile: + outfile.write(json.dumps(data)) + + +def read_pickle(file_name): + print(f"Loading pickel {file_name}...") + with open(file_name, 'rb') as f: + data = pickle.load(f) + return data + + +def write_pickle(file_name, data): + print(f"writing pickle {file_name}...") + with open('filename.pickle', 'wb') as handle: + pickle.dump(data, handle, protocol=pickle.HIGHEST_PROTOCOL) + + +if __name__ == "__main__": + a = {1:1} + write_json("./asdf.json", a) + b = {2:2323} + write_json("./asdf.json", b) + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/res_measure.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/res_measure.py new file mode 100644 index 0000000000..ffec80cd4c --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/res_measure.py @@ -0,0 +1,84 @@ +import os +import psutil +import gpustat +import threading +import time +from src.tools.io_tools import write_json +import sys + +def print_cpu_gpu_usage(interval=1, output_file="path_to_folder", stop_event=None): + def print_usage(): + print("Starting to print usage") # Debugging print + # Get current process + main_process = psutil.Process(os.getpid()) + + # Create an empty dictionary to store metrics + metrics = {'cpu_usage': [], 'memory_usage': [], 'gpu_usage': []} + + while not stop_event.is_set(): + cpu_percent = 0 + mem_usage_mb = 0 + main_process.cpu_percent() + for process in main_process.children(recursive=True): # Include all child processes + try: + cpu_percent += process.cpu_percent() + mem_usage_mb += process.memory_info().rss / (1024 ** 2) + except psutil.NoSuchProcess: + # Process does not exist, so add 0 to cpu_percent and mem_usage_mb + pass + cpu_percent += main_process.cpu_percent() + mem_usage_mb += main_process.memory_info().rss / (1024 ** 2) + + metrics['cpu_usage'].append(cpu_percent) + metrics['memory_usage'].append(mem_usage_mb) + + try: + gpu_stats = gpustat.GPUStatCollection.new_query() + for gpu in gpu_stats: + metrics['gpu_usage'].append((gpu.index, gpu.utilization, gpu.memory_used)) + except Exception as e: + pass + # print(f"Exception encountered when fetching GPU stats: {e}") + + # If it's time to write metrics to a file, do so + if len(metrics['cpu_usage']) % 40 == 0: + write_json(output_file, metrics) + + time.sleep(interval) + + print("Stop monitering, flust to disk") + write_json(output_file, metrics) + + stop_event = stop_event or threading.Event() + thread = threading.Thread(target=print_usage) + thread.start() + return stop_event, thread + +def get_variable_memory_size(obj): + # If it's a PyTorch tensor and on the GPU + import torch + if torch.is_tensor(obj) and obj.is_cuda: + return obj.element_size() * obj.nelement() + else: + return sys.getsizeof(obj) + +def print_memory_usage(): + # Get current process + main_process = psutil.Process(os.getpid()) + # Create an empty dictionary to store metrics + metrics = {'cpu_usage': [], 'memory_usage': []} + cpu_percent = 0 + mem_usage_mb = 0 + main_process.cpu_percent() + for process in main_process.children(recursive=True): # Include all child processes + try: + cpu_percent += process.cpu_percent() + mem_usage_mb += process.memory_info().rss / (1024 ** 2) + except psutil.NoSuchProcess: + # Process does not exist, so add 0 to cpu_percent and mem_usage_mb + pass + cpu_percent += main_process.cpu_percent() + mem_usage_mb += main_process.memory_info().rss / (1024 ** 2) + metrics['cpu_usage'].append(cpu_percent) + metrics['memory_usage'].append(mem_usage_mb) + print(metrics) diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/utils.py b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/utils.py new file mode 100644 index 0000000000..70acf26a79 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_selection/src/tools/utils.py @@ -0,0 +1,454 @@ +import math +import os +import random +import sys +import time +import warnings + +import numpy +import numpy as np +import shutil +import logging + +warnings.filterwarnings("error") + + +def timeSince(since=None, s=None): + if s is None: + s = int(time.time() - since) + m = math.floor(s / 60) + s %= 60 + h = math.floor(m / 60) + m %= 60 + return '%dh %dm %ds' % (h, m, s) + + +class AvgrageMeter(object): + """Computes and stores the average and current value""" + + def __init__(self): + self.reset() + + def reset(self): + self.val = 0 + self.avg = 0 + self.sum = 0 + self.count = 0 + + def update(self, val, n=1): + self.val = val + self.sum += val * n + self.count += n + self.avg = self.sum / self.count + +import torch +def get_correct_num(y, target): + pred_label = torch.argmax(y, dim=1) + return (target == pred_label).sum().item() + + +def accuracy(output, target, topk=(1,)): + maxk = max(topk) + batch_size = target.size(0) + + _, pred = output.topk(maxk, 1, True, True) + pred = pred.t() + correct = pred.eq(target.view(1, -1).expand_as(pred)) + + res = [] + for k in topk: + correct_k = correct[:k].view(-1).float().sum(0) + res.append(correct_k.mul_(100.0 / batch_size)) + return res + + +class Cutout(object): + def __init__(self, length): + self.length = length + + def __call__(self, img): + h, w = img.size(1), img.size(2) + mask = np.ones((h, w), np.float32) + y = np.random.randint(h) + x = np.random.randint(w) + + y1 = np.clip(y - self.length // 2, 0, h) + y2 = np.clip(y + self.length // 2, 0, h) + x1 = np.clip(x - self.length // 2, 0, w) + x2 = np.clip(x + self.length // 2, 0, w) + + mask[y1:y2, x1:x2] = 0. + mask = torch.from_numpy(mask) + mask = mask.expand_as(img) + img *= mask + return img + +import torchvision.transforms as transforms +def _data_transforms_cifar10(args): + CIFAR_MEAN = [0.49139968, 0.48215827, 0.44653124] + CIFAR_STD = [0.24703233, 0.24348505, 0.26158768] + + train_transform = transforms.Compose([ + transforms.RandomCrop(32, padding=4), + transforms.RandomHorizontalFlip(), + transforms.ToTensor(), + transforms.Normalize(CIFAR_MEAN, CIFAR_STD), + ]) + if args.cutout: + train_transform.transforms.append(Cutout(args.cutout_length)) + + valid_transform = transforms.Compose([ + transforms.ToTensor(), + transforms.Normalize(CIFAR_MEAN, CIFAR_STD), + ]) + return train_transform, valid_transform + +import torchvision.datasets as dset +def _get_cifar10(args): + train_transform, valid_transform = _data_transforms_cifar10(args) + train_data = dset.CIFAR10( + root=args.data, train=True, download=True, transform=train_transform + ) + valid_data = dset.CIFAR10( + root=args.data, train=False, download=True, transform=valid_transform + ) + + train_queue = torch.utils.data.DataLoader( + train_data, + batch_size=args.batch_size, + shuffle=True, + pin_memory=True, + num_workers=4, + ) + + valid_queue = torch.utils.data.DataLoader( + valid_data, + batch_size=args.batch_size, + shuffle=False, + pin_memory=True, + num_workers=4, + ) + return train_queue, valid_queue + + +def _get_dist_cifar10(args): + train_transform, valid_transform = _data_transforms_cifar10(args) + train_data = dset.CIFAR10( + root=args.data, train=True, download=True, transform=train_transform + ) + valid_data = dset.CIFAR10( + root=args.data, train=False, download=True, transform=valid_transform + ) + + sampler = torch.utils.data.distributed.DistributedSampler( + train_data, num_replicas=args.gpu_num, rank=args.local_rank) + + train_queue = torch.utils.data.DataLoader( + train_data, + batch_size=args.batch_size // args.gpu_num, + pin_memory=True, + num_workers=4, + drop_last=True, + sampler=sampler + ) + + valid_queue = torch.utils.data.DataLoader( + valid_data, + batch_size=args.batch_size, + shuffle=False, + pin_memory=True, + num_workers=4, + ) + return train_queue, valid_queue, sampler + + +def _get_dist_imagenet(args): + traindir = os.path.join(args.data_dir, 'train') + valdir = os.path.join(args.data_dir, 'val') + normalize = transforms.Normalize(mean=[0.485, 0.456, 0.406], + std=[0.229, 0.224, 0.225]) + + train_dataset = dset.ImageFolder( + traindir, + transforms.Compose([ + transforms.RandomResizedCrop(224), + transforms.RandomHorizontalFlip(), + transforms.ColorJitter( + brightness=0.4, + contrast=0.4, + saturation=0.4, + hue=0.2), + transforms.ToTensor(), + normalize, + ])) + + sampler = torch.utils.data.distributed.DistributedSampler( + train_dataset, num_replicas=args.gpu_num, rank=args.local_rank) + + train_loader = torch.utils.data.DataLoader( + train_dataset, batch_size=args.batch_size // args.gpu_num, num_workers=max(args.gpu_num * 2, 4), + pin_memory=True, drop_last=True, sampler=sampler) + + val_loader = torch.utils.data.DataLoader( + dset.ImageFolder(valdir, transforms.Compose([ + transforms.Resize(256), + transforms.CenterCrop(224), + transforms.ToTensor(), + normalize, + ])), + batch_size=args.batch_size, shuffle=False, + num_workers=4, pin_memory=True) + + return train_loader, val_loader, sampler + + +def _data_transforms_cifar100(args): + CIFAR_MEAN = [0.5070751592371323, 0.48654887331495095, 0.4409178433670343] + CIFAR_STD = [0.2673342858792401, 0.2564384629170883, 0.27615047132568404] + + train_transform = transforms.Compose([ + transforms.RandomCrop(32, padding=4), + transforms.RandomHorizontalFlip(), + transforms.ToTensor(), + transforms.Normalize(CIFAR_MEAN, CIFAR_STD), + ]) + if args.cutout: + train_transform.transforms.append(Cutout(args.cutout_length)) + + valid_transform = transforms.Compose([ + transforms.ToTensor(), + transforms.Normalize(CIFAR_MEAN, CIFAR_STD), + ]) + return train_transform, valid_transform + + +def _get_cifar100(args): + train_transform, valid_transform = _data_transforms_cifar100(args) + train_data = dset.CIFAR100( + root=args.data, train=True, download=True, transform=train_transform + ) + valid_data = dset.CIFAR100( + root=args.data, train=False, download=True, transform=valid_transform + ) + + train_queue = torch.utils.data.DataLoader( + train_data, + batch_size=args.batch_size, + shuffle=True, + pin_memory=True, + num_workers=4, + ) + + valid_queue = torch.utils.data.DataLoader( + valid_data, + batch_size=args.batch_size, + shuffle=False, + pin_memory=True, + num_workers=4, + ) + return train_queue, valid_queue + + +def _get_imagenet_tiny(args): + traindir = os.path.join(args.data, 'train') + validdir = os.path.join(args.data, 'val') + normalize = transforms.Normalize( + mean=[0.4802, 0.4481, 0.3975], + std=[0.2302, 0.2265, 0.2262] + ) + train_transform = transforms.Compose([ + transforms.RandomCrop(64, padding=4), + transforms.RandomHorizontalFlip(), + transforms.ToTensor(), + normalize, + ]) + if args.cutout: + train_transform.transforms.append(Cutout(args.cutout_length)) + + train_data = dset.ImageFolder( + traindir, + train_transform + ) + valid_data = dset.ImageFolder( + validdir, + transforms.Compose([ + transforms.ToTensor(), + normalize, + ]) + ) + + train_queue = torch.utils.data.DataLoader( + train_data, batch_size=args.batch_size, shuffle=True, pin_memory=True, num_workers=4) + + valid_queue = torch.utils.data.DataLoader( + valid_data, batch_size=args.batch_size // 2, shuffle=False, pin_memory=True, num_workers=4) + return train_queue, valid_queue + + +def count_parameters_in_MB(model): + return np.sum([np.prod(v.size()) for v in model.parameters()]) / 1e6 + + +def count_parameters(model): + """ + Get element number of all parameters matrix. + :param model: + :return: + """ + return sum([torch.numel(v) for v in model.parameters()]) + + +def save(model, model_path): + torch.save(model.state_dict(), model_path) + + +def load(model, model_path): + model.load_state_dict(torch.load(model_path)) + + +def load_ckpt(ckpt_path): + print(f'=> loading checkpoint {ckpt_path}...') + try: + checkpoint = torch.load(ckpt_path) + except: + print(f"=> fail loading {ckpt_path}..."); + exit() + return checkpoint + + +def save_ckpt(ckpt, file_dir, file_name='model.ckpt', is_best=False): + if not os.path.exists(file_dir): os.makedirs(file_dir) + ckpt_path = os.path.join(file_dir, file_name) + torch.save(ckpt, ckpt_path) + if is_best: shutil.copyfile(ckpt_path, os.path.join(file_dir, f'best_{file_name}')) + + +def drop_path(x, drop_prob, dims=(0,)): + from torch.autograd import Variable + var_size = [1 for _ in range(x.dim())] + for i in dims: + var_size[i] = x.size(i) + if drop_prob > 0.: + keep_prob = 1. - drop_prob + mask = Variable(torch.cuda.FloatTensor(*var_size).bernoulli_(keep_prob)) + x.div_(keep_prob) + x.mul_(mask) + return x + + +def create_exp_dir(path, scripts_to_save=None): + if not os.path.exists(path): + os.makedirs(path) + print('Experiment dir : {}'.format(path)) + + if scripts_to_save is not None: + os.makedirs(os.path.join(path, 'tools')) + for script in scripts_to_save: + dst_file = os.path.join(path, 'tools', os.path.basename(script)) + shutil.copyfile(script, dst_file) + + +class Performance(object): + def __init__(self, path): + self.path = path + self.data = None + + def update(self, alphas_normal, alphas_reduce, val_loss): + import torch.nn.functional as F + a_normal = F.softmax(alphas_normal, dim=-1) + # print("alpha normal size: ", a_normal.data.size()) + a_reduce = F.softmax(alphas_reduce, dim=-1) + # print("alpha reduce size: ", a_reduce.data.size()) + data = np.concatenate([a_normal.data.view(-1), + a_reduce.data.view(-1), + np.array([val_loss.data])]).reshape(1, -1) + if self.data is not None: + self.data = np.concatenate([self.data, data], axis=0) + else: + self.data = data + + def save(self): + np.save(self.path, self.data) + + +def logger(log_dir, need_time=True, need_stdout=False): + log = logging.getLogger(__name__) + log.setLevel(logging.DEBUG) + fh = logging.FileHandler(log_dir) + fh.setLevel(logging.DEBUG) + formatter = logging.Formatter(fmt='%(asctime)s %(message)s', datefmt='%m/%d/%Y-%I:%M:%S') + if need_stdout: + ch = logging.StreamHandler(sys.stdout) + ch.setLevel(logging.DEBUG) + log.addHandler(ch) + if need_time: + fh.setFormatter(formatter) + if need_stdout: + ch.setFormatter(formatter) + log.addHandler(fh) + return log + +import torch.nn as nn +class CrossEntropyLabelSmooth(nn.Module): + + def __init__(self, num_classes, epsilon): + super(CrossEntropyLabelSmooth, self).__init__() + self.num_classes = num_classes + self.epsilon = epsilon + self.logsoftmax = nn.LogSoftmax(dim=1) + + def forward(self, inputs, targets): + log_probs = self.logsoftmax(inputs) + targets = torch.zeros_like(log_probs).scatter_(1, targets.unsqueeze(1), 1) + targets = (1 - self.epsilon) * targets + self.epsilon / self.num_classes + loss = (-targets * log_probs).mean(0).sum() + return loss + + +def roc_auc_compute_fn(y_pred, y_target): + """ IGNITE.CONTRIB.METRICS.ROC_AUC """ + try: + from sklearn.metrics import roc_auc_score + except ImportError: + raise RuntimeError("This contrib module requires sklearn to be installed.") + + if y_pred.requires_grad: + y_pred = y_pred.detach() + + if y_target.is_cuda: + y_target = y_target.cpu() + if y_pred.is_cuda: + y_pred = y_pred.cpu() + + y_true = y_target.numpy() + y_pred = y_pred.numpy() + try: + return roc_auc_score(y_true, y_pred) + except ValueError: + # print('ValueError: Only one class present in y_true. ROC AUC score is not defined in that case.') + return 0. + + +def load_checkpoint(args): + try: + return torch.load(args.resume) + except RuntimeError: + raise RuntimeError(f"Fail to load checkpoint at {args.resume}") + + +def save_checkpoint(ckpt, is_best, file_dir, file_name='model.ckpt'): + if not os.path.exists(file_dir): + os.makedirs(file_dir) + ckpt_name = "{0}{1}".format(file_dir, file_name) + torch.save(ckpt, ckpt_name) + if is_best: shutil.copyfile(ckpt_name, "{0}{1}".format(file_dir, 'best_' + file_name)) + + +def seed_everything(seed=2022): + ''' [reference] https://gist.github.com/KirillVladimirov/005ec7f762293d2321385580d3dbe335 ''' + random.seed(seed) + os.environ['PYTHONHASHSEED'] = str(seed) + np.random.seed(seed) + torch.manual_seed(seed) + torch.cuda.manual_seed(seed) + torch.backends.cudnn.deterministic = True diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_slicing/.gitkeep b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/ml/model_slicing/.gitkeep new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.cargo/config.toml b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.cargo/config.toml new file mode 100644 index 0000000000..9dd8fcf873 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.cargo/config.toml @@ -0,0 +1,3 @@ +[target.'cfg(target_os="macos")'] +# Postgres symbols won't be available until runtime +rustflags = ["-Clink-arg=-Wl,-undefined,dynamic_lookup"] \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.gitignore b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.gitignore new file mode 100644 index 0000000000..3906c33241 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/.gitignore @@ -0,0 +1,6 @@ +.DS_Store +.idea/ +/target +*.iml +**/*.rs.bk +Cargo.lock diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/Cargo.toml b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/Cargo.toml new file mode 100644 index 0000000000..5c3e747391 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/Cargo.toml @@ -0,0 +1,39 @@ +[package] +name = "pg_extension" +version = "0.1.0" +edition = "2021" + +[lib] +crate-type = ["cdylib"] + +[features] +default = ["pg14", "python"] +python = ["pyo3"] +pg11 = ["pgrx/pg11", "pgrx-tests/pg11" ] +pg12 = ["pgrx/pg12", "pgrx-tests/pg12" ] +pg13 = ["pgrx/pg13", "pgrx-tests/pg13" ] +pg14 = ["pgrx/pg14", "pgrx-tests/pg14" ] +pg15 = ["pgrx/pg15", "pgrx-tests/pg15" ] +pg_test = [] + +[dependencies] +pgrx = "=0.9.7" +pgrx-pg-sys = "=0.9.7" +serde_json = { version = "1.0.85", features = ["preserve_order"] } +pyo3 = { version = "0.17", features = ["auto-initialize"], optional = true } +once_cell = "1.8.0" +log = "0.4.14" +serde = "1.0" +serde_derive = "1.0" + +[dev-dependencies] +pgrx-tests = "=0.9.7" + +[profile.dev] +panic = "unwind" + +[profile.release] +panic = "unwind" +opt-level = 3 +lto = "fat" +codegen-units = 1 \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/pg_extension.control b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/pg_extension.control new file mode 100644 index 0000000000..a28d2e9d87 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/pg_extension.control @@ -0,0 +1,5 @@ +comment = 'pg_extension: Created by pgrx' +default_version = '@CARGO_VERSION@' +module_pathname = '$libdir/pg_extension' +relocatable = false +superuser = true \ No newline at end of file diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/filter_phase.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/filter_phase.sql new file mode 100644 index 0000000000..173999b437 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/filter_phase.sql @@ -0,0 +1,35 @@ + + +CREATE OR REPLACE +PROCEDURE model_selection_sp( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + N INTEGER, --number of models to evaluate + batch_size INTEGER, --batch size, for profiling, filtering + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + + -- 4. Run filtering phase to get top K models. + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + LIMIT %s OFFSET 0 + ) + SELECT filtering_phase( + json_agg(row_to_json(t))::text, %s, %s, %L + ) + FROM batch_rows AS t', column_list, dataset, batch_size, N, 1, config_file) INTO result_status; + RAISE NOTICE '4. run filtering phase, k models = %', result_status; + +END; $$; diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu.sql new file mode 100644 index 0000000000..366d88d6d4 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu.sql @@ -0,0 +1,31 @@ + + +CREATE OR REPLACE +PROCEDURE model_selection_end2end( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + budget TEXT, --user given time budget + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; + +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + ) + SELECT model_selection( + json_agg(row_to_json(t))::text, %L, %L + ) + FROM batch_rows AS t', column_list, dataset, budget, config_file) INTO result_status; + RAISE NOTICE '1. model_selection result: %', result_status; +END; $$; diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu_workloads.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu_workloads.sql new file mode 100644 index 0000000000..835c627d95 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_cpu_workloads.sql @@ -0,0 +1,32 @@ + + +CREATE OR REPLACE +PROCEDURE model_selection_workloads( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + N INTEGER, --explore N models + K INTEGER, --keep K models + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; + +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + ) + SELECT model_selection_workloads( + json_agg(row_to_json(t))::text, %s, %s, %L + ) + FROM batch_rows AS t', column_list, dataset, N, K, config_file) INTO result_status; + RAISE NOTICE '1. model_selection result: %', result_status; +END; $$; diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_dev.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_dev.sql new file mode 100644 index 0000000000..12a4fdd4c3 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_dev.sql @@ -0,0 +1,80 @@ + + +CREATE OR REPLACE +PROCEDURE model_selection_sp( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + budget TEXT, --user given time budget + batch_size INTEGER, --batch size, for profiling, filtering + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; + + -- UDF outputs + score_time TEXT; + train_time TEXT; + coordinator_k integer; + coordinator_u integer; + coordinator_n integer; +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + + -- 1. Profiling time to score a model with TFMEM + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + LIMIT %s OFFSET 0 + ) + SELECT profiling_filtering_phase( + json_agg(row_to_json(t))::text, %L + ) + FROM batch_rows AS t', column_list, dataset, batch_size, config_file) INTO result_status; + score_time := json_extract_path_text(result_status::json, 'time'); + RAISE NOTICE '1. profiling_filtering_phase, get score_time: %', score_time; + + -- 2. Profiling time of training a model for one epoch + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + LIMIT %s OFFSET 0 + ) + SELECT profiling_refinement_phase( + json_agg(row_to_json(t))::text, %L + ) + FROM batch_rows AS t', column_list, dataset, batch_size, config_file) INTO result_status; + train_time := json_extract_path_text(result_status::json, 'time'); + RAISE NOTICE '2. profiling_refinement_phase, get train_time: %', train_time; + + -- 3. Coordinator to get N, K ,U + EXECUTE format('SELECT "coordinator"(%L, %L, %L, false, %L)', score_time, train_time, budget, config_file) INTO result_status; + + coordinator_k := (json_extract_path_text(result_status::json, 'k'))::integer; + coordinator_u := (json_extract_path_text(result_status::json, 'u'))::integer; + coordinator_n := (json_extract_path_text(result_status::json, 'n'))::integer; + RAISE NOTICE '3. coordinator result: k = %, u = %, n = %', coordinator_k, coordinator_u, coordinator_n; + + -- 4. Run filtering phase to get top K models. + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + LIMIT %s OFFSET 0 + ) + SELECT filtering_phase( + json_agg(row_to_json(t))::text, %s, %s, %L + ) + FROM batch_rows AS t', column_list, dataset, batch_size, coordinator_n, coordinator_k, config_file) INTO result_status; + RAISE NOTICE '4. run filtering phase, k models = %', result_status; + +END; $$; diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails.sql new file mode 100644 index 0000000000..24cbd60781 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails.sql @@ -0,0 +1,31 @@ + + +CREATE OR REPLACE +PROCEDURE model_selection_end2end( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + budget TEXT, --user given time budget + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; + +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + ) + SELECT model_selection_trails( + json_agg(row_to_json(t))::text, %L, %L + ) + FROM batch_rows AS t', column_list, dataset, budget, config_file) INTO result_status; + RAISE NOTICE '1. model_selection result: %', result_status; +END; $$; diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails_workloads.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails_workloads.sql new file mode 100644 index 0000000000..89c0db1530 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/model_selection_trails_workloads.sql @@ -0,0 +1,32 @@ + + +CREATE OR REPLACE +PROCEDURE model_selection_workloads( + dataset TEXT, --dataset name + selected_columns TEXT[], --used columns + N INTEGER, --explore N models + K INTEGER, --keep K models + config_file TEXT --config file path +) +LANGUAGE plpgsql +AS $$ +DECLARE + -- global inputs/outputs + result_status TEXT; + column_list TEXT; + +BEGIN + -- combine the columns into a string + column_list := array_to_string(selected_columns, ', '); + EXECUTE format(' + WITH batch_rows AS ( + SELECT %s + FROM %I + ORDER BY RANDOM() + ) + SELECT model_selection_trails_workloads( + json_agg(row_to_json(t))::text, %s, %s, %L + ) + FROM batch_rows AS t', column_list, dataset, N, K, config_file) INTO result_status; + RAISE NOTICE '1. model_selection result: %', result_status; +END; $$; diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/pg_extension--0.1.0.sql b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/pg_extension--0.1.0.sql new file mode 100644 index 0000000000..c096c00312 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/sql/pg_extension--0.1.0.sql @@ -0,0 +1,139 @@ +/* +This file is auto generated by pgrx. + +The ordering of items is not stable, it is driven by a dependency graph. +*/ + +-- src/lib.rs:80 +-- pg_extension::refinement_phase +CREATE FUNCTION "refinement_phase"( + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'refinement_phase_wrapper'; + +-- src/lib.rs:31 +-- pg_extension::profiling_refinement_phase +CREATE FUNCTION "profiling_refinement_phase"( + "mini_batch" TEXT, /* alloc::string::String */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'profiling_refinement_phase_wrapper'; + +-- src/lib.rs:16 +-- pg_extension::profiling_filtering_phase +CREATE FUNCTION "profiling_filtering_phase"( + "mini_batch" TEXT, /* alloc::string::String */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'profiling_filtering_phase_wrapper'; + +-- src/lib.rs:66 +-- pg_extension::filtering_phase +CREATE FUNCTION "filtering_phase"( + "mini_batch" TEXT, /* alloc::string::String */ + "n" INT, /* i32 */ + "k" INT, /* i32 */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'filtering_phase_wrapper'; + +-- src/lib.rs:46 +-- pg_extension::coordinator +CREATE FUNCTION "coordinator"( + "time_score" TEXT, /* alloc::string::String */ + "time_train" TEXT, /* alloc::string::String */ + "time_budget" TEXT, /* alloc::string::String */ + "only_phase1" bool, /* bool */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'coordinator_wrapper'; + + +-- src/lib.rs:110 +-- pg_extension::model_selection_workloads +CREATE FUNCTION "model_selection_workloads"( + "mini_batch" TEXT, /* alloc::string::String */ + "n" INT, /* i32 */ + "k" INT, /* i32 */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'model_selection_workloads_wrapper'; + +-- src/lib.rs:138 +-- pg_extension::model_selection_trails_workloads +CREATE FUNCTION "model_selection_trails_workloads"( + "mini_batch" TEXT, /* alloc::string::String */ + "n" INT, /* i32 */ + "k" INT, /* i32 */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'model_selection_trails_workloads_wrapper'; + +-- src/lib.rs:125 +-- pg_extension::model_selection_trails +CREATE FUNCTION "model_selection_trails"( + "mini_batch" TEXT, /* alloc::string::String */ + "time_budget" TEXT, /* alloc::string::String */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'model_selection_trails_wrapper'; + +-- src/lib.rs:94 +-- pg_extension::model_selection +CREATE FUNCTION "model_selection"( + "mini_batch" TEXT, /* alloc::string::String */ + "time_budget" TEXT, /* alloc::string::String */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'model_selection_wrapper'; + +-- src/lib.rs:153 +-- pg_extension::benchmark_filtering_phase_latency +CREATE FUNCTION "benchmark_filtering_phase_latency"( + "explore_models" INT, /* i32 */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'benchmark_filtering_phase_latency_wrapper'; + + +-- src/lib.rs:152 +-- pg_extension::benchmark_filtering_phase_latency +CREATE FUNCTION "benchmark_filtering_phase_latency"( + "explore_models" INT, /* i32 */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'benchmark_filtering_phase_latency_wrapper'; + +-- src/lib.rs:163 +-- pg_extension::benchmark_filtering_latency_in_db +CREATE FUNCTION "benchmark_filtering_latency_in_db"( + "explore_models" INT, /* i32 */ + "dataset" TEXT, /* alloc::string::String */ + "config_file" TEXT /* alloc::string::String */ +) RETURNS TEXT /* alloc::string::String */ + IMMUTABLE STRICT PARALLEL SAFE +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'benchmark_filtering_latency_in_db_wrapper'; + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ml_register.rs b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ml_register.rs new file mode 100644 index 0000000000..d107203011 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ml_register.rs @@ -0,0 +1,55 @@ +use log::error; +use once_cell::sync::Lazy; +use pyo3::prelude::*; +use pyo3::types::PyTuple; + + +pub fn run_python_function( + py_module: &Lazy>, + parameters: &String, + function_name: &str, +) -> serde_json::Value { + let parameters_str = parameters.to_string(); + let results = Python::with_gil(|py| -> String { + let run_script: Py = py_module.getattr(py, function_name).unwrap().into(); + let result = run_script.call1( + py, + PyTuple::new( + py, + &[parameters_str.into_py(py)], + ), + ); + let result = match result { + Err(e) => { + let traceback = e.traceback(py).unwrap().format().unwrap(); + error!("{traceback} {e}"); + format!("{traceback} {e}") + } + Ok(o) => o.extract(py).unwrap(), + }; + result + }); + + serde_json::from_str(&results).unwrap() +} + + +/* + Python Module Path for Model Selection + */ +pub static PY_MODULE: Lazy> = Lazy::new(|| { + Python::with_gil(|py| -> Py { + let src = include_str!(concat!( + env!("CARGO_MANIFEST_DIR"), + "/../ml/model_selection/pg_interface.py" + )); + PyModule::from_code(py, src, "", "").unwrap().into() + }) +}); + + + + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/mod.rs b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/mod.rs new file mode 100644 index 0000000000..70be9ec7e1 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/mod.rs @@ -0,0 +1,7 @@ + + +#[cfg(feature = "python")] +pub mod ms; +mod ml_register; +mod model; + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/model.rs b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/model.rs new file mode 100644 index 0000000000..af09a5c82f --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/model.rs @@ -0,0 +1,19 @@ + +use serde::{Serialize, Deserialize}; + + +#[derive(Debug, Serialize, Deserialize)] +pub(crate) struct Frappe { + pub(crate) id: i32, + pub(crate) label: i32, + pub(crate) col1: String, + pub(crate) col2: String, + pub(crate) col3: String, + pub(crate) col4: String, + pub(crate) col5: String, + pub(crate) col6: String, + pub(crate) col7: String, + pub(crate) col8: String, + pub(crate) col9: String, + pub(crate) col10: String, +} diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ms.rs b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ms.rs new file mode 100644 index 0000000000..21fb5f65cb --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/bindings/ms.rs @@ -0,0 +1,207 @@ +use serde_json::json; +use std::collections::HashMap; +use pgrx::prelude::*; +use crate::bindings::ml_register::PY_MODULE; +use crate::bindings::ml_register::run_python_function; +use std::time::{Instant, Duration}; + + +pub fn profiling_filtering_phase( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "profiling_filtering_phase") +} + + +pub fn profiling_refinement_phase( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "profiling_refinement_phase") +} + + +pub fn coordinator( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "coordinator") +} + + +pub fn filtering_phase( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "filtering_phase_dataLoader") +} + + +pub fn refinement_phase() -> serde_json::Value { + let task = "refinement_phase".to_string(); + run_python_function(&PY_MODULE, &task, "refinement_phase") +} + + +// this two are filtering + refinement in UDF runtime +pub fn model_selection( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "model_selection") +} + + +pub fn model_selection_workloads( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "model_selection_workloads") +} + + +// this two are filtering + refinement in GPU server +pub fn model_selection_trails( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "model_selection_trails") +} + + +pub fn model_selection_trails_workloads( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "model_selection_trails_workloads") +} + +// micro benchmarks + +pub fn benchmark_filtering_phase_latency( + task: &String +) -> serde_json::Value { + run_python_function(&PY_MODULE, task, "benchmark_filtering_phase_latency") +} + +pub fn benchmark_filtering_latency_in_db( + explore_models: i32, dataset: &String, config_file: &String) -> serde_json::Value { + + let overall_start_time = Instant::now(); + + let database_name = "pg_extension"; + let mut last_id = 0; + let mut eva_results = serde_json::Value::Null; // Initializing the eva_results + + for i in 1..explore_models { + + // Step 1: Initialize State in Python + let mut task_map = HashMap::new(); + task_map.insert("config_file", config_file.clone()); + task_map.insert("dataset", dataset.clone()); + task_map.insert("eva_results", eva_results.to_string()); + let task_json = json!(task_map).to_string(); + + // here it cache a state + let sample_result = run_python_function( + &PY_MODULE, + &task_json, + "in_db_filtering_state_init"); + + // 2. query data via SPI + let start_time = Instant::now(); + let results: Result>, String> = Spi::connect(|client| { + let query = format!("SELECT * FROM {}_train WHERE id > {} ORDER BY id ASC LIMIT 32", dataset, last_id); + let mut cursor = client.open_cursor(&query, None); + let table = match cursor.fetch(32) { + Ok(table) => table, + Err(e) => return Err(e.to_string()), // Convert the error to a string and return + }; + + let mut mini_batch = Vec::new(); + + for row in table.into_iter() { + let mut each_row = Vec::new(); + // add primary key + let col0 = match row.get::(1) { + Ok(Some(val)) => { + // Update last_id with the retrieved value + if val > 100000{ + last_id = 0; + }else{ + last_id = val + } + val.to_string() + } + Ok(None) => "".to_string(), // Handle the case when there's no valid value + Err(e) => e.to_string(), + }; + each_row.push(col0); + // add label + let col1 = match row.get::(2) { + Ok(val) => val.map(|i| i.to_string()).unwrap_or_default(), + Err(e) => e.to_string(), + }; + each_row.push(col1); + // add fields + let texts: Vec = (3..row.columns()+1) + .filter_map(|i| { + match row.get::<&str>(i) { + Ok(Some(s)) => Some(s.to_string()), + Ok(None) => None, + Err(e) => Some(e.to_string()), // Convert error to string + } + }).collect(); + each_row.extend(texts); + mini_batch.push(each_row) + } + // return + Ok(mini_batch) + }); + // serialize the mini-batch data + let tup_table = match results { + Ok(data) => { + serde_json::json!({ + "status": "success", + "data": data + }) + } + Err(e) => { + serde_json::json!({ + "status": "error", + "message": format!("Error while connecting: {}", e) + }) + } + }; + + let end_time = Instant::now(); + let elapsed_time = end_time.duration_since(start_time); + let elapsed_seconds = elapsed_time.as_secs_f64(); + + // Step 3: model evaluate in Python + let mut eva_task_map = HashMap::new(); + eva_task_map.insert("config_file", config_file.clone()); + eva_task_map.insert("sample_result", sample_result.to_string()); + let mini_batch_json = tup_table.to_string(); + eva_task_map.insert("mini_batch", mini_batch_json); + eva_task_map.insert("spi_seconds", elapsed_seconds.to_string()); + eva_task_map.insert("model_index", i.to_string()); + + let eva_task_json = json!(eva_task_map).to_string(); // Corrected this line + + eva_results = run_python_function( + &PY_MODULE, + &eva_task_json, + "in_db_filtering_evaluate"); + } + + let mut record_task_map = HashMap::new(); + record_task_map.insert("config_file", config_file.clone()); + record_task_map.insert("dataset", dataset.clone()); + let record_task_json = json!(record_task_map).to_string(); + run_python_function( + &PY_MODULE, + &record_task_json, + "records_results"); + + let overall_end_time = Instant::now(); + let overall_elapsed_time = overall_end_time.duration_since(overall_start_time); + let overall_elapsed_seconds = overall_elapsed_time.as_secs_f64(); + + // Step 4: Return to PostgresSQL + return serde_json::json!(overall_elapsed_seconds.to_string()); +} + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/lib.rs b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/lib.rs new file mode 100644 index 0000000000..cb99ed8329 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/src/lib.rs @@ -0,0 +1,171 @@ +use pgrx::prelude::*; +pgrx::pg_module_magic!(); +use serde_json::json; +use std::collections::HashMap; + +pub mod bindings; +extern crate serde_derive; + +/* + * @param mini_batch: mini_batch of data. Assume all columns are string type in + * libsvm codding + */ +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "profiling_filtering_phase")] +#[allow(unused_variables)] +pub fn profiling_filtering_phase(mini_batch: String, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::profiling_filtering_phase(&task_json).to_string() +} + +/* + * @param mini_batch: training for one iteration. + * libsvm codding + */ +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "profiling_refinement_phase")] +#[allow(unused_variables)] +pub fn profiling_refinement_phase(mini_batch: String, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::profiling_refinement_phase(&task_json).to_string() +} + +/* + * @param mini_batch: training for one iteration. + * libsvm codding + */ +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "coordinator")] +#[allow(unused_variables)] +pub fn coordinator(time_score: String, time_train: String, time_budget: String, only_phase1: bool, + config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("budget", time_budget); + task_map.insert("score_time_per_model", time_score); + task_map.insert("train_time_per_epoch", time_train); + task_map.insert("only_phase1", only_phase1.to_string()); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::coordinator(&task_json).to_string() +} + + +/* + * @param mini_batch: mini_batch of data. Assume all columns are string type in + * libsvm codding + */ +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "filtering_phase")] +#[allow(unused_variables)] +pub fn filtering_phase(mini_batch: String, n: i32, k: i32, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("n", n.to_string()); + task_map.insert("k", k.to_string()); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::filtering_phase(&task_json).to_string() +} + + +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "refinement_phase")] +#[allow(unused_variables)] +pub fn refinement_phase(config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::refinement_phase().to_string() +} + + +/* + End-2-End model selection, All in UDF runtime. + */ +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "model_selection")] +#[allow(unused_variables)] +pub fn model_selection(mini_batch: String, time_budget: String, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("budget", time_budget); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::model_selection(&task_json).to_string() +} + +/* + * @param mini_batch: mini_batch of data. Assume all columns are string type in + * libsvm codding + */ +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "model_selection_workloads")] +#[allow(unused_variables)] +pub fn model_selection_workloads(mini_batch: String, n: i32, k: i32, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("n", n.to_string()); + task_map.insert("k", k.to_string()); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::model_selection_workloads(&task_json).to_string() +} + + +// this two are filtering + refinement in GPU server +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "model_selection_trails")] +#[allow(unused_variables)] +pub fn model_selection_trails(mini_batch: String, time_budget: String, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("budget", time_budget); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::model_selection_trails(&task_json).to_string() +} + + +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "model_selection_trails_workloads")] +#[allow(unused_variables)] +pub fn model_selection_trails_workloads(mini_batch: String, n: i32, k: i32, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("mini_batch", mini_batch); + task_map.insert("n", n.to_string()); + task_map.insert("k", k.to_string()); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::model_selection_trails_workloads(&task_json).to_string() +} + +// micro benchmarks +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "benchmark_filtering_phase_latency")] +#[allow(unused_variables)] +pub fn benchmark_filtering_phase_latency(explore_models: i32, config_file: String) -> String { + let mut task_map = HashMap::new(); + task_map.insert("explore_models", explore_models.to_string()); + task_map.insert("config_file", config_file); + let task_json = json!(task_map).to_string(); + crate::bindings::ms::benchmark_filtering_phase_latency(&task_json).to_string() +} + +#[cfg(feature = "python")] +#[pg_extern(immutable, parallel_safe, name = "benchmark_filtering_latency_in_db")] +#[allow(unused_variables)] +pub fn benchmark_filtering_latency_in_db( + explore_models: i32, dataset: String, config_file: String) -> String { + crate::bindings::ms::benchmark_filtering_latency_in_db(explore_models, &dataset, &config_file).to_string() +} + + + + + diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/test/lib.rs b/examples/model_selection/TRAILS-Database-Native-Model-Selection/internal/pg_extension/test/lib.rs new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/model_selection/TRAILS-Database-Native-Model-Selection/requirement.txt b/examples/model_selection/TRAILS-Database-Native-Model-Selection/requirement.txt new file mode 100644 index 0000000000..ab233d87b5 --- /dev/null +++ b/examples/model_selection/TRAILS-Database-Native-Model-Selection/requirement.txt @@ -0,0 +1,32 @@ +ConfigSpace==0.7.1 +contourpy==1.1.0 +cycler==0.11.0 +fonttools==4.41.0 +importlib-resources==6.0.0 +joblib==1.3.1 +kiwisolver==1.4.4 +matplotlib==3.7.2 +more-itertools==9.1.0 +numpy==1.24.4 +orjson==3.9.2 +packaging==23.1 +palettable==3.3.3 +pandas==2.0.3 +Pillow==10.0.0 +pyparsing==3.0.9 +python-dateutil==2.8.2 +pytz==2023.3 +scikit-learn==1.3.0 +scipy==1.10.1 +seaborn==0.12.2 +six==1.16.0 +sklearn==0.0 +threadpoolctl==3.1.0 +torch==1.8.1 +torchaudio==0.8.1 +torchvision==0.9.1 +tqdm==4.47.0 +typing_extensions==4.7.1 +tzdata==2023.3 +zipp==3.16.2 +requests==2.31.0 diff --git a/setup.py b/setup.py index b3147e57ed..cfd87d6114 100644 --- a/setup.py +++ b/setup.py @@ -83,7 +83,7 @@ from datetime import date # stable version -VERSION = '4.0.0' +VERSION = '4.1.0' # get the git hash # git_hash = subprocess.check_output(["git", "describe"]).strip().split('-')[-1][1:] # comment the next line to build wheel for stable version diff --git a/tool/conda/singa/meta.yaml b/tool/conda/singa/meta.yaml index 5a01ef3a92..5bcc362128 100644 --- a/tool/conda/singa/meta.yaml +++ b/tool/conda/singa/meta.yaml @@ -20,7 +20,7 @@ # https://docs.conda.io/projects/conda-build/en/latest/resources/define-metadata.html#templating-with-jinja # {% set data = load_setup_py_data(setup_file='../../../python/singa/setup.py', from_recipe_dir=True) %} -{% set version = "4.0.0" %} +{% set version = "4.1.0" %} package: name: singa diff --git a/tool/docker/devel/centos6/cuda10/Dockerfile.manylinux2014 b/tool/docker/devel/centos6/cuda10/Dockerfile.manylinux2014 index 1472c9fc90..107e3465c8 100644 --- a/tool/docker/devel/centos6/cuda10/Dockerfile.manylinux2014 +++ b/tool/docker/devel/centos6/cuda10/Dockerfile.manylinux2014 @@ -60,8 +60,8 @@ RUN /opt/python/cp38-cp38/bin/pip install numpy # install cuda and cudnn # Refer to https://gitlab.com/nvidia/container-images/cuda/-/tree/master/dist for other cuda and cudnn versions # 10.2-base-centos7 -RUN NVIDIA_GPGKEY_SUM=d1be581509378368edeec8c1eb2958702feedf3bc3d17011adbf24efacce4ab5 && \ - curl -fsSL https://developer.download.nvidia.com/compute/cuda/repos/rhel7/x86_64/7fa2af80.pub | sed '/^Version/d' > /etc/pki/rpm-gpg/RPM-GPG-KEY-NVIDIA && \ +RUN NVIDIA_GPGKEY_SUM=d0664fbbdb8c32356d45de36c5984617217b2d0bef41b93ccecd326ba3b80c87 && \ + curl -fsSL https://developer.download.nvidia.com/compute/cuda/repos/rhel7/x86_64/D42D0685.pub | sed '/^Version/d' > /etc/pki/rpm-gpg/RPM-GPG-KEY-NVIDIA && \ echo "$NVIDIA_GPGKEY_SUM /etc/pki/rpm-gpg/RPM-GPG-KEY-NVIDIA" | sha256sum -c --strict - COPY cuda.repo /etc/yum.repos.d/cuda.repo ENV CUDA_VERSION 10.2.89