From 5655ea2f39a2f5b543edd357c9926c804c6c2c2d Mon Sep 17 00:00:00 2001 From: Rongfeng Fu Date: Thu, 23 Mar 2023 11:33:44 +0800 Subject: [PATCH] V2.0.0 (#162) --- .gitignore | 31 +- _cmd.py | 239 +- _deploy.py | 125 +- _environ.py | 5 + _errno.py | 197 +- _lock.py | 36 +- _mirror.py | 64 +- _plugin.py | 199 +- _repository.py | 51 +- _stdio.py | 164 +- .../oceanbase/cluster_config_parser.py | 8 +- core.py | 1235 ++-- example/all-components-min.yaml | 38 +- example/all-components.yaml | 36 +- example/autodeploy/all-components.yaml | 240 +- example/autodeploy/default-example.yaml | 216 +- example/autodeploy/distributed-example.yaml | 4 +- ...uted-with-obproxy-and-obagent-example.yaml | 73 +- .../distributed-with-obproxy-example.yaml | 4 +- example/autodeploy/single-example.yaml | 4 +- .../single-with-obproxy-example.yaml | 4 +- example/distributed-example.yaml | 1 - example/distributed-with-obproxy-example.yaml | 1 - ...omponents-with-prometheus-and-grafana.yaml | 300 + example/local-example.yaml | 1 - example/mini-distributed-example.yaml | 3 +- ...mini-distributed-with-obproxy-example.yaml | 3 +- example/mini-local-example.yaml | 3 +- example/mini-single-example.yaml | 3 +- example/mini-single-with-obproxy-example.yaml | 3 +- ...uted-with-obproxy-and-obagent-example.yaml | 78 +- .../obagent/obagent-only-1.2.0-example.yaml | 84 + example/obagent/obagent-only-example.yaml | 62 +- .../distributed-with-obproxy-example.yaml | 1 - ...uted-with-obproxy-and-obagent-example.yaml | 159 + example/ocp-express/ocp-express-only.yaml | 34 + ...d-with-obagent-and-prometheus-example.yaml | 65 +- example/single-example.yaml | 1 - example/single-with-obproxy-example.yaml | 1 - optimize/obproxy/4.1.0/sysbench.yaml | 10 + optimize/obproxy/4.1.0/tpcc.yaml | 10 + optimize/oceanbase-ce/4.1.0/optimizer.yaml | 14 + optimize/oceanbase-ce/4.1.0/sysbench.yaml | 61 + optimize/oceanbase-ce/4.1.0/tpcc.yaml | 63 + optimize/oceanbase/4.1.0/optimizer.yaml | 14 + optimize/oceanbase/4.1.0/sysbench.yaml | 61 + optimize/oceanbase/4.1.0/tpcc.yaml | 63 + .../optimize_parser/0.1/optimize_parser.py | 31 +- plugins/commands/0.1/command_template.yaml | 15 +- plugins/general/0.1/install_repo.py | 2 +- plugins/grafana/7.5.17/display.py | 15 +- plugins/grafana/7.5.17/generate_config.py | 31 +- plugins/grafana/7.5.17/init.py | 2 +- plugins/grafana/7.5.17/parameter.yaml | 10 +- plugins/grafana/7.5.17/reload.py | 2 +- plugins/grafana/7.5.17/restart.py | 62 +- plugins/grafana/7.5.17/start.py | 93 +- plugins/grafana/7.5.17/start_check.py | 161 +- plugins/grafana/7.5.17/stop.py | 2 +- plugins/mysqltest/3.1.0/check_opt.py | 20 +- plugins/mysqltest/3.1.0/check_test.py | 18 +- plugins/mysqltest/3.1.0/init.py | 9 +- plugins/mysqltest/3.1.0/run_test.py | 10 +- plugins/mysqltest/4.0.0.0/check_test.py | 18 +- plugins/mysqltest/4.0.0.0/init.py | 3 +- plugins/obagent/0.1/display.py | 6 +- plugins/obagent/0.1/generate_config.py | 27 +- plugins/obagent/0.1/init.py | 4 +- plugins/obagent/0.1/parameter.yaml | 28 +- plugins/obagent/0.1/reload.py | 8 +- plugins/obagent/0.1/restart.py | 47 +- plugins/obagent/0.1/start.py | 45 +- plugins/obagent/0.1/start_check.py | 125 +- plugins/obagent/0.1/stop.py | 2 +- plugins/obagent/0.1/upgrade.py | 28 +- plugins/obagent/1.1.0/parameter.yaml | 30 +- plugins/obagent/1.1.0/start.py | 45 +- plugins/obagent/1.1.1/start_check.py | 124 +- plugins/obagent/1.3.0/connect.py | 100 + plugins/obagent/1.3.0/display.py | 46 + plugins/obagent/1.3.0/file_map.yaml | 29 + plugins/obagent/1.3.0/init.py | 84 + plugins/obagent/1.3.0/parameter.yaml | 243 + plugins/obagent/1.3.0/reload.py | 109 + plugins/obagent/1.3.0/restart.py | 135 + plugins/obagent/1.3.0/start.py | 337 + plugins/obagent/1.3.0/start_check.py | 250 + plugins/obagent/1.3.0/status.py | 40 + plugins/obagent/1.3.0/stop.py | 110 + plugins/obagent/1.3.0/upgrade.py | 159 + plugins/obproxy/3.1.0/bootstrap.py | 12 +- plugins/obproxy/3.1.0/connect.py | 122 +- plugins/obproxy/3.1.0/display.py | 37 +- plugins/obproxy/3.1.0/generate_config.py | 56 +- plugins/obproxy/3.1.0/init.py | 4 +- plugins/obproxy/3.1.0/parameter.yaml | 19 +- plugins/obproxy/3.1.0/reload.py | 19 +- plugins/obproxy/3.1.0/restart.py | 67 +- plugins/obproxy/3.1.0/start.py | 8 +- plugins/obproxy/3.1.0/start_check.py | 134 +- plugins/obproxy/3.1.0/stop.py | 2 +- plugins/obproxy/3.1.0/upgrade.py | 17 +- plugins/oceanbase/3.1.0/bootstrap.py | 46 +- plugins/oceanbase/3.1.0/connect.py | 104 +- plugins/oceanbase/3.1.0/create_tenant.py | 445 +- plugins/oceanbase/3.1.0/display.py | 23 +- plugins/oceanbase/3.1.0/drop_tenant.py | 70 +- plugins/oceanbase/3.1.0/generate_config.py | 207 +- plugins/oceanbase/3.1.0/init.py | 4 +- plugins/oceanbase/3.1.0/list_tenant.py | 85 + plugins/oceanbase/3.1.0/major_freeze.py | 34 +- plugins/oceanbase/3.1.0/ocp_check.py | 6 +- plugins/oceanbase/3.1.0/parameter.yaml | 71 +- plugins/oceanbase/3.1.0/reload.py | 39 +- plugins/oceanbase/3.1.0/restart.py | 97 +- plugins/oceanbase/3.1.0/start.py | 14 +- plugins/oceanbase/3.1.0/start_check.py | 384 +- plugins/oceanbase/3.1.0/stop.py | 2 +- plugins/oceanbase/3.1.0/upgrade.py | 56 +- plugins/oceanbase/3.1.0/upgrade_check.py | 21 +- plugins/oceanbase/3.1.0/upgrade_file_check.py | 4 +- plugins/oceanbase/4.0.0.0/bootstrap.py | 103 +- plugins/oceanbase/4.0.0.0/create_tenant.py | 392 +- plugins/oceanbase/4.0.0.0/drop_tenant.py | 64 +- plugins/oceanbase/4.0.0.0/generate_config.py | 424 +- plugins/oceanbase/4.0.0.0/init.py | 4 +- plugins/oceanbase/4.0.0.0/list_tenant.py | 89 + plugins/oceanbase/4.0.0.0/major_freeze.py | 34 +- plugins/oceanbase/4.0.0.0/parameter.yaml | 171 +- plugins/oceanbase/4.0.0.0/restart.py | 87 +- plugins/oceanbase/4.0.0.0/start.py | 19 +- plugins/oceanbase/4.0.0.0/start_check.py | 506 +- plugins/oceanbase/4.0.0.0/upgrade.py | 544 ++ plugins/oceanbase/4.0.0.0/upgrade_check.py | 82 + plugins/ocp-express/1.0/bootstrap.py | 57 + plugins/ocp-express/1.0/connect.py | 111 + plugins/ocp-express/1.0/destroy.py | 56 + plugins/ocp-express/1.0/display.py | 50 + plugins/ocp-express/1.0/file_map.yaml | 6 + plugins/ocp-express/1.0/generate_config.py | 64 + plugins/ocp-express/1.0/init.py | 124 + plugins/ocp-express/1.0/parameter.yaml | 348 + plugins/ocp-express/1.0/reload.py | 27 + plugins/ocp-express/1.0/restart.py | 157 + plugins/ocp-express/1.0/start.py | 453 ++ plugins/ocp-express/1.0/start_check.py | 527 ++ plugins/ocp-express/1.0/status.py | 39 + plugins/ocp-express/1.0/stop.py | 107 + plugins/prometheus/2.37.1/display.py | 12 +- plugins/prometheus/2.37.1/generate_config.py | 18 +- plugins/prometheus/2.37.1/init.py | 2 +- plugins/prometheus/2.37.1/parameter.yaml | 6 + plugins/prometheus/2.37.1/restart.py | 62 +- plugins/prometheus/2.37.1/start.py | 27 +- plugins/prometheus/2.37.1/start_check.py | 129 +- plugins/prometheus/2.37.1/stop.py | 4 +- plugins/sysbench/3.1.0/pre_test.py | 41 +- plugins/sysbench/4.0.0.0/pre_test.py | 42 +- plugins/tpcc/3.1.0/build.py | 91 +- plugins/tpcc/3.1.0/pre_test.py | 56 +- plugins/tpcc/3.1.0/run_test.py | 38 +- plugins/tpcc/4.0.0.0/analyze.sql | 20 +- plugins/tpcc/4.0.0.0/build.py | 84 +- plugins/tpcc/4.0.0.0/pre_test.py | 56 +- plugins/tpcc/4.0.0.0/run_test.py | 31 +- plugins/tpch/3.1.0/pre_test.py | 49 +- plugins/tpch/3.1.0/run_test.py | 43 +- plugins/tpch/4.0.0.0/analyze.sql | 16 +- plugins/tpch/4.0.0.0/pre_test.py | 51 +- plugins/tpch/4.0.0.0/run_test.py | 35 +- profile/obd.sh | 8 +- rpm/build.sh | 14 +- rpm/ob-deploy-build.sh | 4 +- rpm/ob-deploy.spec | 22 +- service/__init__.py | 0 service/api/__init__.py | 0 service/api/response.py | 35 + service/api/response_utils.py | 64 + service/api/v1/__init__.py | 0 service/api/v1/components.py | 76 + service/api/v1/deployments.py | 214 + service/api/v1/mirror.py | 42 + service/api/v1/process.py | 38 + service/api/v1/service_info.py | 38 + service/app.py | 70 + service/common/__init__.py | 0 service/common/const.py | 60 + service/common/core.py | 49 + service/common/log.py | 66 + service/common/task.py | 146 + service/common/util.py | 26 + service/handler/__init__.py | 0 service/handler/base_handler.py | 42 + service/handler/component_handler.py | 181 + service/handler/deployment_handler.py | 705 ++ service/handler/handler_utils.py | 44 + service/handler/mirror_handler.py | 56 + service/handler/process_handler.py | 36 + service/handler/service_info_handler.py | 33 + service/middleware/process_time.py | 36 + service/middleware/request_response_log.py | 44 + service/model/__init__.py | 0 service/model/components.py | 72 + service/model/deployments.py | 235 + service/model/mirror.py | 35 + service/model/service_info.py | 25 + service/service-requirements.txt | 9 + service/tests/__init__.py | 0 ssh.py | 26 +- tool.py | 26 +- web/.editorconfig | 16 + web/.eslintrc.js | 3 + web/.prettierignore | 9 + web/.prettierrc | 11 + web/README.md | 15 + web/config/config.ts | 31 + web/config/plugin.ts | 9 + web/mock/createDeploymentConfig.mock.ts | 15 + web/mock/deleteDeployment .mock.ts | 15 + web/mock/finishInstallAndKillProcess.mock.ts | 13 + web/mock/getDestroyTaskInfo.mock.ts | 31 + web/mock/getObdInfo.mock.ts | 13 + web/mock/installDeployment.mock.ts | 15 + web/mock/preCheckStatus.mock.ts | 91 + web/mock/precheck.mock.ts | 15 + web/mock/queryAllComponentVersions.mock.ts | 602 ++ .../queryComponentByComponentName.mock.ts | 105 + web/mock/queryComponentParameters.mock.ts | 681 ++ web/mock/queryConnectInfo.mock.ts | 94 + web/mock/queryConnectionInfo.mock.ts | 87 + web/mock/queryDeploymentConfig.mock.ts | 960 +++ web/mock/queryDeploymentInfo.mock.ts | 13 + ...ueryDeploymentInfoByTaskStatusType.mock.ts | 29 + web/mock/queryDeploymentReport.mock.ts | 77 + web/mock/queryInstallLog.mock.ts | 16 + web/mock/queryInstallStatus.mock.ts | 20 + web/mock/recover.mock.ts | 47 + web/package.json | 53 + web/public/assets/computer/data.json | 2123 ++++++ web/public/assets/database/data.json | 2123 ++++++ web/public/assets/empty.png | Bin 0 -> 14913 bytes web/public/assets/failed.png | Bin 0 -> 55696 bytes web/public/assets/logo.png | Bin 0 -> 3552 bytes web/public/assets/oceanbase.png | Bin 0 -> 13258 bytes web/public/assets/progress/data.json | 5269 ++++++++++++++ web/public/assets/spaceman/data.json | 6075 +++++++++++++++++ web/public/assets/successful.png | Bin 0 -> 56334 bytes web/public/assets/welcome/cover.jpg | Bin 0 -> 236659 bytes web/public/assets/welcome/data.mp4 | Bin 0 -> 1162582 bytes web/src/app.ts | 33 + web/src/global.less | 394 ++ web/src/models/global.ts | 68 + web/src/pages/components/CheckInfo.tsx | 489 ++ web/src/pages/components/ClusterConfig.tsx | 758 ++ .../pages/components/DeleteDeployModal.tsx | 198 + web/src/pages/components/DeployType.tsx | 64 + web/src/pages/components/DirInput.tsx | 116 + web/src/pages/components/ExitPage.tsx | 38 + web/src/pages/components/InstallConfig.tsx | 811 +++ web/src/pages/components/InstallFinished.tsx | 379 + web/src/pages/components/InstallProcess.tsx | 255 + web/src/pages/components/NodeConfig.tsx | 752 ++ web/src/pages/components/Parameter.tsx | 63 + web/src/pages/components/PreCheck.tsx | 9 + web/src/pages/components/PreCheckStatus.tsx | 677 ++ web/src/pages/components/ServerTags.tsx | 151 + web/src/pages/components/Steps.tsx | 80 + web/src/pages/components/Welcome.tsx | 76 + web/src/pages/components/index.less | 773 +++ web/src/pages/constants/index.ts | 95 + web/src/pages/index.less | 48 + web/src/pages/index.tsx | 113 + web/src/pages/theme.ts | 84 + web/src/services/ob-deploy-web/Components.ts | 45 + web/src/services/ob-deploy-web/Deployments.ts | 217 + web/src/services/ob-deploy-web/Info.ts | 11 + web/src/services/ob-deploy-web/Mirror.ts | 25 + web/src/services/ob-deploy-web/Processes.ts | 13 + web/src/services/ob-deploy-web/index.ts | 16 + web/src/services/ob-deploy-web/typings.d.ts | 683 ++ web/src/typings/global.d.ts | 54 + web/src/utils/index.tsx | 48 + web/src/utils/useRequest.ts | 70 + web/tsconfig.json | 37 + web/typings.d.ts | 10 + 285 files changed, 39993 insertions(+), 3517 deletions(-) create mode 100644 example/grafana/all-components-with-prometheus-and-grafana.yaml create mode 100644 example/obagent/obagent-only-1.2.0-example.yaml create mode 100644 example/oceanbase-3.x/distributed-with-obproxy-and-obagent-example.yaml create mode 100644 example/ocp-express/ocp-express-only.yaml create mode 100644 optimize/obproxy/4.1.0/sysbench.yaml create mode 100644 optimize/obproxy/4.1.0/tpcc.yaml create mode 100644 optimize/oceanbase-ce/4.1.0/optimizer.yaml create mode 100644 optimize/oceanbase-ce/4.1.0/sysbench.yaml create mode 100644 optimize/oceanbase-ce/4.1.0/tpcc.yaml create mode 100644 optimize/oceanbase/4.1.0/optimizer.yaml create mode 100644 optimize/oceanbase/4.1.0/sysbench.yaml create mode 100644 optimize/oceanbase/4.1.0/tpcc.yaml create mode 100644 plugins/obagent/1.3.0/connect.py create mode 100644 plugins/obagent/1.3.0/display.py create mode 100644 plugins/obagent/1.3.0/file_map.yaml create mode 100644 plugins/obagent/1.3.0/init.py create mode 100644 plugins/obagent/1.3.0/parameter.yaml create mode 100644 plugins/obagent/1.3.0/reload.py create mode 100644 plugins/obagent/1.3.0/restart.py create mode 100644 plugins/obagent/1.3.0/start.py create mode 100644 plugins/obagent/1.3.0/start_check.py create mode 100644 plugins/obagent/1.3.0/status.py create mode 100644 plugins/obagent/1.3.0/stop.py create mode 100644 plugins/obagent/1.3.0/upgrade.py create mode 100644 plugins/oceanbase/3.1.0/list_tenant.py create mode 100644 plugins/oceanbase/4.0.0.0/list_tenant.py create mode 100644 plugins/oceanbase/4.0.0.0/upgrade.py create mode 100644 plugins/oceanbase/4.0.0.0/upgrade_check.py create mode 100644 plugins/ocp-express/1.0/bootstrap.py create mode 100644 plugins/ocp-express/1.0/connect.py create mode 100644 plugins/ocp-express/1.0/destroy.py create mode 100644 plugins/ocp-express/1.0/display.py create mode 100644 plugins/ocp-express/1.0/file_map.yaml create mode 100644 plugins/ocp-express/1.0/generate_config.py create mode 100644 plugins/ocp-express/1.0/init.py create mode 100644 plugins/ocp-express/1.0/parameter.yaml create mode 100644 plugins/ocp-express/1.0/reload.py create mode 100644 plugins/ocp-express/1.0/restart.py create mode 100644 plugins/ocp-express/1.0/start.py create mode 100644 plugins/ocp-express/1.0/start_check.py create mode 100644 plugins/ocp-express/1.0/status.py create mode 100644 plugins/ocp-express/1.0/stop.py create mode 100644 service/__init__.py create mode 100644 service/api/__init__.py create mode 100644 service/api/response.py create mode 100644 service/api/response_utils.py create mode 100644 service/api/v1/__init__.py create mode 100644 service/api/v1/components.py create mode 100644 service/api/v1/deployments.py create mode 100644 service/api/v1/mirror.py create mode 100644 service/api/v1/process.py create mode 100644 service/api/v1/service_info.py create mode 100644 service/app.py create mode 100644 service/common/__init__.py create mode 100644 service/common/const.py create mode 100644 service/common/core.py create mode 100644 service/common/log.py create mode 100644 service/common/task.py create mode 100644 service/common/util.py create mode 100644 service/handler/__init__.py create mode 100644 service/handler/base_handler.py create mode 100644 service/handler/component_handler.py create mode 100644 service/handler/deployment_handler.py create mode 100644 service/handler/handler_utils.py create mode 100644 service/handler/mirror_handler.py create mode 100644 service/handler/process_handler.py create mode 100644 service/handler/service_info_handler.py create mode 100644 service/middleware/process_time.py create mode 100644 service/middleware/request_response_log.py create mode 100644 service/model/__init__.py create mode 100644 service/model/components.py create mode 100644 service/model/deployments.py create mode 100644 service/model/mirror.py create mode 100644 service/model/service_info.py create mode 100644 service/service-requirements.txt create mode 100644 service/tests/__init__.py create mode 100755 web/.editorconfig create mode 100644 web/.eslintrc.js create mode 100644 web/.prettierignore create mode 100644 web/.prettierrc create mode 100644 web/README.md create mode 100644 web/config/config.ts create mode 100644 web/config/plugin.ts create mode 100644 web/mock/createDeploymentConfig.mock.ts create mode 100644 web/mock/deleteDeployment .mock.ts create mode 100644 web/mock/finishInstallAndKillProcess.mock.ts create mode 100644 web/mock/getDestroyTaskInfo.mock.ts create mode 100644 web/mock/getObdInfo.mock.ts create mode 100644 web/mock/installDeployment.mock.ts create mode 100644 web/mock/preCheckStatus.mock.ts create mode 100644 web/mock/precheck.mock.ts create mode 100644 web/mock/queryAllComponentVersions.mock.ts create mode 100644 web/mock/queryComponentByComponentName.mock.ts create mode 100644 web/mock/queryComponentParameters.mock.ts create mode 100644 web/mock/queryConnectInfo.mock.ts create mode 100644 web/mock/queryConnectionInfo.mock.ts create mode 100644 web/mock/queryDeploymentConfig.mock.ts create mode 100644 web/mock/queryDeploymentInfo.mock.ts create mode 100644 web/mock/queryDeploymentInfoByTaskStatusType.mock.ts create mode 100644 web/mock/queryDeploymentReport.mock.ts create mode 100644 web/mock/queryInstallLog.mock.ts create mode 100644 web/mock/queryInstallStatus.mock.ts create mode 100644 web/mock/recover.mock.ts create mode 100644 web/package.json create mode 100644 web/public/assets/computer/data.json create mode 100644 web/public/assets/database/data.json create mode 100644 web/public/assets/empty.png create mode 100644 web/public/assets/failed.png create mode 100644 web/public/assets/logo.png create mode 100644 web/public/assets/oceanbase.png create mode 100644 web/public/assets/progress/data.json create mode 100644 web/public/assets/spaceman/data.json create mode 100644 web/public/assets/successful.png create mode 100644 web/public/assets/welcome/cover.jpg create mode 100644 web/public/assets/welcome/data.mp4 create mode 100644 web/src/app.ts create mode 100644 web/src/global.less create mode 100755 web/src/models/global.ts create mode 100644 web/src/pages/components/CheckInfo.tsx create mode 100644 web/src/pages/components/ClusterConfig.tsx create mode 100644 web/src/pages/components/DeleteDeployModal.tsx create mode 100644 web/src/pages/components/DeployType.tsx create mode 100644 web/src/pages/components/DirInput.tsx create mode 100644 web/src/pages/components/ExitPage.tsx create mode 100644 web/src/pages/components/InstallConfig.tsx create mode 100644 web/src/pages/components/InstallFinished.tsx create mode 100644 web/src/pages/components/InstallProcess.tsx create mode 100644 web/src/pages/components/NodeConfig.tsx create mode 100644 web/src/pages/components/Parameter.tsx create mode 100644 web/src/pages/components/PreCheck.tsx create mode 100644 web/src/pages/components/PreCheckStatus.tsx create mode 100644 web/src/pages/components/ServerTags.tsx create mode 100644 web/src/pages/components/Steps.tsx create mode 100644 web/src/pages/components/Welcome.tsx create mode 100644 web/src/pages/components/index.less create mode 100644 web/src/pages/constants/index.ts create mode 100644 web/src/pages/index.less create mode 100644 web/src/pages/index.tsx create mode 100644 web/src/pages/theme.ts create mode 100644 web/src/services/ob-deploy-web/Components.ts create mode 100644 web/src/services/ob-deploy-web/Deployments.ts create mode 100644 web/src/services/ob-deploy-web/Info.ts create mode 100644 web/src/services/ob-deploy-web/Mirror.ts create mode 100644 web/src/services/ob-deploy-web/Processes.ts create mode 100644 web/src/services/ob-deploy-web/index.ts create mode 100644 web/src/services/ob-deploy-web/typings.d.ts create mode 100644 web/src/typings/global.d.ts create mode 100644 web/src/utils/index.tsx create mode 100644 web/src/utils/useRequest.ts create mode 100644 web/tsconfig.json create mode 100644 web/typings.d.ts diff --git a/.gitignore b/.gitignore index 56e5fc0..c3578b3 100644 --- a/.gitignore +++ b/.gitignore @@ -6,4 +6,33 @@ dist .vscode .git __pycache__ -.idea/workspace.xml +.idea +.obd +plugins/oceanbase-ce +config_parser/oceanbase-ce +tags + .DS_store + + + +# dependencies +/web/node_modules +/web/npm-debug.log* +/web/yarn-error.log +/web/yarn.lock +/web/package-lock.json +/web/.mfsu-dev +/web/.mfsu-prod + +# production +/web/dist + +# misc +/web/**/.DS_Store +/web/.DS_Store + +# umi +/web/src/.umi +/web/src/.umi-production +/web/src/.umi-test +/web/.env.local diff --git a/_cmd.py b/_cmd.py index 6ad96f0..4fe1115 100644 --- a/_cmd.py +++ b/_cmd.py @@ -24,18 +24,16 @@ import os import sys import time -import logging import textwrap -from logging import handlers -from uuid import uuid1 as uuid +from uuid import uuid1 as uuid, UUID from optparse import OptionParser, OptionGroup, BadOptionError, Option, IndentedHelpFormatter from core import ObdHome from _stdio import IO -from log import Logger -from tool import DirectoryUtil, FileUtil, COMMAND_ENV +from _lock import LockMode +from tool import DirectoryUtil, FileUtil, NetUtil, COMMAND_ENV from _errno import DOC_LINK_MSG, LockError -from _environ import ENV_DEV_MODE +import _environ as ENV ROOT_IO = IO(1) @@ -43,11 +41,11 @@ REVISION = '' BUILD_BRANCH = '' BUILD_TIME = '' -DEBUG = True if '' else False CONST_OBD_HOME = "OBD_HOME" CONST_OBD_INSTALL_PRE = "OBD_INSTALL_PRE" -FORBIDDEN_VARS = (CONST_OBD_HOME, CONST_OBD_INSTALL_PRE) +CONST_OBD_INSTALL_PATH = "OBD_INSTALL_PATH" +FORBIDDEN_VARS = (CONST_OBD_HOME, CONST_OBD_INSTALL_PRE, CONST_OBD_INSTALL_PATH) OBD_HOME_PATH = os.path.join(os.environ.get(CONST_OBD_HOME, os.getenv('HOME')), '.obd') COMMAND_ENV.load(os.path.join(OBD_HOME_PATH, '.obd_environ'), ROOT_IO) @@ -147,6 +145,7 @@ def __init__(self, name, summary): self.prev_cmd = '' self.is_init = False self.hidden = False + self.has_trace = True self.parser = AllowUndefinedOptionParser(add_help_option=False) self.parser.add_option('-h', '--help', action='callback', callback=self._show_help, help='Show help and exit.') self.parser.add_option('-v', '--verbose', action='callback', callback=self._set_verbose, help='Activate verbose output.') @@ -185,10 +184,10 @@ class ObdCommand(BaseCommand): OBD_PATH = OBD_HOME_PATH OBD_INSTALL_PRE = os.environ.get(CONST_OBD_INSTALL_PRE, '/') + OBD_INSTALL_PATH = os.environ.get(CONST_OBD_INSTALL_PATH, os.path.join(OBD_INSTALL_PRE, 'usr/obd/')) def init_home(self): version_path = os.path.join(self.OBD_PATH, 'version') - need_update = True version_fobj = FileUtil.open(version_path, 'a+', stdio=ROOT_IO) version_fobj.seek(0) version = version_fobj.read() @@ -196,7 +195,7 @@ def init_home(self): for part in ['plugins', 'config_parser', 'optimize', 'mirror/remote']: obd_part_dir = os.path.join(self.OBD_PATH, part) if DirectoryUtil.mkdir(self.OBD_PATH): - root_part_path = os.path.join(self.OBD_INSTALL_PRE, 'usr/obd/', part) + root_part_path = os.path.join(self.OBD_INSTALL_PATH, part) if os.path.exists(root_part_path): DirectoryUtil.copy(root_part_path, obd_part_dir, ROOT_IO) version_fobj.seek(0) @@ -207,7 +206,11 @@ def init_home(self): @property def dev_mode(self): - return COMMAND_ENV.get(ENV_DEV_MODE) == "1" + return COMMAND_ENV.get(ENV.ENV_DEV_MODE) == "1" + + @property + def lock_mode(self): + return COMMAND_ENV.get(ENV.ENV_LOCK_MODE) def parse_command(self): if self.parser.allow_undefine != True: @@ -224,10 +227,12 @@ def do_command(self): DirectoryUtil.mkdir(log_dir) log_path = os.path.join(log_dir, 'obd') ROOT_IO.init_trace_logger(log_path, 'obd', trace_id) - obd = ObdHome(self.OBD_PATH, self.dev_mode, ROOT_IO) + obd = ObdHome(home_path=self.OBD_PATH, dev_mode=self.dev_mode, lock_mode=self.lock_mode, stdio=ROOT_IO) ROOT_IO.track_limit += 1 ROOT_IO.verbose('cmd: %s' % self.cmds) ROOT_IO.verbose('opts: %s' % self.opts) + obd.set_options(self.opts) + obd.set_cmds(self.cmds) ret = self._do_command(obd) if not ret: ROOT_IO.print(DOC_LINK_MSG) @@ -242,8 +247,9 @@ def do_command(self): except: e = sys.exc_info()[1] ROOT_IO.exception('Running Error: %s' % e) - if DEBUG: + if self.has_trace: ROOT_IO.print('Trace ID: %s' % trace_id) + ROOT_IO.print('If you want to view detailed obd logs, please run: obd display-trace %s' % trace_id) return ret def _do_command(self, obd): @@ -283,13 +289,11 @@ def do_command(self): cmd = '%s %s' % (self.prev_cmd, base) ROOT_IO.track_limit += 1 return self.commands[base].init(cmd, args).do_command() - + def register_command(self, command): self.commands[command.name] = command - - class HiddenObdCommand(ObdCommand): def __init__(self, name, summary): @@ -317,7 +321,7 @@ def __init__(self): super(DevModeEnableCommand, self).__init__('enable', 'Enable Dev Mode') def _do_command(self, obd): - if COMMAND_ENV.set(ENV_DEV_MODE, "1", save=True, stdio=obd.stdio): + if COMMAND_ENV.set(ENV.ENV_DEV_MODE, "1", save=True, stdio=obd.stdio): obd.stdio.print("Dev Mode: ON") return True return False @@ -329,7 +333,7 @@ def __init__(self): super(DevModeDisableCommand, self).__init__('disable', 'Disable Dev Mode') def _do_command(self, obd): - if COMMAND_ENV.set(ENV_DEV_MODE, "0", save=True, stdio=obd.stdio): + if COMMAND_ENV.set(ENV.ENV_DEV_MODE, "0", save=True, stdio=obd.stdio): obd.stdio.print("Dev Mode: OFF") return True return False @@ -429,7 +433,7 @@ def init(self, cmd, args): def _do_command(self, obd): if self.cmds: for src in self.cmds: - if not obd.add_mirror(src, self.opts): + if not obd.add_mirror(src): return False return True else: @@ -449,7 +453,7 @@ def __init__(self): self.parser.conflict_handler = 'error' def _do_command(self, obd): - return obd.create_repository(self.opts) + return obd.create_repository() class MirrorListCommand(ObdCommand): @@ -464,8 +468,8 @@ def init(self, cmd, args): def show_pkg(self, name, pkgs): ROOT_IO.print_list( - pkgs, - ['name', 'version', 'release', 'arch', 'md5'], + pkgs, + ['name', 'version', 'release', 'arch', 'md5'], lambda x: [x.name, x.version, x.release, x.arch, x.md5], title='%s Package List' % name ) @@ -493,8 +497,8 @@ def _do_command(self, obd): repos = obd.mirror_manager.get_mirrors(is_enabled=None) ROOT_IO.print_list( repos, - ['SectionName', 'Type', 'Enabled','Update Time'], - lambda x: [x.section_name, x.mirror_type.value, x.enabled, time.strftime("%Y-%m-%d %H:%M", time.localtime(x.repo_age))], + ['SectionName', 'Type', 'Enabled', 'Avaiable' , 'Update Time'], + lambda x: [x.section_name, x.mirror_type.value, x.enabled, x.available, time.strftime("%Y-%m-%d %H:%M", time.localtime(x.repo_age))], title='Mirror Repository List' ) ROOT_IO.print("Use `obd mirror list
` for more details") @@ -505,7 +509,7 @@ class MirrorUpdateCommand(ObdCommand): def __init__(self): super(MirrorUpdateCommand, self).__init__('update', 'Update remote mirror information.') - + def _do_command(self, obd): success = True current = int(time.time()) @@ -527,8 +531,10 @@ def __init__(self): super(MirrorEnableCommand, self).__init__('enable', 'Enable remote mirror repository.') def _do_command(self, obd): - name = self.cmds[0] - return obd.mirror_manager.set_remote_mirror_enabled(name, True) + ret = True + for name in self.cmds: + ret = obd.mirror_manager.set_remote_mirror_enabled(name, True) and ret + return ret class MirrorDisableCommand(ObdCommand): @@ -537,9 +543,20 @@ def __init__(self): super(MirrorDisableCommand, self).__init__('disable', 'Disable remote mirror repository.') def _do_command(self, obd): - name = self.cmds[0] - return obd.mirror_manager.set_remote_mirror_enabled(name, False) + ret = True + for name in self.cmds: + ret = obd.mirror_manager.set_remote_mirror_enabled(name, False) and ret + return ret + +class MirrorAddRepoCommand(ObdCommand): + + def __init__(self): + super(MirrorAddRepoCommand, self).__init__('add-repo', 'Add remote mirror repository file.') + def _do_command(self, obd): + url = self.cmds[0] + return obd.mirror_manager.add_repo(url) + class MirrorMajorCommand(MajorCommand): @@ -551,17 +568,22 @@ def __init__(self): self.register_command(MirrorUpdateCommand()) self.register_command(MirrorEnableCommand()) self.register_command(MirrorDisableCommand()) + self.register_command(MirrorAddRepoCommand()) class RepositoryListCommand(ObdCommand): def __init__(self): super(RepositoryListCommand, self).__init__('list', 'List local repository.') + + @property + def lock_mode(self): + return LockMode.NO_LOCK def show_repo(self, repos, name=None): ROOT_IO.print_list( repos, - ['name', 'version', 'release', 'arch', 'md5', 'tags'], + ['name', 'version', 'release', 'arch', 'md5', 'tags'], lambda x: [x.name, x.version, x.release, x.arch, x.md5, ', '.join(x.tags)], title='%s Local Repository List' % name if name else 'Local Repository List' ) @@ -597,7 +619,7 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.change_deploy_config_style(self.cmds[0], self.opts) + return obd.change_deploy_config_style(self.cmds[0]) else: return self._show_help() @@ -612,7 +634,7 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.check_for_ocp(self.cmds[0], self.opts) + return obd.check_for_ocp(self.cmds[0]) else: return self._show_help() @@ -628,13 +650,34 @@ def __init__(self): self.parser.undefine_warn = False def _do_command(self, obd): - setattr(self.opts, 'mini', True) setattr(self.opts, 'force', True) setattr(self.opts, 'clean', True) setattr(self.opts, 'force', True) setattr(self.opts, 'force_delete', True) - return obd.demo(self.opts) + obd.set_options(self.opts) + return obd.demo() + +class WebCommand(ObdCommand): + + def __init__(self): + super(WebCommand, self).__init__('web', 'Start obd deploy application as web.') + self.parser.add_option('-p', '--port', type='int', help="web server listen port", default=8680) + + def _do_command(self, obd): + from service.app import OBDWeb + ROOT_IO.print('start OBD WEB in 0.0.0.0:%s' % self.opts.port) + ROOT_IO.print('please open http://{0}:{1}'.format(NetUtil.get_host_ip(), self.opts.port)) + try: + COMMAND_ENV.set(ENV.ENV_DISABLE_PARALLER_EXTRACT, True, stdio=obd.stdio) + OBDWeb(obd, self.OBD_INSTALL_PATH).start(self.opts.port) + except KeyboardInterrupt: + ROOT_IO.print('Keyboard Interrupt') + except BaseException as e: + ROOT_IO.exception('Runtime Error %s' % e) + finally: + ROOT_IO.print('stop OBD WEB') + return True class ClusterAutoDeployCommand(ClusterMirrorCommand): @@ -642,7 +685,8 @@ def __init__(self): super(ClusterAutoDeployCommand, self).__init__('autodeploy', 'Deploy a cluster automatically by using a simple configuration file.') self.parser.add_option('-c', '--config', type='string', help="Path to the configuration file.") self.parser.add_option('-f', '--force', action='store_true', help="Force autodeploy, overwrite the home_path.") - self.parser.add_option('-C', '--clean', action='store_true', help="Clean the home path if the directory belong to you.", default=False) + self.parser.add_option('-C', '--clean', action='store_true', help="Clean the home_path if the directory belong to you.", default=False) + self.parser.add_option('--generate-consistent-config', '--gcc', action='store_true', help="Generate consistent config") self.parser.add_option('-U', '--unuselibrepo', '--ulp', action='store_true', help="Disable OBD from installing the libs mirror automatically.") self.parser.add_option('-A', '--auto-create-tenant', '--act', action='store_true', help="Automatically create a tenant named `test` by using all the available resource of the cluster.") self.parser.add_option('--force-delete', action='store_true', help="Force delete, delete the registered cluster.") @@ -652,11 +696,13 @@ def _do_command(self, obd): if self.cmds: if getattr(self.opts, 'force', False) or getattr(self.opts, 'clean', False): setattr(self.opts, 'skip_cluster_status_check', True) + obd.set_options(self.opts) name = self.cmds[0] - if obd.genconfig(name, self.opts): + if obd.genconfig(name): self.opts.config = '' - return obd.deploy_cluster(name, self.opts) and obd.start_cluster(name, self.cmds[1:], self.opts) - return False + obd.set_cmds(self.cmds[1:]) + return obd.deploy_cluster(name) and obd.start_cluster(name) + return False else: return self._show_help() @@ -676,7 +722,8 @@ def _do_command(self, obd): if self.cmds: if getattr(self.opts, 'force', False) or getattr(self.opts, 'clean', False): setattr(self.opts, 'skip_cluster_status_check', True) - return obd.deploy_cluster(self.cmds[0], self.opts) + obd.set_options(self.opts) + return obd.deploy_cluster(self.cmds[0]) else: return self._show_help() @@ -685,15 +732,16 @@ class ClusterStartCommand(ClusterMirrorCommand): def __init__(self): super(ClusterStartCommand, self).__init__('start', 'Start a deployed cluster.') - self.parser.add_option('-s', '--servers', type='string', help="List the started servers. Multiple servers are separated with commas.") - self.parser.add_option('-c', '--components', type='string', help="List the started components. Multiple components are separated with commas.") + self.parser.add_option('-s', '--servers', type='string', help="List of servers to be started. Multiple servers are separated with commas.") + self.parser.add_option('-c', '--components', type='string', help="List of components to be started. Multiple components are separated with commas.") self.parser.add_option('-f', '--force-delete', action='store_true', help="Force delete, delete the registered cluster.") self.parser.add_option('-S', '--strict-check', action='store_true', help="Throw errors instead of warnings when check fails.") self.parser.add_option('--without-parameter', '--wop', action='store_true', help='Start without parameters.') def _do_command(self, obd): if self.cmds: - return obd.start_cluster(self.cmds[0], self.cmds[1:], self.opts) + obd.set_cmds(self.cmds[1:]) + return obd.start_cluster(self.cmds[0]) else: return self._show_help() @@ -702,12 +750,12 @@ class ClusterStopCommand(ClusterMirrorCommand): def __init__(self): super(ClusterStopCommand, self).__init__('stop', 'Stop a started cluster.') - self.parser.add_option('-s', '--servers', type='string', help="List the started servers. Multiple servers are separated with commas.") - self.parser.add_option('-c', '--components', type='string', help="List the stoped components. Multiple components are separated with commas.") + self.parser.add_option('-s', '--servers', type='string', help="List of servers to be stoped. Multiple servers are separated with commas.") + self.parser.add_option('-c', '--components', type='string', help="List of components to be stoped. Multiple components are separated with commas.") def _do_command(self, obd): if self.cmds: - return obd.stop_cluster(self.cmds[0], self.opts) + return obd.stop_cluster(self.cmds[0]) else: return self._show_help() @@ -720,7 +768,7 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.destroy_cluster(self.cmds[0], self.opts) + return obd.destroy_cluster(self.cmds[0]) else: return self._show_help() @@ -741,15 +789,16 @@ class ClusterRestartCommand(ClusterMirrorCommand): def __init__(self): super(ClusterRestartCommand, self).__init__('restart', 'Restart a started cluster.') - self.parser.add_option('-s', '--servers', type='string', help="List the started servers. Multiple servers are separated with commas.") - self.parser.add_option('-c', '--components', type='string', help="List the restarted components. Multiple components are separated with commas.") + self.parser.add_option('-s', '--servers', type='string', help="List of servers to be restarted. Multiple servers are separated with commas.") + self.parser.add_option('-c', '--components', type='string', help="List of components to be restarted. Multiple components are separated with commas.") self.parser.add_option('--with-parameter', '--wp', action='store_true', help='Restart with parameters.') def _do_command(self, obd): if self.cmds: if not getattr(self.opts, 'with_parameter', False): setattr(self.opts, 'without_parameter', True) - return obd.restart_cluster(self.cmds[0], self.opts) + obd.set_options(self.opts) + return obd.restart_cluster(self.cmds[0]) else: return self._show_help() @@ -762,7 +811,7 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.redeploy_cluster(self.cmds[0], self.opts) + return obd.redeploy_cluster(self.cmds[0]) else: return self._show_help() @@ -783,6 +832,10 @@ class ClusterListCommand(ClusterMirrorCommand): def __init__(self): super(ClusterListCommand, self).__init__('list', 'List all the deployments.') + + @property + def lock_mode(self): + return LockMode.NO_LOCK def _do_command(self, obd): if self.cmds: @@ -813,7 +866,7 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.reinstall(self.cmds[0], self.opts) + return obd.reinstall(self.cmds[0]) else: return self._show_help() @@ -827,11 +880,11 @@ def __init__(self): self.parser.add_option('--skip-check', action='store_true', help="Skip all the possible checks.") self.parser.add_option('--usable', type='string', help="Hash list for priority mirrors, separated with `,`.", default='') self.parser.add_option('--disable', type='string', help="Hash list for disabled mirrors, separated with `,`.", default='') - self.parser.add_option('-e', '--executer-path', type='string', help="Executer path.", default=os.path.join(ObdCommand.OBD_INSTALL_PRE, 'usr/obd/lib/executer')) + self.parser.add_option('-e', '--executer-path', type='string', help="Executer path.", default=os.path.join(ObdCommand.OBD_INSTALL_PATH, 'lib/executer')) def _do_command(self, obd): if self.cmds: - return obd.upgrade_cluster(self.cmds[0], self.opts) + return obd.upgrade_cluster(self.cmds[0]) else: return self._show_help() @@ -866,7 +919,7 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.create_tenant(self.cmds[0], self.opts) + return obd.create_tenant(self.cmds[0]) else: return self._show_help() @@ -879,7 +932,19 @@ def __init__(self): def _do_command(self, obd): if self.cmds: - return obd.drop_tenant(self.cmds[0], self.opts) + return obd.drop_tenant(self.cmds[0]) + else: + return self._show_help() + + +class ClusterTenantListCommand(ClusterMirrorCommand): + + def __init__(self): + super(ClusterTenantListCommand, self).__init__('show', 'Show the list of tenant.') + + def _do_command(self, obd): + if self.cmds: + return obd.list_tenant(self.cmds[0]) else: return self._show_help() @@ -887,9 +952,10 @@ def _do_command(self, obd): class ClusterTenantCommand(MajorCommand): def __init__(self): - super(ClusterTenantCommand, self).__init__('tenant', 'Create or drop a tenant.') + super(ClusterTenantCommand, self).__init__('tenant', 'Create, drop or list a tenant.') self.register_command(ClusterTenantCreateCommand()) self.register_command(ClusterTenantDropCommand()) + self.register_command(ClusterTenantListCommand()) class ClusterMajorCommand(MajorCommand): @@ -1014,7 +1080,7 @@ def __init__(self): self.parser.add_option('--rand-type', type='string', help='Random numbers distribution {uniform,gaussian,special,pareto}.') self.parser.add_option('--percentile', type='int', help='Percentile to calculate in latency statistics. Available values are 1-100. 0 means to disable percentile calculations.') self.parser.add_option('--skip-trx', type='string', help='Open or close a transaction in a read-only test. {on/off}') - self.parser.add_option('-O', '--optimization', type='int', help='optimization level {0/1}', default=1) + self.parser.add_option('-O', '--optimization', type='int', help='Optimization level {0/1/2}. [1] 0 - No optimization. 1 - Optimize some of the parameters which do not need to restart servers. 2 - Optimize all the parameters and maybe RESTART SERVERS for better performance.', default=1) self.parser.add_option('-S', '--skip-cluster-status-check', action='store_true', help='Skip cluster status check', default=False) self.parser.add_option('--mysql-ignore-errors', type='string', help='list of errors to ignore, or "all". ', default='1062') @@ -1140,6 +1206,10 @@ def init(self, cmd, args): super(DbConnectCommand, self).init(cmd, args) self.parser.set_usage('%s [options]' % self.prev_cmd) return self + + @property + def lock_mode(self): + return LockMode.NO_LOCK def __init__(self): super(DbConnectCommand, self).__init__('db_connect', 'Establish a database connection to the deployment.') @@ -1165,6 +1235,10 @@ def init(self, cmd, args): super(DoobaCommand, self).init(cmd, args) self.parser.set_usage('%s [options]' % self.prev_cmd) return self + + @property + def lock_mode(self): + return LockMode.NO_LOCK def __init__(self): super(DoobaCommand, self).__init__('dooba', 'A curses powerful tool for OceanBase admin, more than a monitor') @@ -1183,12 +1257,16 @@ def _do_command(self, obd): return self._show_help() -class CommandsCommand(HiddenObdCommand): +class CommandsCommand(ObdCommand): def init(self, cmd, args): super(CommandsCommand, self).init(cmd, args) self.parser.set_usage('%s [options]' % self.prev_cmd) return self + + @property + def lock_mode(self): + return LockMode.NO_LOCK def __init__(self): super(CommandsCommand, self).__init__('command', 'Common tool commands') @@ -1223,26 +1301,63 @@ def __init__(self): super(UpdateCommand, self).__init__('update', 'Update OBD.') def do_command(self): - if os.getuid() != 0: - ROOT_IO.error('To update OBD, you must be a root user.') + uid = os.getuid() + if uid != 0 and not DirectoryUtil.get_owner(self.OBD_INSTALL_PRE): + ROOT_IO.error('To update OBD, you must be the owner of %s.' % self.OBD_INSTALL_PRE) return False return super(UpdateCommand, self).do_command() - + def _do_command(self, obd): return obd.update_obd(VERSION, self.OBD_INSTALL_PRE) +class DisplayTraceCommand(ObdCommand): + + def __init__(self): + super(DisplayTraceCommand, self).__init__('display-trace', 'display trace_id log.') + self.has_trace = False + + @property + def lock_mode(self): + return LockMode.NO_LOCK + + def _do_command(self, obd): + from ssh import LocalClient + if self.cmds: + if obd.stdio.log_path: + log_dir = obd.stdio.log_path + obd.stdio = IO(0, 20) + trace_id = self.cmds[0] + obd._call_stdio('verbose', 'Get log by trace_id') + try: + if UUID(trace_id).version != 1: + obd._call_stdio('critical', '%s is not trace id' % trace_id) + return False + except: + obd._call_stdio('critical', '%s is not trace id' % trace_id) + return False + cmd = 'grep -h "\[{}\]" {}* | sed "s/\[{}\] //g" '.format(trace_id, log_dir, trace_id) + data = LocalClient.execute_command(cmd) + obd.stdio.print(data.stdout) + return True + else: + self._show_help() + return False + + class MainCommand(MajorCommand): def __init__(self): super(MainCommand, self).__init__('obd', '') self.register_command(DevModeMajorCommand()) self.register_command(DemoCommand()) + self.register_command(WebCommand()) self.register_command(MirrorMajorCommand()) self.register_command(ClusterMajorCommand()) self.register_command(RepositoryMajorCommand()) self.register_command(TestMajorCommand()) self.register_command(UpdateCommand()) + self.register_command(DisplayTraceCommand()) self.register_command(EnvironmentMajorCommand()) self.register_command(ToolCommand()) self.parser.version = '''OceanBase Deploy: %s @@ -1264,7 +1379,7 @@ def __init__(self): pass reload(sys) sys.setdefaultencoding(defaultencoding) - sys.path.append(os.path.join(ObdCommand.OBD_INSTALL_PRE, 'usr/obd/lib/site-packages')) + sys.path.append(os.path.join(ObdCommand.OBD_INSTALL_PATH, 'lib/site-packages')) ROOT_IO.track_limit += 2 if MainCommand().init('obd', sys.argv[1:]).do_command(): ROOT_IO.exit(0) diff --git a/_deploy.py b/_deploy.py index 7f51587..9e71381 100644 --- a/_deploy.py +++ b/_deploy.py @@ -23,7 +23,6 @@ import os import re import sys -import pickle import getpass import hashlib from copy import deepcopy @@ -31,9 +30,9 @@ from ruamel.yaml.comments import CommentedMap +import _errno as err from tool import ConfigUtil, FileUtil, YamlLoader, OrderedDict, COMMAND_ENV from _manager import Manager -from _repository import Repository from _stdio import SafeStdio from _environ import ENV_BASE_DIR @@ -360,6 +359,7 @@ def __init__(self, servers, name, version, tag, release, package_hash, parser=No self.origin_package_hash = package_hash self._package_hash = package_hash self._temp_conf = {} + self._all_default_conf = {} self._default_conf = {} self._global_conf = None self._server_conf = {} @@ -371,6 +371,8 @@ def __init__(self, servers, name, version, tag, release, package_hash, parser=No self._include_file = None self._origin_include_file = None self._origin_include_config = None + self._unprocessed_global_conf = None + self._unprocessed_server_conf = {} self._environments = None self._origin_environments = {} self._inner_config = {} @@ -414,7 +416,14 @@ def __eq__(self, other): if not isinstance(other, self.__class__): return False # todo 检查 rsync include等 - return self._global_conf == other._global_conf and self._server_conf == other._server_conf + if self.servers != other.servers: + return False + if self.get_global_conf() != other.get_global_conf(): + return False + for server in self.servers: + if self.get_server_conf(server) != other.get_server_conf(server): + return False + return True def __deepcopy__(self, memo): cluster_config = self.__class__(deepcopy(self.servers), self.name, self.version, self.tag, self.package_hash, self.parser) @@ -451,6 +460,8 @@ def depends(self): def _clear_cache_server(self): for server in self._cache_server: self._cache_server[server] = None + if server in self._unprocessed_server_conf: + del self._unprocessed_server_conf[server] def get_inner_config(self): return self._inner_config @@ -485,11 +496,14 @@ def get_depend_servers(self, name): cluster_config = self._depends[name] return deepcopy(cluster_config.original_servers) - def get_depend_config(self, name, server=None): + def get_depend_config(self, name, server=None, with_default=True): if name not in self._depends: return None cluster_config = self._depends[name] - config = cluster_config.get_server_conf_with_default(server) if server else cluster_config.get_global_conf() + if with_default: + config = cluster_config.get_server_conf_with_default(server) if server else cluster_config.get_global_conf_with_default() + else: + config = cluster_config.get_server_conf(server) if server else cluster_config.get_global_conf() return deepcopy(config) def update_server_conf(self, server, key, value, save=True): @@ -514,15 +528,13 @@ def update_global_conf(self, key, value, save=True): if not self._deploy_config.update_component_global_conf(self.name, key, value, save): return False self._update_global_conf(key, value) - for server in self._cache_server: - if self._cache_server[server] is not None: - self._cache_server[server][key] = value return True def _update_global_conf(self, key, value): self._original_global_conf[key] = value - if self._global_conf: - self._global_conf[key] = value + self._global_conf = None + self._unprocessed_global_conf = None + self._clear_cache_server() def update_rsync_list(self, rsync_list, save=True): if self._deploy_config is None: @@ -541,11 +553,13 @@ def update_environments(self, environments, save=True): self._environments = None return True - def get_unconfigured_require_item(self, server): + def get_unconfigured_require_item(self, server, skip_keys=[]): items = [] - config = self.get_server_conf(server) + config = self._get_unprocessed_server_conf(server) if config is not None: for key in self._temp_conf: + if key in skip_keys: + continue if not self._temp_conf[key].require: continue if key in config: @@ -556,11 +570,10 @@ def get_unconfigured_require_item(self, server): def get_server_conf_with_default(self, server): if server not in self._server_conf: return None - config = {} - for key in self._temp_conf: - if self._temp_conf[key].default is not None: - config[key] = self._temp_conf[key].default - config.update(self.get_server_conf(server)) + config = deepcopy(self._all_default_conf) + server_config = self.get_server_conf(server) + if server_config: + config.update(server_config) return config def get_need_redeploy_items(self, server): @@ -585,11 +598,15 @@ def get_need_restart_items(self, server): def update_temp_conf(self, temp_conf): self._default_conf = {} + self._all_default_conf = {} self._temp_conf = temp_conf for key in self._temp_conf: if self._temp_conf[key].require and self._temp_conf[key].default is not None: self._default_conf[key] = self._temp_conf[key].default + if self._temp_conf[key].default is not None: + self._all_default_conf[key] = self._temp_conf[key].default self._global_conf = None + self._unprocessed_global_conf = None self._clear_cache_server() def _apply_temp_conf(self, conf): @@ -606,23 +623,44 @@ def get_temp_conf_item(self, key): return None def check_param(self): - error = [] + errors = [] + if self._temp_conf: + _, g_errs = self.global_check_param() + errors += g_errs + for server in self._server_conf: + s_errs, _ = self._check_param(self._server_conf[server]) + errors += s_errs + return not errors, set(errors) + + def global_check_param(self): + errors = [] + if self._temp_conf: + errors, _ = self._check_param(self._get_unprocessed_global_conf()) + return not errors, errors + + def servers_check_param(self): + check_res = {} if self._temp_conf: - error += self._check_param(self.get_global_conf()) + global_config = self._get_unprocessed_global_conf() for server in self._server_conf: - error += self._check_param(self._server_conf[server]) - return not error, set(error) + config = deepcopy(self._server_conf[server]) + config.update(global_config) + errors, items = self._check_param(config) + check_res[server] = {'errors': errors, 'items': items} + return check_res def _check_param(self, config): - error = [] + errors = [] + items = [] for key in config: item = self._temp_conf.get(key) if item: try: item.check_value(config[key]) except Exception as e: - error.append(str(e)) - return error + errors.append(str(e)) + items.append(item) + return errors, items def set_global_conf(self, conf): if not isinstance(conf, dict): @@ -652,15 +690,24 @@ def add_server_conf(self, server, conf): self._server_conf[server] = conf self._cache_server[server] = None + def _get_unprocessed_global_conf(self): + if self._unprocessed_global_conf is None: + self._unprocessed_global_conf = deepcopy(self._default_conf) + self._unprocessed_global_conf.update(self._get_include_config('config', {})) + if self._original_global_conf: + self._unprocessed_global_conf.update(self._original_global_conf) + return self._unprocessed_global_conf + def get_global_conf(self): if self._global_conf is None: - self._global_conf = deepcopy(self._default_conf) - self._global_conf.update(self._get_include_config('config', {})) - if self._original_global_conf: - self._global_conf.update(self._original_global_conf) - self._global_conf = self._apply_temp_conf(self._global_conf) + self._global_conf = self._apply_temp_conf(self._get_unprocessed_global_conf()) return self._global_conf + def get_global_conf_with_default(self): + config = deepcopy(self._all_default_conf) + config.update(self.get_global_conf()) + return config + def _add_base_dir(self, path): if not os.path.isabs(path): if self._base_dir: @@ -758,22 +805,32 @@ def get_environments(self): self._environments.update(self._origin_environments) return self._environments + def _get_unprocessed_server_conf(self, server): + if server not in self._unprocessed_server_conf: + conf = deepcopy(self._inner_config.get(server.name, {})) + conf.update(self._get_unprocessed_global_conf()) + conf.update(self._server_conf[server]) + self._unprocessed_server_conf[server] = conf + return self._unprocessed_server_conf[server] + def get_server_conf(self, server): if server not in self._server_conf: return None if self._cache_server[server] is None: - conf = self._apply_temp_conf(deepcopy(self._inner_config.get(server.name, {}))) - conf.update(self.get_global_conf()) - conf.update(self._apply_temp_conf(self._server_conf[server])) - self._cache_server[server] = conf + self._cache_server[server] = self._apply_temp_conf(self._get_unprocessed_server_conf(server)) return self._cache_server[server] def get_original_global_conf(self): - return self._original_global_conf + return deepcopy(self._original_global_conf) def get_original_server_conf(self, server): return self._server_conf.get(server) + def get_original_server_conf_with_global(self, server): + config = self.get_original_global_conf() + config.update(self._server_conf.get(server, {})) + return config + class DeployStatus(Enum): diff --git a/_environ.py b/_environ.py index a544ac8..8d022d0 100644 --- a/_environ.py +++ b/_environ.py @@ -23,6 +23,9 @@ # obd dev mode. {0/1} ENV_DEV_MODE = "OBD_DEV_MODE" +# obd lock mode. 0 - No lock mode, 1 - The deploy lock wiil be downgraded to shared lock, 2 - Default lock mode. +ENV_LOCK_MODE = "OBD_LOCK_MODE" + # base path which will be used by runtime dependencies sync and include config. {absolute path style} ENV_BASE_DIR = "OBD_DEPLOY_BASE_DIR" @@ -31,3 +34,5 @@ # disable rsync mode even if the rsync exists. {0/1} ENV_DISABLE_RSYNC = "OBD_DISABLE_RSYNC" + +ENV_DISABLE_PARALLER_EXTRACT = "OBD_DISALBE_PARALLER_EXTRACT" diff --git a/_errno.py b/_errno.py index b1c09e2..4c92c32 100644 --- a/_errno.py +++ b/_errno.py @@ -29,16 +29,76 @@ class LockError(Exception): class OBDErrorCode(object): + def __init__(self, code, msg): + self.code = code + self.msg = msg + + def __str__(self): + return self.msg + + +class OBDErrorCodeTemplate(object): + def __init__(self, code, msg): self.code = code self.msg = msg self._str_ = ('OBD-%04d: ' % code) + msg def format(self, *args, **kwargs): - return self._str_.format(*args, **kwargs) + return OBDErrorCode( + self.code, + self._str_.format(*args, **kwargs), + ) def __str__(self): - return self._str_ + return self.msg + + +class FixEval(object): + + DEL = 0 + SET = 1 + + def __init__(self, operation, key, value=None, is_global=False): + self.operation = operation + self.key = key + self.value = value + self.is_global = is_global + +class OBDErrorSuggestion(object): + + def __init__(self, msg, auto_fix=False, fix_eval=[]): + self.msg = msg + self.auto_fix = auto_fix + self.fix_eval = fix_eval + + +class OBDErrorSuggestionTemplate(object): + + def __init__(self, msg, auto_fix=False, fix_eval=[]): + self._msg = msg + self.auto_fix = auto_fix + self.fix_eval = fix_eval if isinstance(fix_eval, list) else [fix_eval] + + def format(self, *args, **kwargs): + return OBDErrorSuggestion( + self._msg.format(*args, **kwargs), + auto_fix=kwargs.get('auto_fix', self.auto_fix), + fix_eval=kwargs.get('fix_eval', self.fix_eval) + ) + + +class CheckStatus(object): + + FAIL = "FAIL" + PASS = "PASS" + WAIT = "WAIT" + + def __init__(self, status=WAIT, error=None, suggests=[]): + self.status = status + self.error = error + self.suggests = suggests + class InitDirFailedErrorMessage(object): @@ -46,36 +106,119 @@ class InitDirFailedErrorMessage(object): PATH_ONLY = ': {path}.' NOT_EMPTY = ': {path} is not empty.' CREATE_FAILED = ': create {path} failed.' + NOT_DIR = ': {path} is not a directory .' PERMISSION_DENIED = ': {path} permission denied .' DOC_LINK = '' DOC_LINK_MSG = 'See {}'.format(DOC_LINK if DOC_LINK else "https://www.oceanbase.com/product/ob-deployer/error-codes .") -EC_CONFIG_CONFLICT_PORT = OBDErrorCode(1000, 'Configuration conflict {server1}:{port} port is used for {server2}\'s {key}') -EC_CONFLICT_PORT = OBDErrorCode(1001, '{server}:{port} port is already used') -EC_FAIL_TO_INIT_PATH = OBDErrorCode(1002, 'Fail to init {server} {key}{msg}') -EC_CLEAN_PATH_FAILED = OBDErrorCode(1003, 'Fail to clean {server}:{path}') -EC_CONFIG_CONFLICT_DIR = OBDErrorCode(1004, 'Configuration conflict {server1}: {path} is used for {server2}\'s {key}') -EC_SOME_SERVER_STOPED = OBDErrorCode(1005, 'Some of the servers in the cluster have been stopped') -EC_FAIL_TO_CONNECT = OBDErrorCode(1006, 'Failed to connect to {component}') -EC_ULIMIT_CHECK = OBDErrorCode(1007, '({server}) {key} must not be less than {need} (Current value: {now})') - -EC_OBSERVER_NOT_ENOUGH_MEMORY = OBDErrorCode(2000, '({ip}) not enough memory. (Free: {free}, Need: {need})') -EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE = OBDErrorCode(2000, '({ip}) not enough memory. (Available: {available}, Need: {need})') -EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED = OBDErrorCode(2000, '({ip}) not enough memory. (Free: {free}, Buff/Cache: {cached}, Need: {need})') -EC_OBSERVER_CAN_NOT_MIGRATE_IN = OBDErrorCode(2001, 'server can not migrate in') -EC_OBSERVER_FAIL_TO_START = OBDErrorCode(2002, 'Failed to start {server} observer') -EC_OBSERVER_NOT_ENOUGH_DISK_4_CLOG = OBDErrorCode(2003, '({ip}) {path} not enough disk space for clog. Use redo_dir to set other disk for clog, or reduce the value of datafile_size') -EC_OBSERVER_INVALID_MODFILY_GLOBAL_KEY = OBDErrorCode(2004, 'Invalid: {key} is not a single server configuration item') - -EC_MYSQLTEST_PARSE_CMD_FAILED = OBDErrorCode(3000, 'parse cmd failed: {path}') -EC_MYSQLTEST_FAILE_NOT_FOUND = OBDErrorCode(3001, '{file} not found in {path}') -EC_TPCC_LOAD_DATA_FAILED = OBDErrorCode(3002, 'Failed to load data.') -EC_TPCC_RUN_TEST_FAILED = OBDErrorCode(3003, 'Failed to run TPC-C benchmark.') - -EC_OBAGENT_RELOAD_FAILED = OBDErrorCode(4000, 'Fail to reload {server}') -EC_OBAGENT_SEND_CONFIG_FAILED = OBDErrorCode(4001, 'Fail to send config file to {server}') +EC_CONFIG_CONFLICT_PORT = OBDErrorCodeTemplate(1000, 'Configuration conflict {server1}:{port} port is used for {server2}\'s {key}') +EC_CONFLICT_PORT = OBDErrorCodeTemplate(1001, '{server}:{port} port is already used') +EC_FAIL_TO_INIT_PATH = OBDErrorCodeTemplate(1002, 'Fail to init {server} {key}{msg}') +EC_CLEAN_PATH_FAILED = OBDErrorCodeTemplate(1003, 'Fail to clean {server}:{path}') +EC_CONFIG_CONFLICT_DIR = OBDErrorCodeTemplate(1004, 'Configuration conflict {server1}: {path} is used for {server2}\'s {key}') +EC_SOME_SERVER_STOPED = OBDErrorCodeTemplate(1005, 'Some of the servers in the cluster have been stopped') +EC_FAIL_TO_CONNECT = OBDErrorCodeTemplate(1006, 'Failed to connect to {component}') +EC_ULIMIT_CHECK = OBDErrorCodeTemplate(1007, '({server}) {key} must not be less than {need} (Current value: {now})') +EC_FAILED_TO_GET_AIO_NR = OBDErrorCodeTemplate(1008, '({ip}) failed to get fs.aio-max-nr and fs.aio-nr') +EC_NEED_CONFIG = OBDErrorCodeTemplate(1009, '{server} {component} need config: {miss_keys}') +EC_NO_SUCH_NET_DEVICE = OBDErrorCodeTemplate(1010, '{server} No such net interface: {devname}') +EC_AIO_NOT_ENOUGH = OBDErrorCodeTemplate(1011, '({ip}) Insufficient AIO remaining (Avail: {avail}, Need: {need}), The recommended value of fs.aio-max-nr is 1048576') +EC_PARAM_CHECK = OBDErrorCodeTemplate(1012, '{errors}') +EC_SSH_CONNECT = OBDErrorCodeTemplate(1013, '{user}@{ip} connect failed: {message}') + +# error code for observer +EC_OBSERVER_NOT_ENOUGH_MEMORY = OBDErrorCodeTemplate(2000, '({ip}) not enough memory. (Free: {free}, Need: {need})') +EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE = OBDErrorCodeTemplate(2000, '({ip}) not enough memory. (Available: {available}, Need: {need})') +EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED = OBDErrorCodeTemplate(2000, '({ip}) not enough memory. (Free: {free}, Buff/Cache: {cached}, Need: {need})') +EC_OBSERVER_CAN_NOT_MIGRATE_IN = OBDErrorCodeTemplate(2001, 'server can not migrate in') +EC_OBSERVER_FAIL_TO_START = OBDErrorCodeTemplate(2002, 'Failed to start {server} observer') +EC_OBSERVER_FAIL_TO_START_WITH_ERR = OBDErrorCodeTemplate(2002, 'Failed to start {server} observer: {stderr}') +EC_OBSERVER_NOT_ENOUGH_DISK = OBDErrorCodeTemplate(2003, '({ip}) {disk} not enough disk space. (Avail: {avail}, Need: {need})') +EC_OBSERVER_NOT_ENOUGH_DISK_4_CLOG = OBDErrorCodeTemplate(2003, '({ip}) {path} not enough disk space for clog. Use redo_dir to set other disk for clog, or reduce the value of datafile_size') +EC_OBSERVER_INVALID_MODFILY_GLOBAL_KEY = OBDErrorCodeTemplate(2004, 'Invalid: {key} is not a single server configuration item') +EC_OBSERVER_FAILED_TO_REGISTER = OBDErrorCodeTemplate(2005, 'Failed to register cluster.') +EC_OBSERVER_FAILED_TO_REGISTER_WITH_DETAILS = OBDErrorCodeTemplate(2005, 'Failed to register cluster. {appname} may have been registered in {obconfig_url}.') +EC_OBSERVER_MULTI_NET_DEVICE = OBDErrorCodeTemplate(2006, '{ip} has more than one network interface. Please set `devname` for ({server})') +EC_OBSERVER_PING_FAILED = OBDErrorCodeTemplate(2007, '{ip1} {devname} fail to ping {ip2}. Please check configuration `devname`') +EC_OBSERVER_TIME_OUT_OF_SYNC = OBDErrorCodeTemplate(2008, 'Cluster clocks are out of sync') +EC_OBSERVER_PRODUCTION_MODE_LIMIT = OBDErrorCodeTemplate(2009, '({server}): when production_mode is True, {key} can not be less then {limit}') +EC_OBSERVER_SYS_MEM_TOO_LARGE = OBDErrorCodeTemplate(2010, '({server}): system_memory too large. system_memory must be less than memory_limit/memory_limit_percentage.') +EC_OBSERVER_GET_MEMINFO_FAIL = OBDErrorCodeTemplate(2011, "{server}: fail to get memory info.\nPlease configure 'memory_limit' manually in configuration file") + +# error code for test commands +EC_MYSQLTEST_PARSE_CMD_FAILED = OBDErrorCodeTemplate(3000, 'parse cmd failed: {path}') +EC_MYSQLTEST_FAILE_NOT_FOUND = OBDErrorCodeTemplate(3001, '{file} not found in {path}') +EC_TPCC_LOAD_DATA_FAILED = OBDErrorCodeTemplate(3002, 'Failed to load data.') +EC_TPCC_RUN_TEST_FAILED = OBDErrorCodeTemplate(3003, 'Failed to run TPC-C benchmark.') + +# error code for other components. +# obagent +EC_OBAGENT_RELOAD_FAILED = OBDErrorCodeTemplate(4000, 'Fail to reload {server}') +EC_OBAGENT_SEND_CONFIG_FAILED = OBDErrorCodeTemplate(4001, 'Fail to send config file to {server}') +# obproxy +EC_OBPROXY_NEED_CONFIG = OBDErrorCodeTemplate(4100, '{server} need config "rs_list" or "obproxy_config_server_url"') +EC_OBPROXY_START_FAILED = OBDErrorCodeTemplate(4101, 'failed to start {server} obproxy: {stderr}') +# grafana +EC_GRAFANA_DEFAULT_PWD = OBDErrorCodeTemplate(4200, "{server} grafana admin password should not be 'admin'") +EC_GRAFANA_PWD_LESS_5 = OBDErrorCodeTemplate(4201, "{server} grafana admin password length should not be less than 5") +# ocp express +EC_OCP_EXPRESS_JAVA_NOT_FOUND = OBDErrorCodeTemplate(4300, "{server}: failed to query java version, you may not have java installed") +EC_OCP_EXPRESS_JAVA_VERSION_ERROR = OBDErrorCodeTemplate(4301, "{server}: ocp-express need java with version {version}") +EC_OCP_EXPRESS_NOT_ENOUGH_MEMORY = OBDErrorCodeTemplate(4302, '({ip}) not enough memory. (Free: {free}, Need: {need})') +EC_OCP_EXPRESS_NOT_ENOUGH_MEMORY_AVAILABLE = OBDErrorCodeTemplate(4302, '({ip}) not enough memory. (Available: {available}, Need: {need})') +EC_OCP_EXPRESS_NOT_ENOUGH_MEMORY_CACHED = OBDErrorCodeTemplate(4302, '({ip}) not enough memory. (Free: {free}, Buff/Cache: {cached}, Need: {need})') +EC_OCP_EXPRESS_NOT_ENOUGH_DISK = OBDErrorCodeTemplate(4303, '({ip}) {disk} not enough disk space. (Avail: {avail}, Need: {need})') +EC_OCP_EXPRESS_DEPENDS_COMP_VERSION = OBDErrorCodeTemplate(4304, 'OCP express {ocp_express_version} needs to use {comp} with version {comp_version} or above') +EC_OCP_EXPRESS_META_DB_NOT_ENOUGH_LOG_DISK_AVAILABLE = OBDErrorCodeTemplate(4305, 'There is not enough log disk for ocp meta tenant. (Avail: {avail}, Need: {need})') +EC_OCP_EXPRESS_META_DB_NOT_ENOUGH_LOG_DISK = OBDErrorCodeTemplate(4305, 'There is not enough log disk for ocp meta tenant.') +EC_OCP_EXPRESS_META_DB_NOT_ENOUGH_MEM = OBDErrorCodeTemplate(4305, 'There is not enough memory for ocp meta tenant') +# sql +EC_SQL_EXECUTE_FAILED = OBDErrorCodeTemplate(5000, "{sql} execute failed") # WARN CODE -WC_ULIMIT_CHECK = OBDErrorCode(1007, '({server}) The recommended number of {key} is {need} (Current value: {now})') \ No newline at end of file +WC_ULIMIT_CHECK = OBDErrorCodeTemplate(1007, '({server}) The recommended number of {key} is {need} (Current value: {now})') +WC_AIO_NOT_ENOUGH = OBDErrorCodeTemplate(1011, '({ip}) The recommended value of fs.aio-max-nr is 1048576 (Current value: {current})') +WC_OBSERVER_SAME_DISK = OBDErrorCodeTemplate(1012, '({ip}) clog and data use the same disk ({disk})') +WC_OBSERVER_SYS_MEM_TOO_LARGE = OBDErrorCodeTemplate(2010, '({server}): system_memory too large. system_memory should be less than {factor} * memory_limit/memory_limit_percentage.') +WC_OCP_EXPRESS_FAILED_TO_GET_DISK_INFO = OBDErrorCodeTemplate(4303, '({ip}) failed to get disk information, skip disk space check') + +# SUGGESTION for ERROR +SUG_SET_CONFIG = OBDErrorSuggestionTemplate('Please set config {key} correctly') +SUG_INCREASE_CONFIG = OBDErrorSuggestionTemplate('Please increase the {key} in configuration') +SUG_DECREASE_CONFIG = OBDErrorSuggestionTemplate('Please decrease the {key} in configuration') +SUG_PORT_CONFLICTS = OBDErrorSuggestionTemplate('Please adjust the configuration to avoid port conflicts') +SUG_USE_OTHER_PORT = OBDErrorSuggestionTemplate('Please choose another unoccupied port or terminate the process occupying the port') +SUG_NO_SUCH_NET_DEVIC = OBDErrorSuggestionTemplate('Please set the network interface corresponding to {ip} to `devname`', fix_eval=[FixEval(FixEval.DEL, 'devname')]) +SUG_CONFIG_CONFLICT_DIR = OBDErrorSuggestionTemplate('Please specify a new `{key}` for the {server}') +SUG_CONFIRM_OS = OBDErrorSuggestionTemplate('Please confirm whether the deployment node is a compatible operating system') +SUG_SPECIFY_PATH = OBDErrorSuggestionTemplate('Please specify the path again') +SUG_SET_DEVICE = OBDErrorSuggestionTemplate('Please set the correct network device name to devname') +SUG_USE_SEPARATE_DISKS = OBDErrorSuggestionTemplate('Please use separate disks for redo_dir and data_dir') +SUG_USE_ANOTHER_DEVICE = OBDErrorSuggestionTemplate('Please specify {dir} to another disk with enough space') +SUB_SET_NO_PRODUCTION_MODE = OBDErrorSuggestionTemplate('Please set production_mode to false', True, [FixEval(FixEval.SET, 'production_mode', False)]) +SUG_CONFIRM_CONFIG_SERVER = OBDErrorSuggestionTemplate('Please confirm that the ob config service is running normally and that obproxy_config_server_url can be connected correctly'), +SUG_USE_RS_LIST = OBDErrorSuggestionTemplate('Instead of using ob config service, please use rs_list configuration in obproxy to proxy observer') +SUG_GRAFANA_PWD = OBDErrorSuggestionTemplate('Grafana password length must be greater than 4 and not "admin"', True, [FixEval(FixEval.DEL, 'login_password', is_global=True)]) +SUG_PARAM_CHECK = OBDErrorSuggestionTemplate('Please check your config') +SUG_SSH_FAILED = OBDErrorSuggestionTemplate('Please check user config and network') +SUG_SYSCTL = OBDErrorSuggestionTemplate('Please execute `echo ‘{var}={value}’ >> /etc/sysctl.conf; sysctl -p` as root in {ip}.') +SUG_ULIMIT = OBDErrorSuggestionTemplate('Please execute `echo -e "* soft {name} {value}\\n* hard {name} {value}" >> /etc/security/limits.d/{name}.conf` as root in {ip}. if it dosen\'t work, please check whether UsePAM is yes in /etc/ssh/sshd_config.') +SUG_CONNECT_EXCEPT = OBDErrorSuggestionTemplate('Connection exception or unsupported OS. Please retry or contact us.') +SUG_UNSUPPORT_OS = OBDErrorSuggestionTemplate('It may be an unsupported OS, please contact us for assistance') +SUG_OBSERVER_SYS_MEM_TOO_LARGE = OBDErrorSuggestionTemplate('`system_memory` should be less than {factor} * memory_limit/memory_limit_percentage.', fix_eval=[FixEval(FixEval.DEL, 'system_memory')]) +SUG_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE = OBDErrorSuggestionTemplate('Please execute `echo 1 > /proc/sys/vm/drop_caches` as root in {ip} to rlease cached.') +SUG_OBSERVER_REDUCE_MEM = OBDErrorSuggestionTemplate('Please reduce the `memory_limit` or `memory_limit_percentage`', fix_eval=[FixEval(FixEval.DEL, 'memory_limit'), FixEval(FixEval.DEL, 'system_memory'), FixEval(FixEval.DEL, 'memory_limit_percentage')]) +SUG_OBSERVER_SAME_DISK = OBDErrorSuggestionTemplate('Configure `redo_dir` and `data_dir` to different disks') +SUG_OBSERVER_NOT_ENOUGH_DISK = OBDErrorSuggestionTemplate('Please reduce the `datafile_size` or `datafile_disk_percentage`', fix_eval=[FixEval(FixEval.DEL, 'datafile_size'), FixEval(FixEval.DEL, 'datafile_disk_percentage')]) +SUG_OBSERVER_REDUCE_REDO = OBDErrorSuggestionTemplate('Please reduce the `log_disk_size` or `log_disk_percentage`', fix_eval=[FixEval(FixEval.DEL, 'log_disk_size'), FixEval(FixEval.DEL, 'log_disk_percentage')]) +SUG_OBSERVER_NOT_ENOUGH_DISK_4_CLOG = OBDErrorSuggestionTemplate('Please increase the `clog_disk_utilization_threshold` and `clog_disk_usage_limit_percentage`', fix_eval=[FixEval(FixEval.DEL, 'clog_disk_utilization_threshold'), FixEval(FixEval.DEL, 'clog_disk_usage_limit_percentage')]) +SUG_OBSERVER_TIME_OUT_OF_SYNC = OBDErrorSuggestionTemplate('Please enable clock synchronization service') +SUG_OCP_EXPRESS_INSTALL_JAVA_WITH_VERSION = OBDErrorSuggestionTemplate('Please install java with version {version}. If java is already installed, please set `java_bin` to the expected java binary path') +SUG_OCP_EXPRESS_NOT_ENOUGH_MEMORY_AVALIABLE = OBDErrorSuggestionTemplate('Please execute `echo 1 > /proc/sys/vm/drop_caches` as root in {ip} to rlease cached.') +SUG_OCP_EXPRESS_REDUCE_MEM = OBDErrorSuggestionTemplate('Please reduce the `memory_size`', fix_eval=[FixEval(FixEval.DEL, 'memory_size')]) +SUG_OCP_EXPRESS_REDUCE_DISK = OBDErrorSuggestionTemplate('Please reduce the `logging_file_total_size_cap`', fix_eval=[FixEval(FixEval.DEL, 'logging_file_total_size_cap')]) +SUG_OCP_EXPRESS_COMP_VERSION = OBDErrorSuggestionTemplate('Please use {comp} with version {version} or above') +SUG_OCP_EXPRESS_REDUCE_META_DB_MEM = OBDErrorSuggestionTemplate('Please reduce the `ocp_meta_tenant_memory_size`', fix_eval=[FixEval(FixEval.DEL, 'ocp_meta_tenant_memory_size')]) +SUG_OCP_EXPRESS_REDUCE_META_DB_LOG_DISK = OBDErrorSuggestionTemplate('Please reduce the `ocp_meta_tenant_log_disk_size`', fix_eval=[FixEval(FixEval.DEL, 'ocp_meta_tenant_log_disk_size')]) \ No newline at end of file diff --git a/_lock.py b/_lock.py index a15de07..19e33fd 100644 --- a/_lock.py +++ b/_lock.py @@ -188,6 +188,12 @@ def unlock(self): self.mix_lock.ex_unlock() +class LockMode(Enum): + NO_LOCK = 0 + DEPLOY_SHARED_LOCK = 1 + DEFAULT = 2 + + class LockManager(Manager): TRY_TIMES = 6000 @@ -198,12 +204,13 @@ class LockManager(Manager): MIR_REPO_FN = LockType.MIR_REPO.value DEPLOY_FN_PERFIX = LockType.DEPLOY.value LOCKS = {} - + def __init__(self, home_path, stdio=None): super(LockManager, self).__init__(home_path, stdio) self.locks = [] self.global_path = os.path.join(self.path, self.GLOBAL_FN) self.mir_repo_path = os.path.join(self.path, self.MIR_REPO_FN) + self.mode = LockMode.DEFAULT @staticmethod def set_try_times(try_times): @@ -226,14 +233,26 @@ def _get_mix_lock(self, path): @classmethod def shutdown(cls): for path in cls.LOCKS: - cls.LOCKS[path] = None + cls.LOCKS[path]._unlock() cls.LOCKS = None + def set_lock_mode(self, mode): + for key in LockMode: + if key.value == mode: + mode = key + break + if not isinstance(mode, LockMode) or mode not in LockMode: + getattr(self.stdio, 'verbose', print)('unknown lock mode {}'.format(mode)) + return + self.stdio and getattr(self.stdio, 'verbose', print)('set lock mode to {}({})'.format(mode.name, mode.value)) + self.mode = mode + def _lock(self, path, clz): - mix_lock = self._get_mix_lock(path) - lock = clz(mix_lock) - lock.lock() - self.locks.append(lock) + if self.mode != LockMode.NO_LOCK: + mix_lock = self._get_mix_lock(path) + lock = clz(mix_lock) + lock.lock() + self.locks.append(lock) return True def _sh_lock(self, path): @@ -258,7 +277,10 @@ def _deploy_lock_fp(self, deploy_name): return os.path.join(self.path, '%s_%s' % (self.DEPLOY_FN_PERFIX, deploy_name)) def deploy_ex_lock(self, deploy_name): - return self._ex_lock(self._deploy_lock_fp(deploy_name)) + if self.mode == LockMode.DEPLOY_SHARED_LOCK: + return self._sh_lock(self._deploy_lock_fp(deploy_name)) + else: + return self._ex_lock(self._deploy_lock_fp(deploy_name)) def deploy_sh_lock(self, deploy_name): return self._sh_lock(self._deploy_lock_fp(deploy_name)) diff --git a/_mirror.py b/_mirror.py index a1ed7ff..b3c41cf 100644 --- a/_mirror.py +++ b/_mirror.py @@ -24,6 +24,7 @@ import re import os import sys +import tempfile import time import pickle import string @@ -33,6 +34,7 @@ from enum import Enum from copy import deepcopy from xml.etree import cElementTree +from ssh import LocalClient try: from ConfigParser import ConfigParser except: @@ -257,6 +259,7 @@ def __init__(self, mirror_path, meta_data, stdio=None): self.gpgcheck = False self._db = None self._repomds = None + self._available = None super(RemoteMirrorRepository, self).__init__(mirror_path, stdio=stdio) self.section_name = meta_data['section_name'] self.baseurl = meta_data['baseurl'] @@ -270,6 +273,17 @@ def __init__(self, mirror_path, meta_data, stdio=None): if repo_age > self.repo_age or int(time.time()) - 86400 > self.repo_age: self.repo_age = repo_age self.update_mirror() + + @property + def available(self): + if self._available is None: + try: + req = requests.request('get', self.baseurl) + self._available = req.status_code < 400 + except Exception: + self.stdio and getattr(self.stdio, 'exception', print)('') + self._available = False + return self._available @property def db(self): @@ -384,16 +398,19 @@ def update_mirror(self): self.get_repomds(True) primary_repomd = self._get_repomd_by_type(self.PRIMARY_REPOMD_TYPE) if not primary_repomd: + self._available = False self.stdio and getattr(self.stdio, 'stop_loading')('fail') return False file_path = self._get_repomd_data_file(primary_repomd) if not file_path: + self._available = False self.stdio and getattr(self.stdio, 'stop_loading')('fail') return False self._db = None self.repo_age = int(time.time()) self._dump_repo_age_data() self.stdio and getattr(self.stdio, 'stop_loading')('succeed') + self._available = True return True def get_repomds(self, update=False): @@ -573,7 +590,8 @@ def download_file(url, save_path, stdio=None): return True except: FileUtil.rm(save_path) - stdio and getattr(stdio, 'exception', print)('Failed to download %s to %s' % (url, save_path)) + stdio and getattr(stdio, 'warn', print)('Failed to download %s to %s' % (url, save_path)) + stdio and getattr(stdio, 'exception', print)('') return False class LocalMirrorRepository(MirrorRepository): @@ -586,6 +604,7 @@ def __init__(self, mirror_path, stdio=None): self.db = {} self.db_path = os.path.join(mirror_path, self._DB_FILE) self.enabled = '-' + self.available = True self._load_db() @property @@ -1050,3 +1069,46 @@ def set_remote_mirror_enabled(self, section_name, enabled=True): mirror_section.meta_data['repo_age'] = repo_age self.stdio and getattr(self.stdio, 'stop_loading')('succeed') return True + + def add_repo(self, url): + self._lock() + download_file_save_name = url.split('/')[-1] + if not download_file_save_name.endswith(".repo"): + self.stdio.error("Can't download. Please use a file in .repo format.") + return False + + download_file_save_path = os.path.join(self.remote_path, download_file_save_name) + + if os.path.exists(download_file_save_path): + if not self.stdio.confirm("the repo file you want to add already exists, overwrite it?"): + self.stdio.print("exit without any changes") + return True + + try: + download_file_res = requests.get(url, timeout=(5, 5)) + except Exception as e: + self.stdio.exception("Failed to download repository file") + return False + + download_status_code = download_file_res.status_code + + if download_status_code != 200: + self.stdio.verbose("http code: {}, http body: {}".format(download_status_code, download_file_res.text)) + self.stdio.error("Failed to download repository file") + return False + + try: + with tempfile.NamedTemporaryFile(mode='w+', suffix='.repo') as tf: + tf.write(download_file_res.content.decode(encoding='utf8')) + tf.seek(0) + ConfigParser().readfp(tf) + tf.seek(0) + if LocalClient.put_file(tf.name, download_file_save_path, stdio=self.stdio): + self.stdio.print("repo file saved to {}".format(download_file_save_path)) + return True + else: + self.stdio.error("Failed to save repository file") + return False + except Exception as e: + self.stdio.exception("Failed to save repository file") + return False diff --git a/_plugin.py b/_plugin.py index 7d714a2..0814944 100644 --- a/_plugin.py +++ b/_plugin.py @@ -25,7 +25,7 @@ import sys from enum import Enum from glob import glob -from copy import deepcopy +from copy import deepcopy, copy from _manager import Manager from _rpm import Version @@ -47,7 +47,7 @@ class PluginType(Enum): class Plugin(object): - + PLUGIN_TYPE = None FLAG_FILE = None @@ -67,6 +67,33 @@ def mirror_type(self): return self.PLUGIN_TYPE +class PluginContextNamespace: + + def __init__(self, spacename): + self.spacename = spacename + self._variables = {} + self._return = {} + + @property + def variables(self): + return self._variables + + def get_variable(self, name): + return self._variables.get(name) + + def set_variable(self, name, value): + self._variables[name] = value + + def get_return(self, plugin_name): + ret = self._return.get(plugin_name) + if isinstance(ret, PluginReturn): + return ret + return None + + def set_return(self, plugin_name, plugin_return): + self._return[plugin_name] = plugin_return + + class PluginReturn(object): def __init__(self, value=False, *arg, **kwargs): @@ -83,7 +110,7 @@ def __bool__(self): @property def value(self): return self._return_value - + @property def args(self): return self._return_args @@ -91,11 +118,9 @@ def args(self): @property def kwargs(self): return self._return_kwargs - - def get_return(self, key): - if key in self.kwargs: - return self.kwargs[key] - return None + + def get_return(self, key, default=None): + return self.kwargs.get(key, default) def set_args(self, *args): self._return_args = args @@ -105,12 +130,12 @@ def set_kwargs(self, **kwargs): def set_return(self, value): self._return_value = value - + def return_true(self, *args, **kwargs): self.set_return(True) self.set_args(*args) self.set_kwargs(**kwargs) - + def return_false(self, *args, **kwargs): self.set_return(False) self.set_args(*args) @@ -119,25 +144,48 @@ def return_false(self, *args, **kwargs): class PluginContext(object): - def __init__(self, components, clients, cluster_config, cmd, options, dev_mode, stdio): + def __init__(self, plugin_name, namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmd, options, dev_mode, stdio): + self.namespace = namespace + self.namespaces = namespaces + self.deploy_name = deploy_name + self.repositories =repositories + self.plugin_name = plugin_name self.components = components self.clients = clients self.cluster_config = cluster_config - self.cmd = cmd + self.cmds = cmd self.options = options self.dev_mode = dev_mode self.stdio = stdio self.concurrent_executor = ConcurrentExecutor(32) self._return = PluginReturn() - def get_return(self): - return self._return + def get_return(self, plugin_name=None, spacename=None): + if spacename: + namespace = self.namespaces.get(spacename) + else: + namespace = self.namespace + if plugin_name is None: + plugin_name = self.plugin_name + return namespace.get_return(plugin_name) if namespace else None def return_true(self, *args, **kwargs): self._return.return_true(*args, **kwargs) - + self.namespace.set_return(self.plugin_name, self._return) + def return_false(self, *args, **kwargs): self._return.return_false(*args, **kwargs) + self.namespace.set_return(self.plugin_name, self._return) + + def get_variable(self, name, spacename=None): + if spacename: + namespace = self.namespaces.get(spacename) + else: + namespace = self.namespace + return namespace.get_variable(name) if namespace else None + + def set_variable(self, name, value): + self.namespace.set_variable(name, value) class SubIO(object): @@ -148,7 +196,7 @@ def __init__(self, stdio): def __del__(self): self.before_close() - + def _temp_function(self, *arg, **kwargs): pass @@ -192,13 +240,21 @@ def _export(self): def __del__(self): self._export() - def before_do(self, components, clients, cluster_config, cmd, options, stdio, *arg, **kwargs): + def before_do( + self, plugin_name, namespace, namespaces, deploy_name, + repositories, components, clients, cluster_config, cmd, + options, stdio, *arg, **kwargs + ): self._import(stdio) sub_stdio = SubIO(stdio) sub_clients = {} for server in clients: sub_clients[server] = ScriptPlugin.ClientForScriptPlugin(clients[server], sub_stdio) - self.context = PluginContext(components, sub_clients, cluster_config, cmd, options, self.dev_mode, sub_stdio) + self.context = PluginContext( + plugin_name, namespace, namespaces, deploy_name, repositories, components, + sub_clients, cluster_config, cmd, options, self.dev_mode, sub_stdio + ) + namespace.set_return(plugin_name, None) def after_do(self, stdio, *arg, **kwargs): self._export(stdio) @@ -206,17 +262,28 @@ def after_do(self, stdio, *arg, **kwargs): def pyScriptPluginExec(func): - def _new_func(self, components, clients, cluster_config, cmd, options, stdio, *arg, **kwargs): - self.before_do(components, clients, cluster_config, cmd, options, stdio, *arg, **kwargs) + def _new_func( + self, namespace, namespaces, deploy_name, + repositories, components, clients, cluster_config, cmd, + options, stdio, *arg, **kwargs + ): + self.before_do(self.name, namespace, namespaces, deploy_name, + repositories, components, clients, cluster_config, cmd, + options, stdio, *arg, **kwargs) if self.module: method_name = func.__name__ method = getattr(self.module, method_name, False) + namespace_vars = copy(self.context.namespace.variables) + namespace_vars.update(kwargs) + kwargs = namespace_vars if method: try: - method(self.context, *arg, **kwargs) + ret = method(self.context, *arg, **kwargs) + if ret is None and self.context and self.context.get_return() is None: + self.context.return_false() except Exception as e: + self.context.return_false(exception=e) stdio and getattr(stdio, 'exception', print)('%s RuntimeError: %s' % (self, e)) - pass ret = self.context.get_return() if self.context else PluginReturn() self.after_do(stdio, *arg, **kwargs) return ret @@ -226,45 +293,57 @@ def _new_func(self, components, clients, cluster_config, cmd, options, stdio, *a class PyScriptPlugin(ScriptPlugin): LIBS_PATH = [] - PLUGIN_COMPONENT_NAME = None + PLUGIN_NAME = None def __init__(self, component_name, plugin_path, version, dev_mode): - if not self.PLUGIN_COMPONENT_NAME: + if not self.PLUGIN_NAME: raise NotImplementedError super(PyScriptPlugin, self).__init__(component_name, plugin_path, version, dev_mode) self.module = None + self.name = self.PLUGIN_NAME self.libs_path = deepcopy(self.LIBS_PATH) self.libs_path.append(self.plugin_path) - def __call__(self, clients, cluster_config, cmd, options, stdio, *arg, **kwargs): - method = getattr(self, self.PLUGIN_COMPONENT_NAME, False) + def __call__( + self, namespace, namespaces, deploy_name, + repositories, components, clients, cluster_config, cmd, + options, stdio, *arg, **kwargs + ): + method = getattr(self, self.PLUGIN_NAME, False) if method: - return method(clients, cluster_config, cmd, options, stdio, *arg, **kwargs) + return method( + namespace, namespaces, deploy_name, + repositories, components, clients, cluster_config, cmd, + options, stdio, *arg, **kwargs + ) else: raise NotImplementedError def _import(self, stdio=None): if self.module is None: DynamicLoading.add_libs_path(self.libs_path) - self.module = DynamicLoading.import_module(self.PLUGIN_COMPONENT_NAME, stdio) + self.module = DynamicLoading.import_module(self.PLUGIN_NAME, stdio) def _export(self, stdio=None): if self.module: DynamicLoading.remove_libs_path(self.libs_path) - DynamicLoading.export_module(self.PLUGIN_COMPONENT_NAME, stdio) + DynamicLoading.export_module(self.PLUGIN_NAME, stdio) # this is PyScriptPlugin demo # class InitPlugin(PyScriptPlugin): # FLAG_FILE = 'init.py' -# PLUGIN_COMPONENT_NAME = 'init' +# PLUGIN_NAME = 'init' # PLUGIN_TYPE = PluginType.INIT # def __init__(self, component_name, plugin_path, version): # super(InitPlugin, self).__init__(component_name, plugin_path, version) # @pyScriptPluginExec -# def init(self, components, ssh_clients, cluster_config, cmd, options, stdio, *arg, **kwargs): +# def init( +# self, namespace, namespaces, deploy_name, +# repositories, components, clients, cluster_config, cmd, +# options, stdio, *arg, **kwargs): # pass class Null(object): @@ -353,7 +432,7 @@ def _format(self): raise Exception('Invalid Value') else: self._value = 0 - + class Time(ConfigItemType): UNITS = { @@ -384,7 +463,7 @@ def _format(self): self._value = 0 class Capacity(ConfigItemType): - + UNITS = {"B": 1, "K": 1<<10, "M": 1<<20, "G": 1<<30, "T": 1<<40, 'P': 1 << 50} def _format(self): @@ -396,7 +475,7 @@ def _format(self): else: r = re.match('^(\d+)(\w)B?$', self._origin.upper()) n, u = r.groups() - unit = self.UNITS.get(u.upper()) + unit = self.UNITS.get(u.upper()) if unit: self._value = int(n) * unit else: @@ -496,18 +575,27 @@ class ConfigItem(object): def __init__( self, name, - param_type=str, - default=None, - min_value=None, - max_value=None, - require=False, - need_restart=False, + param_type=str, + default=None, + min_value=None, + max_value=None, + require=False, + essential=False, + section="", + need_reload=False, + need_restart=False, need_redeploy=False, - modify_limit=None + modify_limit=None, + name_local=None, + description_en=None, + description_local=None ): self.name = name self.default = default self.require = require + self.essential = essential + self.section = section + self.need_reload = need_reload self.need_restart = need_restart self.need_redeploy = need_redeploy self._param_type = param_type @@ -515,6 +603,9 @@ def __init__( self.max_value = param_type(max_value) if max_value is not None else None self.modify_limit = getattr(self, ('_%s_limit' % modify_limit).lower(), self._none_limit) self.had_modify_limit = self.modify_limit != self._none_limit + self.name_local = name_local if name_local is not None else self.name + self.description_en = description_en + self.description_local = description_local if description_local is not None else self.description_en def param_type(self, value): try: @@ -535,12 +626,12 @@ def _modify_limit(self, old_value, new_value): if old_value == new_value: return True raise Exception('DO NOT modify %s after startup' % self.name) - + def _increase_limit(self, old_value, new_value): if self.param_type(new_value) > self.param_type(old_value): raise Exception('DO NOT increase %s after startup' % self.name) return True - + def _decrease_limit(self, old_value, new_value): if self.param_type(new_value) < self.param_type(old_value): raise Exception('DO NOT decrease %s after startup' % self.name) @@ -548,7 +639,7 @@ def _decrease_limit(self, old_value, new_value): def _none_limit(self, old_value, new_value): return True - + PLUGIN_TYPE = PluginType.PARAM DEF_PARAM_YAML = 'parameter.yaml' FLAG_FILE = DEF_PARAM_YAML @@ -598,8 +689,13 @@ def params(self): max_value=ConfigUtil.get_value_from_dict(conf, 'max_value', None), modify_limit=ConfigUtil.get_value_from_dict(conf, 'modify_limit', None), require=ConfigUtil.get_value_from_dict(conf, 'require', False), + section=ConfigUtil.get_value_from_dict(conf, 'section', ""), + essential=ConfigUtil.get_value_from_dict(conf, 'essential', False), + need_reload=ConfigUtil.get_value_from_dict(conf, 'need_reload', False), need_restart=ConfigUtil.get_value_from_dict(conf, 'need_restart', False), - need_redeploy=ConfigUtil.get_value_from_dict(conf, 'need_redeploy', False) + need_redeploy=ConfigUtil.get_value_from_dict(conf, 'need_redeploy', False), + description_en=ConfigUtil.get_value_from_dict(conf, 'description_en', None), + description_local=ConfigUtil.get_value_from_dict(conf, 'description_local', None), ) except: pass @@ -647,7 +743,7 @@ def params_default(self): params = self.params for name in params: conf = params[name] - temp[conf.name] = conf.default + self._params_default[conf.name] = conf.default return self._params_default @@ -722,7 +818,7 @@ def __init__(self, component_name, plugin_path, version, dev_mode): def var_replace(cls, string, var): if not var: return string - done = [] + done = [] while string: m = cls._KEYCRE.search(string) @@ -830,7 +926,7 @@ def get_best_plugin(self, version): if plugins: plugin = max(plugins, key=lambda x: x.version) # self.stdio and getattr(self.stdio, 'warn', print)( - # '%s %s plugin version %s not found, use the best suitable version %s.\n Use `obd update` to update local plugin repository' % + # '%s %s plugin version %s not found, use the best suitable version %s.\n Use `obd update` to update local plugin repository' % # (self.component_name, self.PLUGIN_TYPE.name.lower(), version, plugin.version) # ) return plugin @@ -862,7 +958,7 @@ def _create_(self, script_name): class %s(PyScriptPlugin): FLAG_FILE = '%s.py' - PLUGIN_COMPONENT_NAME = '%s' + PLUGIN_NAME = '%s' def __init__(self, component_name, plugin_path, version, dev_mode): super(%s, self).__init__(component_name, plugin_path, version, dev_mode) @@ -872,7 +968,10 @@ def set_plugin_type(plugin_type): %s.PLUGIN_TYPE = plugin_type @pyScriptPluginExec - def %s(self, components, ssh_clients, cluster_config, cmd, options, stdio, *arg, **kwargs): + def %s( + self, namespace, namespaces, deploy_name, + repositories, components, clients, cluster_config, cmd, + options, stdio, *arg, **kwargs): pass ''' % (self.PLUGIN_TYPE.value, script_name, script_name, self.PLUGIN_TYPE.value, self.PLUGIN_TYPE.value, script_name)) clz = locals()[self.PLUGIN_TYPE.value] diff --git a/_repository.py b/_repository.py index 7830a4b..64c5757 100644 --- a/_repository.py +++ b/_repository.py @@ -30,7 +30,8 @@ from _rpm import Package, PackageInfo, Version from _arch import getBaseArch -from tool import DirectoryUtil, FileUtil, YamlLoader +from _environ import ENV_DISABLE_PARALLER_EXTRACT +from tool import DirectoryUtil, FileUtil, YamlLoader, COMMAND_ENV from _manager import Manager from _plugin import InstallPlugin @@ -150,24 +151,24 @@ def __init__(self, src_path, target_path, mode): self.mode = mode -class ParallerExtractWorker(object): +class Extractor(object): def __init__(self, pkg, files, stdio=None): self.pkg = pkg self.files = files self.stdio = stdio - @staticmethod - def extract(worker): - with worker.pkg.open() as rpm: - for info in worker.files: + def extract(self): + with self.pkg.open() as rpm: + for info in self.files: if os.path.exists(info.target_path): continue fd = rpm.extractfile(info.src_path) - with FileUtil.open(info.target_path, 'wb', stdio=worker.stdio) as f: + with FileUtil.open(info.target_path, 'wb', stdio=self.stdio) as f: FileUtil.copy_fileobj(fd, f) if info.mode != 0o744: os.chmod(info.target_path, info.mode) + return True class ParallerExtractor(object): @@ -180,10 +181,30 @@ def __init__(self, pkg, files, stdio=None): self.pkg = pkg self.files = files self.stdio = stdio + + @staticmethod + def _extract(worker): + return worker.extract() def extract(self): if not self.files: return + + if sys.version_info.major == 2 or COMMAND_ENV.get(ENV_DISABLE_PARALLER_EXTRACT, False): + return self._single() + else: + return self._paraller() + + def _single(self): + self.stdio and getattr(self.stdio, 'verbose', print)('extract mode: single') + return Extractor( + self.pkg, + self.files, + stdio=self.stdio + ).extract() + + def _paraller(self): + self.stdio and getattr(self.stdio, 'verbose', print)('extract mode: paraller') workers = [] file_num = len(self.files) paraller = int(min(self.MAX_PARALLER, file_num)) @@ -192,7 +213,7 @@ def extract(self): index = 0 while index < file_num: p_index = index + size - workers.append(ParallerExtractWorker( + workers.append(Extractor( self.pkg, self.files[index:p_index], stdio=self.stdio @@ -201,16 +222,20 @@ def extract(self): pool = Pool(processes=paraller) try: - results = pool.map(ParallerExtractWorker.extract, workers) + results = pool.map(ParallerExtractor._extract, workers) for r in results: if not r: return False + return True except KeyboardInterrupt: if pool: pool.close() pool = None + except: + self.stdio and getattr(self.stdio, 'exception', print)() finally: pool and pool.close() + return False class Repository(PackageInfo): @@ -309,14 +334,14 @@ def _dump(self): except: self.stdio and getattr(self.stdio, 'exception', print)('dump %s to %s failed' % (data, self.data_file_path)) return False + + def need_load(self, pkg, plugin): + return self.hash != pkg.md5 or not self.install_time > plugin.check_value or not self.file_check(plugin) def load_pkg(self, pkg, plugin): if self.is_shadow_repository(): self.stdio and getattr(self.stdio, 'print', '%s is a shadow repository' % self) return False - hash_path = os.path.join(self.repository_dir, '.hash') - if self.hash == pkg.md5 and self.file_check(plugin) and self.install_time > plugin.check_value: - return True self.clear() try: with pkg.open() as rpm: @@ -385,7 +410,7 @@ def load_pkg(self, pkg, plugin): if not os.path.exists(path) and n_dir[:-1] in dirnames: DirectoryUtil.mkdir(path) if not os.path.isdir(path): - raise Exception('%s in %s is not dir.' % (pkg.path, n_dir)) + raise Exception('%s in %s is not dir.' % (n_dir, pkg.path)) self.set_version(pkg.version) self.set_release(pkg.release) self.md5 = pkg.md5 diff --git a/_stdio.py b/_stdio.py index 67ac4c3..13d4080 100644 --- a/_stdio.py +++ b/_stdio.py @@ -48,24 +48,61 @@ class BufferIO(object): + + def __init__(self, auto_clear=True): + self._buffer = [] + self.auto_clear = auto_clear + self.closed = False + + def isatty(self): + return False + + def writable(self): + return not self.closed - def __init__(self): + def close(self): + self.closed = True + return self + + def open(self): + self.closed = False self._buffer = [] + return self + + def __enter__(self): + return self.open() + + def __exit__(self, *args, **kwargs): + return self.close() def write(self, s): self._buffer.append(s) - def read(self): + def read(self, *args, **kwargs): s = ''.join(self._buffer) - self._buffer = [] + self.auto_clear and self.clear() return s + def clear(self): + self._buffer = [] + + def flush(self): + self.auto_clear and self.clear() + return True + class SysStdin(object): NONBLOCK = False STATS = None FD = None + IS_TTY = None + + @classmethod + def isatty(cls): + if cls.IS_TTY is None: + cls.IS_TTY = sys.stdin.isatty() + return cls.IS_TTY @classmethod def fileno(cls): @@ -140,29 +177,33 @@ def _readlines(cls): return sys.stdin.readlines() +class FormtatText(object): + def __init__(self, text, color): + self.text = text + self.color_text = color + text + Fore.RESET -class FormtatText(object): + def format(self, istty=True): + return self.color_text if istty else self.text - @staticmethod - def format(text, color): - return color + text + Fore.RESET + def __str__(self): + return self.format() @staticmethod def info(text): - return FormtatText.format(text, Fore.BLUE) + return FormtatText(text, Fore.BLUE) @staticmethod def success(text): - return FormtatText.format(text, Fore.GREEN) + return FormtatText(text, Fore.GREEN) @staticmethod def warning(text): - return FormtatText.format(text, Fore.YELLOW) + return FormtatText(text, Fore.YELLOW) @staticmethod def error(text): - return FormtatText.format(text, Fore.RED) + return FormtatText(text, Fore.RED) class LogSymbols(Enum): @@ -220,7 +261,7 @@ def stop_and_persist(self, symbol=' ', text=None): if getattr(self._stream, 'isatty', lambda : False)(): return super(IOHalo, self).stop_and_persist(symbol=symbol, text=text) else: - self._stream.write(' %s\n' % symbol) + self._stream.write(' %s\n' % symbol.format(istty=False)) def succeed(self, text=None): return self.stop_and_persist(symbol=LogSymbols.SUCCESS.value, text=text) @@ -238,9 +279,11 @@ def info(self, text=None): class IOProgressBar(ProgressBar): @staticmethod - def _get_widgets(widget_type, text): - if widget_type == 'download': - return ['%s: ' % text, Percentage(), ' ', Bar(marker='#', left='[', right=']'), ' ', ETA(), ' ', FileTransferSpeed()] + def _get_widgets(widget_type, text, istty=True): + if istty is False: + return [text] + elif widget_type == 'download': + return ['%s: ' % text, Percentage(), ' ', Bar(marker='#', left='[', right=']'), ' ', ETA(), ' ', FileTransferSpeed()] elif widget_type == 'timer': return ['%s: ' % text, Percentage(), ' ', Bar(marker='#', left='[', right=']'), ' ', AdaptiveETA()] elif widget_type == 'simple_progress': @@ -249,7 +292,8 @@ def _get_widgets(widget_type, text): return ['%s: ' % text, Percentage(), ' ', Bar(marker='#', left='[', right=']')] def __init__(self, maxval=None, text='', term_width=None, poll=1, left_justify=True, stream=None, widget_type='download'): - super(IOProgressBar, self).__init__(maxval=maxval, widgets=self._get_widgets(widget_type, text), term_width=term_width, poll=poll, left_justify=left_justify, fd=stream) + self.stream_isatty = getattr(stream, 'isatty', lambda : False)() + super(IOProgressBar, self).__init__(maxval=maxval, widgets=self._get_widgets(widget_type, text, self.stream_isatty), term_width=term_width, poll=poll, left_justify=left_justify, fd=stream) def start(self): self._hide_cursor() @@ -261,21 +305,23 @@ def update(self, value=None): def finish(self): if self.finished: return - self._show_cursor() - return super(IOProgressBar, self).finish() + self.update(self.maxval) + self._finish() def interrupt(self): if self.finished: return - self._show_cursor() + self._finish() + + def _finish(self): self.finished = True self.fd.write('\n') + self._show_cursor() if self.signal_set: signal.signal(signal.SIGWINCH, signal.SIG_DFL) def _need_update(self): - return (self.currval == self.maxval or self.currval == 0 or getattr(self.fd, 'isatty', lambda : False)()) \ - and super(IOProgressBar, self)._need_update() + return (self.currval == self.maxval or self.currval == 0 or self.stream_isatty) and super(IOProgressBar, self)._need_update() def _check_stream(self): if self.fd.closed: @@ -291,13 +337,13 @@ def _check_stream(self): def _hide_cursor(self): """Disable the user's blinking cursor """ - if self._check_stream() and self.fd.isatty(): + if self._check_stream() and self.stream_isatty: cursor.hide(stream=self.fd) def _show_cursor(self): """Re-enable the user's blinking cursor """ - if self._check_stream() and self.fd.isatty(): + if self._check_stream() and self.stream_isatty: cursor.show(stream=self.fd) @@ -320,8 +366,6 @@ class IO(object): VERBOSE_LEVEL = 0 WARNING_PREV = FormtatText.warning('[WARN]') ERROR_PREV = FormtatText.error('[ERROR]') - IS_TTY = sys.stdin.isatty() - INPUT = SysStdin def __init__(self, level, @@ -329,7 +373,8 @@ def __init__(self, use_cache=False, track_limit=0, root_io=None, - stream=sys.stdout + input_stream=SysStdin, + output_stream=sys.stdout ): self.level = level self.msg_lv = msg_lv @@ -344,9 +389,34 @@ def __init__(self, self._verbose_prefix = '-' * self.level self.sub_ios = {} self.sync_obj = None - self._out_obj = None if self._root_io else stream - self._cur_out_obj = self._out_obj + self.input_stream = None + self._out_obj = None + self._cur_out_obj = None self._before_critical = None + self._output_is_tty = False + self._input_is_tty = False + self.set_input_stream(input_stream) + self.set_output_stream(output_stream) + + def isatty(self): + if self._root_io: + return self._root_io.isatty() + return self._output_is_tty and self._input_is_tty + + def set_input_stream(self, input_stream): + if self._root_io: + return False + self.input_stream = input_stream + self._input_is_tty = input_stream.isatty() + + def set_output_stream(self, output_stream): + if self._root_io: + return False + if self._cur_out_obj == self._out_obj: + self._cur_out_obj = output_stream + self._out_obj = output_stream + self._output_is_tty = output_stream.isatty() + return True def init_trace_logger(self, log_path, log_name=None, trace_id=None): if self._trace_logger is None: @@ -360,7 +430,7 @@ def __getstate__(self): state = {} for key in self.__dict__: state[key] = self.__dict__[key] - for key in ['_trace_logger', 'sync_obj', '_out_obj', '_cur_out_obj', '_before_critical']: + for key in ['_trace_logger', 'input_stream', 'sync_obj', '_out_obj', '_cur_out_obj', '_before_critical']: state[key] = None return state @@ -418,6 +488,11 @@ def _cache_off(self): self._flush_log() self._log_cache = None return True + + def get_input_stream(self): + if self._root_io: + return self._root_io.get_input_stream() + return self.input_stream def get_cur_out_obj(self): if self._root_io: @@ -571,15 +646,15 @@ def print_list(self, ary, field_names=None, exp=lambda x: x if isinstance(x, (li def read(self, msg='', blocked=False): if msg: self._print(MsgLevel.INFO, msg) - return self.INPUT.read(blocked) + return self.get_input_stream().read(blocked) def confirm(self, msg): msg = '%s [y/n]: ' % msg self.print(msg, end='') - if self.IS_TTY: + if self._input_is_tty: while True: try: - ans = raw_input() + ans = self.get_input_stream().readline(blocked=True).strip().lower() if ans == 'y': return True if ans == 'n': @@ -598,8 +673,13 @@ def _format(self, msg, *args): def _print(self, msg_lv, msg, *args, **kwargs): if msg_lv < self.msg_lv: return + if 'prev_msg' in kwargs: + print_msg = '%s %s' % (kwargs['prev_msg'], msg) + del kwargs['prev_msg'] + else: + print_msg = msg kwargs['file'] = self.get_cur_out_obj() - kwargs['file'] and print(self._format(msg, *args), **kwargs) + kwargs['file'] and print(self._format(print_msg, *args), **kwargs) del kwargs['file'] self.log(msg_lv, msg, *args, **kwargs) @@ -621,6 +701,7 @@ def _flush_log(self): for levelno, line, args, kwargs in self._log_cache: self.trace_logger.log(levelno, line, *args, **kwargs) self._log_cache = [] + def _log(self, levelno, msg, *args, **kwargs): if self.trace_logger: self.trace_logger.log(levelno, msg, *args, **kwargs) @@ -629,16 +710,15 @@ def print(self, msg, *args, **kwargs): self._print(MsgLevel.INFO, msg, *args, **kwargs) def warn(self, msg, *args, **kwargs): - self._print(MsgLevel.WARN, '%s %s' % (self.WARNING_PREV, msg), *args, **kwargs) + self._print(MsgLevel.WARN, msg, prev_msg=self.WARNING_PREV.format(self.isatty()), *args, **kwargs) def error(self, msg, *args, **kwargs): - self._print(MsgLevel.ERROR, '%s %s' % (self.ERROR_PREV, msg), *args, **kwargs) + self._print(MsgLevel.ERROR, msg, prev_msg=self.ERROR_PREV.format(self.isatty()), *args, **kwargs) def critical(self, msg, *args, **kwargs): - if self._root_io: - return self._root_io.critical(msg, *args, **kwargs) self._print(MsgLevel.CRITICAL, '%s %s' % (self.ERROR_PREV, msg), *args, **kwargs) - self.exit(kwargs['code'] if 'code' in kwargs else 255) + if not self._root_io: + self.exit(kwargs['code'] if 'code' in kwargs else 255) def verbose(self, msg, *args, **kwargs): if self.level > self.VERBOSE_LEVEL: @@ -728,13 +808,17 @@ def __getattr__(self, item): if item.startswith('__'): return super(StdIO, self).__getattribute__(item) if self.io is None: - return FAKE_RETURN + if item == 'sub_io': + return self + else: + return FAKE_RETURN if item not in self._attrs: attr = getattr(self.io, item, EMPTY) if attr is not EMPTY: self._attrs[item] = attr else: - self._warn_func(FormtatText.warning("WARNING: {} has no attribute '{}'".format(self.io, item))) + is_tty = getattr(self._stream, 'isatty', lambda : False)() + self._warn_func(FormtatText.warning("WARNING: {} has no attribute '{}'".format(self.io, item)).format(is_tty)) self._attrs[item] = FAKE_RETURN return self._attrs[item] diff --git a/config_parser/oceanbase/cluster_config_parser.py b/config_parser/oceanbase/cluster_config_parser.py index d83a203..25183e2 100644 --- a/config_parser/oceanbase/cluster_config_parser.py +++ b/config_parser/oceanbase/cluster_config_parser.py @@ -56,6 +56,12 @@ def get_server_src_conf(cls, cluster_config, component_config, server): zone_config[server.name] = {} return zone_config[server.name] + @classmethod + def get_global_src_conf(cls, cluster_config, component_config): + if 'config' not in component_config: + component_config['config'] = {} + return component_config['config'] + @classmethod def _to_cluster_config(cls, component_name, conf): servers = OrderedDict() @@ -242,8 +248,8 @@ def _from_cluster_config(cls, conf, cluster_config): conf['name'] = global_config['appname'] del global_config['appname'] + conf['zones'] = zones if global_config: conf['config'] = global_config - conf['zones'] = zones return conf \ No newline at end of file diff --git a/core.py b/core.py index 4bcc5d3..1316bb2 100644 --- a/core.py +++ b/core.py @@ -24,21 +24,21 @@ import os import time from optparse import Values -from copy import deepcopy +from copy import deepcopy, copy import tempfile from subprocess import call as subprocess_call from ssh import SshClient, SshConfig -from tool import ConfigUtil, FileUtil, DirectoryUtil, YamlLoader, timeout, COMMAND_ENV, OrderedDict +from tool import FileUtil, DirectoryUtil, YamlLoader, timeout, COMMAND_ENV, OrderedDict from _stdio import MsgLevel from _rpm import Version from _mirror import MirrorRepositoryManager, PackageInfo -from _plugin import PluginManager, PluginType, InstallPlugin +from _plugin import PluginManager, PluginType, InstallPlugin, PluginContextNamespace from _deploy import DeployManager, DeployStatus, DeployConfig, DeployConfigStatus, Deploy from _repository import RepositoryManager, LocalPackage, Repository -from _errno import EC_SOME_SERVER_STOPED -from _lock import LockManager +import _errno as err +from _lock import LockManager, LockMode from _optimize import OptimizeManager from _environ import ENV_REPO_INSTALL_MODE, ENV_BASE_DIR @@ -47,7 +47,7 @@ class ObdHome(object): HOME_LOCK_RELATIVE_PATH = 'obd.conf' - def __init__(self, home_path, dev_mode=False, stdio=None): + def __init__(self, home_path, dev_mode=False, lock_mode=None, stdio=None): self.home_path = home_path self.dev_mode = dev_mode self._lock = None @@ -61,7 +61,15 @@ def __init__(self, home_path, dev_mode=False, stdio=None): self.stdio = None self._stdio_func = None self.ssh_clients = {} + self.deploy = None + self.cmds = [] + self.options = Values() + self.repositories = None + self.namespaces = {} self.set_stdio(stdio) + if lock_mode is None: + lock_mode = LockMode.DEPLOY_SHARED_LOCK if dev_mode else LockMode.DEFAULT + self.lock_manager.set_lock_mode(lock_mode) self.lock_manager.global_sh_lock() @property @@ -103,6 +111,32 @@ def optimize_manager(self): def _obd_update_lock(self): self.lock_manager.global_ex_lock() + def fork(self, deploy=None, repositories=None, cmds=None, options=None, stdio=None): + new_obd = copy(self) + if deploy: + new_obd.set_deploy(deploy) + if repositories: + new_obd.set_repositories(repositories) + if cmds: + new_obd.set_cmds(cmds) + if options: + new_obd.set_options(options) + if stdio: + new_obd.set_stdio(stdio) + return new_obd + + def set_deploy(self, deploy): + self.deploy = deploy + + def set_repositories(self, repositories): + self.repositories = repositories + + def set_cmds(self, cmds): + self.cmds = cmds + + def set_options(self, options): + self.options = options + def set_stdio(self, stdio): def _print(msg, *arg, **kwarg): sep = kwarg['sep'] if 'sep' in kwarg else None @@ -115,51 +149,128 @@ def _print(msg, *arg, **kwarg): for func in ['start_loading', 'stop_loading', 'print', 'confirm', 'verbose', 'warn', 'exception', 'error', 'critical', 'print_list', 'read']: self._stdio_func[func] = getattr(self.stdio, func, _print) + def get_namespace(self, spacename): + if spacename in self.namespaces: + namespace = self.namespaces[spacename] + else: + namespace = PluginContextNamespace(spacename=spacename) + self.namespaces[spacename] = namespace + return namespace + + def call_plugin(self, plugin, repository, spacename=None, **kwargs): + args = { + 'namespace': self.get_namespace(repository.name if spacename == None else spacename), + 'namespaces': self.namespaces, + 'deploy_name': None, + 'cluster_config': None, + 'repositories': self.repositories, + 'repository': repository, + 'components': None, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.stdio + } + if self.deploy: + args['deploy_name'] = self.deploy.name + args['components'] = self.deploy.deploy_info.components + args['cluster_config'] = self.deploy.deploy_config.components[repository.name] + if "clients" not in kwargs: + args['clients'] = self.get_clients(self.deploy.deploy_config, self.repositories) + args.update(kwargs) + + self._call_stdio('verbose', 'Call %s for %s' % (plugin, repository)) + return plugin(**args) + def _call_stdio(self, func, msg, *arg, **kwarg): if func not in self._stdio_func: return None return self._stdio_func[func](msg, *arg, **kwarg) - def add_mirror(self, src, opts): + def add_mirror(self, src): if re.match('^https?://', src): return self.mirror_manager.add_remote_mirror(src) else: - return self.mirror_manager.add_local_mirror(src, getattr(opts, 'force', False)) + return self.mirror_manager.add_local_mirror(src, getattr(self.options, 'force', False)) - def deploy_param_check(self, repositories, deploy_config): + def deploy_param_check(self, repositories, deploy_config, gen_config_plugins={}): # parameter check errors = [] for repository in repositories: cluster_config = deploy_config.components[repository.name] errors += cluster_config.check_param()[1] + skip_keys = [] + if repository in gen_config_plugins: + ret = self.call_plugin(gen_config_plugins[repository], repository, return_generate_keys=True, clients={}) + if ret: + skip_keys = ret.get_return('generate_keys', []) for server in cluster_config.servers: self._call_stdio('verbose', '%s %s param check' % (server, repository)) - need_items = cluster_config.get_unconfigured_require_item(server) + need_items = cluster_config.get_unconfigured_require_item(server, skip_keys=skip_keys) if need_items: - errors.append('%s %s need config: %s' % (server, repository.name, ','.join(need_items))) + errors.append(str(err.EC_NEED_CONFIG.format(server=server, component=repository.name, miss_keys=','.join(need_items)))) return errors + def deploy_param_check_return_check_status(self, repositories, deploy_config, gen_config_plugins={}): + # parameter check + param_check_status = {} + check_pass = True + for repository in repositories: + cluster_config = deploy_config.components[repository.name] + check_status = param_check_status[repository.name] = {} + skip_keys = [] + if repository in gen_config_plugins: + ret = self.call_plugin(gen_config_plugins[repository], repository, return_generate_keys=True, clients={}) + if ret: + skip_keys = ret.get_return('generate_keys', []) + check_res = cluster_config.servers_check_param() + for server in check_res: + status = err.CheckStatus() + errors = check_res[server].get('errors', []) + self._call_stdio('verbose', '%s %s param check' % (server, repository)) + need_items = cluster_config.get_unconfigured_require_item(server, skip_keys=skip_keys) + if need_items: + errors.append(err.EC_NEED_CONFIG.format(server=server, component=repository.name, miss_keys=','.join(need_items))) + if errors: + status.status = err.CheckStatus.FAIL + check_pass = False + status.error = err.EC_PARAM_CHECK.format(errors=errors) + status.suggests.append(err.SUG_PARAM_CHECK.format()) + else: + status.status = err.CheckStatus.PASS + check_status[server] = status + return param_check_status, check_pass + def get_clients(self, deploy_config, repositories): + ssh_clients, _ = self.get_clients_with_connect_status(deploy_config, repositories, True) + return ssh_clients + + def get_clients_with_connect_status(self, deploy_config, repositories, fail_exit=False): servers = set() user_config = deploy_config.user if user_config not in self.ssh_clients: self.ssh_clients[user_config] = {} ssh_clients = self.ssh_clients[user_config] - + connect_status = {} + for repository in repositories: cluster_config = deploy_config.components[repository.name] for server in cluster_config.servers: if server not in ssh_clients: servers.add(server) + else: + connect_status[server] = err.CheckStatus(err.CheckStatus.PASS) if servers: - self.ssh_clients_connect(servers, ssh_clients, user_config) - return ssh_clients + connect_status.update(self.ssh_clients_connect(servers, ssh_clients, user_config, fail_exit)) + return ssh_clients, connect_status - def ssh_clients_connect(self, servers, ssh_clients, user_config): + def ssh_clients_connect(self, servers, ssh_clients, user_config, fail_exit=False): self._call_stdio('start_loading', 'Open ssh connection') + connect_io = self.stdio if fail_exit else self.stdio.sub_io() + connect_status = {} + success = True for server in servers: if server not in ssh_clients: - ssh_clients[server] = SshClient( + client = SshClient( SshConfig( server.ip, user_config.username, @@ -170,9 +281,18 @@ def ssh_clients_connect(self, servers, ssh_clients, user_config): ), self.stdio ) - ssh_clients[server].connect() - self._call_stdio('stop_loading', 'succeed') - return ssh_clients + error = client.connect(stdio=connect_io) + connect_status[server] = status = err.CheckStatus() + if error is not True: + success = False + status.status = err.CheckStatus.FAIL + status.error = error + status.suggests.append(err.SUG_SSH_FAILED.format()) + else: + status.status = err.CheckStatus.PASS + ssh_clients[server] = client + self._call_stdio('stop_loading', 'succeed' if success else 'fail') + return connect_status def search_plugin(self, repository, plugin_type, no_found_exit=True): self._call_stdio('verbose', 'Search %s plugin for %s' % (plugin_type.name.lower(), repository.name)) @@ -375,12 +495,13 @@ def is_server_list_change(deploy_config): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) param_plugins = {} repositories, pkgs = [], [] is_deployed = deploy and deploy.deploy_info.status not in [DeployStatus.STATUS_CONFIGURED, DeployStatus.STATUS_DESTROYED] is_started = deploy and deploy.deploy_info.status in [DeployStatus.STATUS_RUNNING, DeployStatus.STATUS_STOPPED] user_input = self._call_stdio('read', '') - if not user_input and not self.stdio.IS_TTY: + if not user_input and not self.stdio.isatty(): time.sleep(0.1) user_input = self._call_stdio('read', '') if not user_input: @@ -600,9 +721,7 @@ def is_server_list_change(deploy_config): FileUtil.copy(tf.name, target_src_path, self.stdio) ret = True if deploy: - if deploy.deploy_info.status == DeployStatus.STATUS_RUNNING or ( - config_status == DeployConfigStatus.NEED_REDEPLOY and is_deployed - ): + if is_started or (config_status == DeployConfigStatus.NEED_REDEPLOY and is_deployed): msg += deploy.effect_tip() except Exception as e: deploy.update_deploy_config_status(old_config_status) @@ -642,19 +761,22 @@ def get_install_plugin_and_install(self, repositories, pkgs): # Install for local # self._call_stdio('print', 'install package for local ...') for pkg in pkgs: - self._call_stdio('start_loading', 'install %s-%s for local' % (pkg.name, pkg.version)) - # self._call_stdio('verbose', 'install %s-%s for local' % (pkg.name, pkg.version)) + self._call_stdio('verbose', 'create instance repository for %s-%s' % (pkg.name, pkg.version)) repository = self.repository_manager.create_instance_repository(pkg.name, pkg.version, pkg.md5) - if not repository.load_pkg(pkg, install_plugins[repository]): - self._call_stdio('stop_loading', 'fail') - self._call_stdio('error', 'Failed to extract file from %s' % pkg.path) - return None - self._call_stdio('stop_loading', 'succeed') - self._call_stdio('verbose', 'get head repository') - head_repository = self.repository_manager.get_repository(pkg.name, pkg.version, pkg.name) - self._call_stdio('verbose', 'head repository: %s' % head_repository) - if repository > head_repository: - self.repository_manager.create_tag_for_repository(repository, pkg.name, True) + if repository.need_load(pkg, install_plugins[repository]): + self._call_stdio('start_loading', 'install %s-%s for local' % (pkg.name, pkg.version)) + if not repository.load_pkg(pkg, install_plugins[repository]): + self._call_stdio('stop_loading', 'fail') + self._call_stdio('error', 'Failed to extract file from %s' % pkg.path) + return None + self._call_stdio('stop_loading', 'succeed') + self._call_stdio('verbose', 'get head repository') + head_repository = self.repository_manager.get_repository(pkg.name, pkg.version, pkg.name) + self._call_stdio('verbose', 'head repository: %s' % head_repository) + if repository > head_repository: + self.repository_manager.create_tag_for_repository(repository, pkg.name, True) + else: + self._call_stdio('verbose', '%s-%s is already install' % (pkg.name, pkg.version)) repositories.append(repository) return install_plugins @@ -800,14 +922,12 @@ def servers_apply_lib_repository_and_check(self, ssh_clients, deploy_config, rep return ret # If the cluster states are consistent, the status value is returned. Else False is returned. - def cluster_status_check(self, ssh_clients, deploy_config, repositories, ret_status={}): + def cluster_status_check(self, repositories, ret_status={}): self._call_stdio('start_loading', 'Cluster status check') status_plugins = self.search_py_script_plugin(repositories, 'status') component_status = {} for repository in repositories: - cluster_config = deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (status_plugins[repository], repository)) - plugin_ret = status_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) + plugin_ret = self.call_plugin(status_plugins[repository], repository) cluster_status = plugin_ret.get_return('cluster_status') ret_status[repository] = cluster_status for server in cluster_status: @@ -819,8 +939,7 @@ def cluster_status_check(self, ssh_clients, deploy_config, repositories, ret_sta break else: continue - self._call_stdio('stop_loading', 'succeed') - return False + status = None for repository in component_status: if status is None: @@ -866,9 +985,10 @@ def sort_repositories_by_depends(self, deploy_config, repositories): imported_depends.append(repository.name) return sort_repositories - def genconfig(self, name, opt=Values()): + def genconfig(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if deploy: deploy_info = deploy.deploy_info if deploy_info.status not in [DeployStatus.STATUS_CONFIGURED, DeployStatus.STATUS_DESTROYED]: @@ -877,13 +997,14 @@ def genconfig(self, name, opt=Values()): # self._call_stdio('error', 'Deploy name `%s` have been occupied.' % name) # return False - config_path = getattr(opt, 'config', '') + config_path = getattr(self.options, 'config', '') if not config_path: self._call_stdio('error', "Configuration file is need.\nPlease use -c to set configuration file") return False self._call_stdio('verbose', 'Create deploy by configuration path') deploy = self.deploy_manager.create_deploy_config(name, config_path) + self.set_deploy(deploy) if not deploy: return False @@ -897,6 +1018,7 @@ def genconfig(self, name, opt=Values()): repositories, install_plugins = self.search_components_from_mirrors_and_install(deploy_config) if not install_plugins or not repositories: return False + self.set_repositories(repositories) for repository in repositories: real_servers = set() @@ -910,10 +1032,11 @@ def genconfig(self, name, opt=Values()): self._call_stdio('start_loading', 'Cluster param config check') # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) + gen_config_plugins = self.search_py_script_plugin(repositories, 'generate_config') - if not getattr(opt, 'skip_param_check', False): + if not getattr(self.options, 'skip_param_check', False): # Parameter check - errors = self.deploy_param_check(repositories, deploy_config) + errors = self.deploy_param_check(repositories, deploy_config, gen_config_plugins=gen_config_plugins) if errors: self._call_stdio('stop_loading', 'fail') self._call_stdio('error', '\n'.join(errors)) @@ -924,15 +1047,10 @@ def genconfig(self, name, opt=Values()): # Get the client ssh_clients = self.get_clients(deploy_config, repositories) - gen_config_plugins = self.search_py_script_plugin(repositories, 'generate_config') - + generate_consistent_config = getattr(self.options, 'generate_consistent_config', False) component_num = len(repositories) - auto_depend = getattr(opt, 'auto_depend', False) for repository in repositories: - cluster_config = deploy_config.components[repository.name] - - self._call_stdio('verbose', 'Call %s for %s' % (gen_config_plugins[repository], repository)) - ret = gen_config_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], opt, self.stdio, deploy_config, auto_depend=auto_depend) + ret = self.call_plugin(gen_config_plugins[repository], repository, generate_consistent_config=generate_consistent_config) if ret: component_num -= 1 @@ -942,9 +1060,10 @@ def genconfig(self, name, opt=Values()): self.deploy_manager.remove_deploy_config(name) return False - def check_for_ocp(self, name, options=Values()): + def check_for_ocp(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -955,13 +1074,13 @@ def check_for_ocp(self, name, options=Values()): self._call_stdio('error', 'Deploy "%s" not RUNNING' % (name)) return False - version = getattr(options, 'version', '') + version = getattr(self.options, 'version', '') if not version: self._call_stdio('error', 'Use the --version option to specify the required OCP version.') return False deploy_config = deploy.deploy_config - components = getattr(options, 'components', '') + components = getattr(self.options, 'components', '') if components: components = components.split(',') for component in components: @@ -974,6 +1093,7 @@ def check_for_ocp(self, name, options=Values()): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) ocp_check = self.search_py_script_plugin(repositories, 'ocp_check', no_found_act='ignore') connect_plugins = self.search_py_script_plugin([repository for repository in ocp_check], 'connect') @@ -1012,16 +1132,14 @@ def check_for_ocp(self, name, options=Values()): new_cluster_config = new_deploy_config.components[repository.name] if new_deploy_config else None cluster_servers = cluster_config.servers - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugins[repository], repository)) - ret = connect_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, '', options, self.stdio) + ret = self.call_plugin(connect_plugins[repository], repository) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') else: break - self._call_stdio('verbose', 'Call %s for %s' % (ocp_check[repository], repository)) - if ocp_check[repository](deploy_config.components.keys(), ssh_clients, cluster_config, '', options, self.stdio, cursor=cursor, ocp_version=version, new_cluster_config=new_cluster_config, new_clients=new_ssh_clients): + if self.call_plugin(ocp_check[repository], repository, cursor=cursor, ocp_version=version, new_cluster_config=new_cluster_config, new_clients=new_ssh_clients): component_num -= 1 self._call_stdio('print', '%s Check passed.' % repository.name) @@ -1047,9 +1165,10 @@ def sort_repository_by_depend(self, repositories, deploy_config): repositories = temp_repositories return sorted_repositories - def change_deploy_config_style(self, name, options=Values()): + def change_deploy_config_style(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -1064,12 +1183,12 @@ def change_deploy_config_style(self, name, options=Values()): self._call_stdio('error', 'Deploy configuration is empty.\nIt may be caused by a failure to resolve the configuration.\nPlease check your configuration file.\nSee https://github.com/oceanbase/obdeploy/blob/master/docs/zh-CN/4.configuration-file-description.md') return False - style = getattr(options, 'style', '') + style = getattr(self.options, 'style', '') if not style: self._call_stdio('error', 'Use the --style option to specify the preferred style.') return False - components = getattr(options, 'components', '') + components = getattr(self.options, 'components', '') if components: components = components.split(',') for component in components: @@ -1088,6 +1207,7 @@ def change_deploy_config_style(self, name, options=Values()): repositories = [] for component_name in components: repositories.append(self.repository_manager.get_repository_allow_shadow(component_name, '100000.0')) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -1112,7 +1232,7 @@ def change_deploy_config_style(self, name, options=Values()): self._call_stdio('stop_loading', 'fail') return False - def demo(self, opt=Values()): + def demo(self): name = 'demo' self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) @@ -1128,9 +1248,12 @@ def demo(self, opt=Values()): return False components = set() - for component_name in getattr(opt, 'components', '').split(','): + for component_name in getattr(self.options, 'components', '').split(','): if component_name: components.add(component_name) + self.get_namespace(component_name).set_variable('generate_config_mini', True) + self.get_namespace(component_name).set_variable('auto_depend', True) + if not components: self._call_stdio('error', 'Use `-c/--components` to set in the components to be deployed') return @@ -1138,11 +1261,11 @@ def demo(self, opt=Values()): home_path_key = 'home_path' global_config = {home_path_key: os.getenv('HOME')} opt_config = {} - for key in opt.__dict__: + for key in self.options.__dict__: tmp = key.split('.', 1) if len(tmp) == 1: if key == home_path_key: - global_config[key] = opt.__dict__[key] + global_config[key] = self.options.__dict__[key] else: component_name = tmp[0] if component_name not in components: @@ -1153,7 +1276,7 @@ def demo(self, opt=Values()): _config = opt_config[component_name] else: _config = opt_config[component_name][global_key] - _config[tmp[1]] = opt.__dict__[key] + _config[tmp[1]] = self.options.__dict__[key] configs = OrderedDict() for component_name in components: @@ -1170,15 +1293,14 @@ def demo(self, opt=Values()): with tempfile.NamedTemporaryFile(suffix=".yaml", mode='w') as tf: yaml_loader = YamlLoader(self.stdio) yaml_loader.dump(configs, tf) - setattr(opt, 'config', tf.name) - setattr(opt, 'skip_param_check', True) - setattr(opt, 'auto_depend', True) - if not self.genconfig(name, opt): + setattr(self.options, 'config', tf.name) + setattr(self.options, 'skip_param_check', True) + if not self.genconfig(name): return False - setattr(opt, 'config', '') - return self.deploy_cluster(name, opt) and self.start_cluster(name, [], opt) + setattr(self.options, 'config', '') + return self.deploy_cluster(name) and self.start_cluster(name) - def deploy_cluster(self, name, opt=Values()): + def deploy_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) if deploy: @@ -1194,9 +1316,9 @@ def deploy_cluster(self, name, opt=Values()): self._call_stdio('error', 'Failed to apply new deploy configuration') return False - config_path = getattr(opt, 'config', '') - unuse_lib_repo = getattr(opt, 'unuselibrepo', False) - auto_create_tenant = getattr(opt, 'auto_create_tenant', False) + config_path = getattr(self.options, 'config', '') + unuse_lib_repo = getattr(self.options, 'unuselibrepo', False) + auto_create_tenant = getattr(self.options, 'auto_create_tenant', False) self._call_stdio('verbose', 'config path is None or not') if config_path: self._call_stdio('verbose', 'Create deploy by configuration path') @@ -1208,6 +1330,8 @@ def deploy_cluster(self, name, opt=Values()): if not deploy: self._call_stdio('error', 'No such deploy: %s. you can input configuration path to create a new deploy' % name) return False + self.set_deploy(deploy) + self._call_stdio('verbose', 'Get deploy configuration') deploy_config = deploy.deploy_config if not deploy_config: @@ -1243,14 +1367,15 @@ def deploy_cluster(self, name, opt=Values()): repositories, install_plugins = self.search_components_from_mirrors_and_install(deploy_config) if not repositories or not install_plugins: return False + self.set_repositories(repositories) if unuse_lib_repo and not deploy_config.unuse_lib_repository: deploy_config.set_unuse_lib_repository(True) if auto_create_tenant and not deploy_config.auto_create_tenant: deploy_config.set_auto_create_tenant(True) - return self._deploy_cluster(deploy, repositories, opt) + return self._deploy_cluster(deploy, repositories) - def _deploy_cluster(self, deploy, repositories, opt=Values()): + def _deploy_cluster(self, deploy, repositories): deploy_config = deploy.deploy_config install_plugins = self.search_plugins(repositories, PluginType.INSTALL) if not install_plugins: @@ -1292,9 +1417,9 @@ def _deploy_cluster(self, deploy, repositories, opt=Values()): ssh_clients = self.get_clients(deploy_config, repositories) # Check the status for the deployed cluster - if not getattr(opt, 'skip_cluster_status_check', False): + if not getattr(self.options, 'skip_cluster_status_check', False): component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 1: if self.stdio: self._call_stdio('error', 'Some of the servers in the cluster have been started') @@ -1309,21 +1434,20 @@ def _deploy_cluster(self, deploy, repositories, opt=Values()): init_plugins = self.search_py_script_plugin(repositories, 'init') component_num = len(repositories) for repository in repositories: - cluster_config = deploy_config.components[repository.name] init_plugin = init_plugins[repository] self._call_stdio('verbose', 'Exec %s init plugin' % repository) self._call_stdio('verbose', 'Apply %s for %s-%s' % (init_plugin, repository.name, repository.version)) - if init_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opt, self.stdio, self.home_path, repository.repository_dir): + if self.call_plugin(init_plugin, repository): component_num -= 1 if component_num != 0: return False # Install repository to servers - if not self.install_repositories_to_servers(deploy_config, repositories, install_plugins, ssh_clients, opt): + if not self.install_repositories_to_servers(deploy_config, repositories, install_plugins, ssh_clients, self.options): return False # Sync runtime dependencies - if not self.sync_runtime_dependencies(deploy_config, repositories, ssh_clients, opt): + if not self.sync_runtime_dependencies(deploy_config, repositories, ssh_clients, self.options): return False for repository in repositories: @@ -1334,18 +1458,17 @@ def _deploy_cluster(self, deploy, repositories, opt=Values()): return True return False - def install_repository_to_servers(self, components, cluster_config, repository, ssh_clients, options=Values(), unuse_lib_repository=False): + def install_repository_to_servers(self, components, cluster_config, repository, ssh_clients, unuse_lib_repository=False): install_repo_plugin = self.plugin_manager.get_best_py_script_plugin('install_repo', 'general', '0.1') install_plugins = self.search_plugins([repository], PluginType.INSTALL) if not install_plugins: return False install_plugin = install_plugins[repository] check_file_map = install_plugin.file_map(repository) - ret = install_repo_plugin(components, ssh_clients, cluster_config, [], options, self.stdio, - obd_home=self.home_path, install_repository=repository, - install_plugin=install_plugin, check_repository=repository, - check_file_map=check_file_map, - msg_lv='error' if unuse_lib_repository else 'warn') + ret = self.call_plugin(install_repo_plugin, repository, obd_home=self.home_path, install_repository=repository, + install_plugin=install_plugin, check_repository=repository, + check_file_map=check_file_map, + msg_lv='error' if unuse_lib_repository else 'warn') if not ret: return False elif ret.get_return('checked'): @@ -1359,11 +1482,9 @@ def install_repository_to_servers(self, components, cluster_config, repository, return False lib_repository = repositories_lib_map[repository]['repositories'] install_plugin = repositories_lib_map[repository]['install_plugin'] - ret = install_repo_plugin(components, ssh_clients, cluster_config, [], options, - self.stdio, - obd_home=self.home_path, install_repository=lib_repository, - install_plugin=install_plugin, check_repository=repository, - check_file_map=check_file_map, msg_lv='error') + ret = self.call_plugin(install_repo_plugin, repository, obd_home=self.home_path, install_repository=lib_repository, + install_plugin=install_plugin, check_repository=repository, + check_file_map=check_file_map, msg_lv='error') if not ret or not ret.get_return('checked'): self._call_stdio('error', 'Failed to install lib package for cluster servers') return False @@ -1376,11 +1497,9 @@ def install_repositories_to_servers(self, deploy_config, repositories, install_p cluster_config = deploy_config.components[repository.name] install_plugin = install_plugins[repository] check_file_map = check_file_maps[repository] = install_plugin.file_map(repository) - ret = install_repo_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, - obd_home=self.home_path, install_repository=repository, - install_plugin=install_plugin, check_repository=repository, - check_file_map=check_file_map, - msg_lv='error' if deploy_config.unuse_lib_repository else 'warn') + ret = self.call_plugin(install_repo_plugin, repository, obd_home=self.home_path, install_repository=repository, + install_plugin=install_plugin, check_repository=repository, check_file_map=check_file_map, + msg_lv='error' if deploy_config.unuse_lib_repository else 'warn') if not ret: return False if not ret.get_return('checked'): @@ -1400,11 +1519,9 @@ def install_repositories_to_servers(self, deploy_config, repositories, install_p check_file_map = check_file_maps[need_lib_repository] lib_repository = repositories_lib_map[need_lib_repository]['repositories'] install_plugin = repositories_lib_map[need_lib_repository]['install_plugin'] - ret = install_repo_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], options, - self.stdio, - obd_home=self.home_path, install_repository=lib_repository, - install_plugin=install_plugin, check_repository=need_lib_repository, - check_file_map=check_file_map, msg_lv='error') + ret = self.call_plugin(install_repo_plugin, need_lib_repository, obd_home=self.home_path, install_repository=lib_repository, + install_plugin=install_plugin, check_repository=need_lib_repository, + check_file_map=check_file_map, msg_lv='error') if not ret or not ret.get_return('checked'): self._call_stdio('error', 'Failed to install lib package for cluster servers') return False @@ -1414,13 +1531,13 @@ def sync_runtime_dependencies(self, deploy_config, repositories, ssh_clients, op rsync_plugin = self.plugin_manager.get_best_py_script_plugin('rsync', 'general', '0.1') ret = True for repository in repositories: - cluster_config = deploy_config.components[repository.name] - ret = rsync_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], option, self.stdio) and ret + ret = self.call_plugin(rsync_plugin, repository) and ret return ret - def start_cluster(self, name, cmd=[], options=Values()): + def start_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -1434,7 +1551,7 @@ def start_cluster(self, name, cmd=[], options=Values()): if deploy_info.config_status == DeployConfigStatus.NEED_REDEPLOY: self._call_stdio('error', 'Deploy needs redeploy') return False - if deploy_info.config_status != DeployConfigStatus.UNCHNAGE and not getattr(options, 'without_parameter', False): + if deploy_info.config_status != DeployConfigStatus.UNCHNAGE and not getattr(self.options, 'without_parameter', False): self._call_stdio('error', 'Deploy %s.%s\nIf you still need to start the cluster, use the `obd cluster start %s --wop` option to start the cluster without loading parameters. ' % (deploy_info.config_status.value, deploy.effect_tip(), name)) return False @@ -1442,17 +1559,18 @@ def start_cluster(self, name, cmd=[], options=Values()): # Get the repository repositories = self.load_local_repositories(deploy_info, False) + self.set_repositories(repositories) self._call_stdio('stop_loading', 'succeed') - return self._start_cluster(deploy, repositories, cmd, options) + return self._start_cluster(deploy, repositories) - def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): + def _start_cluster(self, deploy, repositories): self._call_stdio('verbose', 'Get deploy config') deploy_config = deploy.deploy_config deploy_info = deploy.deploy_info name = deploy.name update_deploy_status = True - components = getattr(options, 'components', '') + components = getattr(self.options, 'components', '') if components: components = components.split(',') for component in components: @@ -1464,12 +1582,12 @@ def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): else: components = deploy_info.components.keys() - servers = getattr(options, 'servers', '') + servers = getattr(self.options, 'servers', '') server_list = servers.split(',') if servers else [] self._call_stdio('start_loading', 'Search plugins') start_check_plugins = self.search_py_script_plugin(repositories, 'start_check', no_found_act='warn') - create_tenant_plugins = self.search_py_script_plugin(repositories, 'create_tenant', no_found_act='ignore') if deploy_config.auto_create_tenant else {} + create_tenant_plugins = self.search_py_script_plugin(repositories, 'create_tenant', no_found_act='ignore') start_plugins = self.search_py_script_plugin(repositories, 'start') connect_plugins = self.search_py_script_plugin(repositories, 'connect') bootstrap_plugins = self.search_py_script_plugin(repositories, 'bootstrap') @@ -1487,16 +1605,18 @@ def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): # Check the status for the deployed cluster component_status = {} if DeployStatus.STATUS_RUNNING == deploy_info.status: - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status == 1: self._call_stdio('print', 'Deploy "%s" is running' % name) return True repositories = self.sort_repository_by_depend(repositories, deploy_config) - strict_check = getattr(options, 'strict_check', False) + strict_check = getattr(self.options, 'strict_check', False) success = True repository_dir_map = {} + repositories_start_all = {} + start_repositories = [] for repository in repositories: repository_dir_map[repository.name] = repository.repository_dir if repository.name not in components: @@ -1504,41 +1624,36 @@ def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): if repository not in start_check_plugins: continue cluster_config = deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (start_check_plugins[repository], repository)) - ret = start_check_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, cmd, options, self.stdio, strict_check=strict_check) + cluster_servers = cluster_config.servers + if servers: + cluster_config.servers = [srv for srv in cluster_servers if srv.ip in server_list or srv.name in server_list] + repositories_start_all[repository] = start_all = cluster_servers == cluster_config.servers + update_deploy_status = update_deploy_status and start_all + if not cluster_config.servers: + continue + ret = self.call_plugin(start_check_plugins[repository], repository, strict_check=strict_check) if not ret: + self._call_stdio('verbose', '%s starting check failed.' % repository.name) success = False + start_repositories.append(repository) if success is False: # self._call_stdio('verbose', 'Starting check failed. Use --skip-check to skip the starting check. However, this may lead to a starting failure.') return False - component_num = len(components) + component_num = len(start_repositories) display_repositories = [] connect_ret = {} - for repository in repositories: - if repository.name not in components: - continue - cluster_config = deploy_config.components[repository.name] - cluster_servers = cluster_config.servers - if servers: - cluster_config.servers = [srv for srv in cluster_servers if srv.ip in server_list or srv.name in server_list] - if not cluster_config.servers: - component_num -= 1 - continue - start_all = cluster_servers == cluster_config.servers - update_deploy_status = update_deploy_status and start_all - - self._call_stdio('verbose', 'Call %s for %s' % (start_plugins[repository], repository)) - ret = start_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, cmd, options, self.stdio, self.home_path, repository.repository_dir, repository_dir_map=repository_dir_map, deploy_name=deploy.name) + for repository in start_repositories: + start_all = repositories_start_all[repository] + ret = self.call_plugin(start_plugins[repository], repository, local_home_path=self.home_path, repository_dir_map=repository_dir_map) if ret: need_bootstrap = ret.get_return('need_bootstrap') else: self._call_stdio('error', '%s start failed' % repository.name) break - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugins[repository], repository)) - ret = connect_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, cmd, options, self.stdio) + ret = self.call_plugin(connect_plugins[repository], repository) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') @@ -1547,17 +1662,19 @@ def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): break if need_bootstrap and start_all: - self._call_stdio('start_loading', 'Initialize cluster') - self._call_stdio('verbose', 'Call %s for %s' % (bootstrap_plugins[repository], repository)) - if not bootstrap_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, cmd, options, self.stdio, cursor): + self._call_stdio('start_loading', 'Initialize %s' % repository.name) + if not self.call_plugin(bootstrap_plugins[repository], repository, cursor=cursor): self._call_stdio('stop_loading', 'fail') self._call_stdio('error', 'Cluster init failed') break self._call_stdio('stop_loading', 'succeed') if repository in create_tenant_plugins: - create_tenant_options = Values({"variables": "ob_tcp_invited_nodes='%'"}) - self._call_stdio('verbose', 'Call %s for %s' % (bootstrap_plugins[repository], repository)) - create_tenant_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], create_tenant_options, self.stdio, cursor) + if self.get_namespace(repository.name).get_variable("create_tenant_options"): + self.call_plugin(create_tenant_plugins[repository], repository, cursor=cursor) + + if deploy_config.auto_create_tenant: + create_tenant_options = Values({"variables": "ob_tcp_invited_nodes='%'", "create_if_not_exists": True}) + self.call_plugin(create_tenant_plugins[repository], repository, create_tenant_options=create_tenant_options, cursor=cursor) if not start_all: component_num -= 1 @@ -1565,9 +1682,7 @@ def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): display_repositories.append(repository) for repository in display_repositories: - cluster_config = deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (display_plugins[repository], repository)) - if display_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, cmd, options, self.stdio, **connect_ret[repository]): + if self.call_plugin(display_plugins[repository], repository, **connect_ret[repository]): component_num -= 1 if component_num == 0: @@ -1581,9 +1696,10 @@ def _start_cluster(self, deploy, repositories, cmd=None, options=Values()): return True return False - def create_tenant(self, name, options=Values()): + def create_tenant(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -1599,6 +1715,7 @@ def create_tenant(self, name, options=Values()): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -1611,25 +1728,23 @@ def create_tenant(self, name, options=Values()): ssh_clients = self.get_clients(deploy_config, repositories) for repository in create_tenant_plugins: - cluster_config = deploy_config.components[repository.name] db = None cursor = None - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugins[repository], repository)) - ret = connect_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) + ret = self.call_plugin(connect_plugins[repository], repository) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') if not db: return False - self._call_stdio('verbose', 'Call %s for %s' % (create_tenant_plugins[repository], repository)) - if not create_tenant_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, cursor): + if not self.call_plugin(create_tenant_plugins[repository], repository, cursor=cursor): return False return True - def drop_tenant(self, name, options=Values()): + def drop_tenant(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -1645,6 +1760,7 @@ def drop_tenant(self, name, options=Values()): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -1660,29 +1776,73 @@ def drop_tenant(self, name, options=Values()): cluster_config = deploy_config.components[repository.name] db = None cursor = None - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugins[repository], repository)) - ret = connect_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) + ret = self.call_plugin(connect_plugins[repository], repository) + if ret: + db = ret.get_return('connect') + cursor = ret.get_return('cursor') + if not db: + return False + + if not self.call_plugin(drop_tenant_plugins[repository], repository, cursor=cursor): + return False + return True + + def list_tenant(self, name): + self._call_stdio('verbose', 'Get Deploy by name') + deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) + if not deploy: + self._call_stdio('error', 'No such deploy: %s.' % name) + return False + + deploy_info = deploy.deploy_info + self._call_stdio('verbose', 'Deploy status judge') + if deploy_info.status != DeployStatus.STATUS_RUNNING: + self._call_stdio('print', 'Deploy "%s" is %s' % (name, deploy_info.status.value)) + return False + self._call_stdio('verbose', 'Get deploy config') + deploy_config = deploy.deploy_config + + self._call_stdio('start_loading', 'Get local repositories and plugins') + # Get the repository + repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) + + # Check whether the components have the parameter plugins and apply the plugins + self.search_param_plugin_and_apply(repositories, deploy_config) + connect_plugins = self.search_py_script_plugin(repositories, 'connect') + list_tenant_plugins = self.search_py_script_plugin(repositories, 'list_tenant', no_found_act='ignore') + self._call_stdio('stop_loading', 'succeed') + + # Get the client + ssh_clients = self.get_clients(deploy_config, repositories) + + for repository in list_tenant_plugins: + cluster_config = deploy_config.components[repository.name] + db = None + cursor = None + ret = self.call_plugin(connect_plugins[repository], repository) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') if not db: return False - self._call_stdio('verbose', 'Call %s for %s' % (drop_tenant_plugins[repository], repository)) - if not drop_tenant_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, cursor): + if not self.call_plugin(list_tenant_plugins[repository], repository, cursor=cursor): return False return True def reload_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s. Input the configuration path to create a new deploy' % name) return False deploy_info = deploy.deploy_info self._call_stdio('verbose', 'Deploy status judge') - if deploy_info.status != DeployStatus.STATUS_RUNNING: + if deploy_info.status not in [DeployStatus.STATUS_RUNNING, DeployStatus.STATUS_STOPPED]: self._call_stdio('error', 'Deploy "%s" is %s. You could not reload an %s cluster.' % (name, deploy_info.status.value, deploy_info.status.value)) return False @@ -1715,6 +1875,7 @@ def _reload_cluster(self, deploy): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) reload_plugins = self.search_py_script_plugin(repositories, 'reload') connect_plugins = self.search_py_script_plugin(repositories, 'connect') @@ -1732,16 +1893,16 @@ def _reload_cluster(self, deploy): # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: - if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) - for repository in component_status: - cluster_status = component_status[repository] - for server in cluster_status: - if cluster_status[server] == 0: - self._call_stdio('print', '%s %s is stopped' % (server, repository.name)) - return False + sub_io = None + if getattr(self.stdio, 'sub_io'): + sub_io = self.stdio.sub_io(msg_lv=MsgLevel.ERROR) + obd = self.fork(options=Values({'without_parameter': True}), stdio=sub_io) + if not obd._start_cluster(deploy, repositories): + if self.stdio: + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) + return False repositories = self.sort_repositories_by_depends(deploy_config, repositories) component_num = len(repositories) @@ -1749,18 +1910,16 @@ def _reload_cluster(self, deploy): cluster_config = deploy_config.components[repository.name] new_cluster_config = new_deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugins[repository], repository)) - ret = connect_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) + ret = self.call_plugin(connect_plugins[repository], repository) + if not ret: + ret = self.call_plugin(connect_plugins[repository], repository, components=new_deploy_config.components.keys(), cluster_config=new_cluster_config) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') else: continue - self._call_stdio('verbose', 'Call %s for %s' % (reload_plugins[repository], repository)) - if not reload_plugins[repository]( - deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, - cursor=cursor, new_cluster_config=new_cluster_config, repository_dir=repository.repository_dir, deploy_name=deploy.name): + if not self.call_plugin(reload_plugins[repository], repository, cursor=cursor, new_cluster_config=new_cluster_config): continue component_num -= 1 if component_num == 0: @@ -1775,6 +1934,7 @@ def _reload_cluster(self, deploy): def display_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -1791,6 +1951,7 @@ def display_cluster(self, name): # Get the repository repositories = self.load_local_repositories(deploy_info) repositories = self.sort_repository_by_depend(repositories, deploy_config) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -1804,37 +1965,35 @@ def display_cluster(self, name): # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) - if cluster_status is False or cluster_status == 0: - if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) - for repository in component_status: - cluster_status = component_status[repository] - for server in cluster_status: - if cluster_status[server] == 0: - self._call_stdio('print', '%s %s is stopped' % (server, repository.name)) - return False + self.cluster_status_check(repositories, component_status) for repository in repositories: - cluster_config = deploy_config.components[repository.name] + cluster_status = component_status[repository] + servers = [] + for server in cluster_status: + if cluster_status[server] == 0: + self._call_stdio('warn', '%s %s is stopped' % (server, repository.name)) + else: + servers.append(server) + if not servers: + continue db = None cursor = None - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugins[repository], repository)) - ret = connect_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) + ret = self.call_plugin(connect_plugins[repository], repository) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') if not db: - return False + continue - self._call_stdio('verbose', 'Call %s for %s' % (display_plugins[repository], repository)) - display_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, cursor) + self.call_plugin(display_plugins[repository], repository, cursor=cursor) return True - def stop_cluster(self, name, options=Values()): + def stop_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -1842,7 +2001,7 @@ def stop_cluster(self, name, options=Values()): deploy_info = deploy.deploy_info self._call_stdio('verbose', 'Check the deploy status') status = [DeployStatus.STATUS_DEPLOYED, DeployStatus.STATUS_STOPPED, DeployStatus.STATUS_RUNNING] - if getattr(options, 'force', False): + if getattr(self.options, 'force', False): status.append(DeployStatus.STATUS_UPRADEING) if deploy_info.status not in status: self._call_stdio('error', 'Deploy "%s" is %s. You could not stop an %s cluster.' % (name, deploy_info.status.value, deploy_info.status.value)) @@ -1851,17 +2010,18 @@ def stop_cluster(self, name, options=Values()): self._call_stdio('start_loading', 'Get local repositories') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) self._call_stdio('stop_loading', 'succeed') - return self._stop_cluster(deploy, repositories, options) + return self._stop_cluster(deploy, repositories) - def _stop_cluster(self, deploy, repositories, options=Values()): + def _stop_cluster(self, deploy, repositories): self._call_stdio('verbose', 'Get deploy config') deploy_config = deploy.deploy_config deploy_info = deploy.deploy_info name = deploy.name update_deploy_status = True - components = getattr(options, 'components', '') + components = getattr(self.options, 'components', '') if components: components = components.split(',') for component in components: @@ -1873,7 +2033,7 @@ def _stop_cluster(self, deploy, repositories, options=Values()): else: components = deploy_info.components.keys() - servers = getattr(options, 'servers', '') + servers = getattr(self.options, 'servers', '') server_list = servers.split(',') if servers else [] self._call_stdio('start_loading', 'Search plugins') @@ -1902,8 +2062,7 @@ def _stop_cluster(self, deploy, repositories, options=Values()): start_all = cluster_servers == cluster_config.servers update_deploy_status = update_deploy_status and start_all - self._call_stdio('verbose', 'Call %s for %s' % (stop_plugins[repository], repository)) - if stop_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio): + if self.call_plugin(stop_plugins[repository], repository): component_num -= 1 if component_num == 0: @@ -1917,25 +2076,38 @@ def _stop_cluster(self, deploy, repositories, options=Values()): return True return False - def restart_cluster(self, name, options=Values()): + def restart_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False deploy_info = deploy.deploy_info + status = [DeployStatus.STATUS_DEPLOYED, DeployStatus.STATUS_STOPPED, DeployStatus.STATUS_RUNNING] + if deploy_info.status not in status: + self._call_stdio('error', 'Deploy "%s" is %s. You could not restart an %s cluster.' % (name, deploy_info.status.value, deploy_info.status.value)) + return False + if deploy_info.config_status == DeployConfigStatus.NEED_REDEPLOY: self._call_stdio('error', 'Deploy needs redeploy') return False + self._call_stdio('verbose', 'Deploy status judge') + if deploy_info.status not in [DeployStatus.STATUS_RUNNING, DeployStatus.STATUS_STOPPED]: + self._call_stdio('error', 'Deploy "%s" is %s. You could not restart an %s cluster.' % (name, deploy_info.status.value, deploy_info.status.value)) + return False + self._call_stdio('start_loading', 'Get local repositories and plugins') deploy_config = deploy.deploy_config # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) restart_plugins = self.search_py_script_plugin(repositories, 'restart') reload_plugins = self.search_py_script_plugin(repositories, 'reload') + start_check_plugins = self.search_py_script_plugin(repositories, 'start_check') start_plugins = self.search_py_script_plugin(repositories, 'start') stop_plugins = self.search_py_script_plugin(repositories, 'stop') connect_plugins = self.search_py_script_plugin(repositories, 'connect') @@ -1947,7 +2119,7 @@ def restart_cluster(self, name, options=Values()): self._call_stdio('start_loading', 'Load cluster param plugin') # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) - if getattr(options, 'without_parameter', False) is False and deploy_info.config_status != DeployConfigStatus.UNCHNAGE: + if getattr(self.options, 'without_parameter', False) is False and deploy_info.config_status != DeployConfigStatus.UNCHNAGE: apply_change = True new_deploy_config = deploy.temp_deploy_config change_user = deploy_config.user.username != new_deploy_config.user.username @@ -1959,7 +2131,7 @@ def restart_cluster(self, name, options=Values()): self._call_stdio('stop_loading', 'succeed') update_deploy_status = True - components = getattr(options, 'components', '') + components = getattr(self.options, 'components', '') if components: components = components.split(',') for component in components: @@ -1974,7 +2146,7 @@ def restart_cluster(self, name, options=Values()): else: components = deploy_info.components.keys() - servers = getattr(options, 'servers', '') + servers = getattr(self.options, 'servers', '') if servers: server_list = servers.split(',') if apply_change: @@ -2005,21 +2177,22 @@ def restart_cluster(self, name, options=Values()): # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: - if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) - for repository in component_status: - cluster_status = component_status[repository] - for server in cluster_status: - if cluster_status[server] == 0: - self._call_stdio('print', '%s %s is stopped' % (server, repository.name)) - return False + sub_io = None + if getattr(self.stdio, 'sub_io'): + sub_io = self.stdio.sub_io(msg_lv=MsgLevel.ERROR) + obd = self.fork(options=Values({'without_parameter': True}), stdio=sub_io) + if not obd._start_cluster(deploy, repositories): + if self.stdio: + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) + return False done_repositories = [] cluster_configs = {} component_num = len(components) repositories = self.sort_repositories_by_depends(deploy_config, repositories) + self.set_repositories(repositories) repository_dir_map = {} for repository in repositories: repository_dir_map[repository.name] = repository.repository_dir @@ -2039,20 +2212,20 @@ def restart_cluster(self, name, options=Values()): start_all = cluster_servers == cluster_config.servers update_deploy_status = update_deploy_status and start_all - self._call_stdio('verbose', 'Call %s for %s' % (restart_plugins[repository], repository)) - if restart_plugins[repository]( - deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, - local_home_path=self.home_path, - start_plugin=start_plugins[repository], - reload_plugin=reload_plugins[repository], - stop_plugin=stop_plugins[repository], - connect_plugin=connect_plugins[repository], - display_plugin=display_plugins[repository], - repository=repository, - new_cluster_config=new_cluster_config, - new_clients=new_ssh_clients, - repository_dir_map=repository_dir_map, - deploy_name=deploy.name, + if self.call_plugin( + restart_plugins[repository], + repository, + local_home_path=self.home_path, + start_check_plugin=start_check_plugins[repository], + start_plugin=start_plugins[repository], + reload_plugin=reload_plugins[repository], + stop_plugin=stop_plugins[repository], + connect_plugin=connect_plugins[repository], + bootstrap_plugin=bootstrap_plugins[repository], + display_plugin=display_plugins[repository], + new_cluster_config=new_cluster_config, + new_clients=new_ssh_clients, + repository_dir_map=repository_dir_map, ): component_num -= 1 done_repositories.append(repository) @@ -2081,31 +2254,30 @@ def restart_cluster(self, name, options=Values()): new_cluster_config = new_deploy_config.components[repository.name] cluster_config = cluster_configs[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (restart_plugins[repository], repository)) - if restart_plugins[repository]( - deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, - local_home_path=self.home_path, - start_plugin=start_plugins[repository], - reload_plugin=reload_plugins[repository], - stop_plugin=stop_plugins[repository], - connect_plugin=connect_plugins[repository], - display_plugin=display_plugins[repository], - repository=repository, - new_cluster_config=new_cluster_config, - new_clients=new_ssh_clients, - rollback=True, - bootstrap_plugin=bootstrap_plugins[repository], - repository_dir_map=repository_dir_map, - deploy_name=deploy.name + if self.call_plugin( + restart_plugins[repository], + repository, + local_home_path=self.home_path, + start_plugin=start_plugins[repository], + reload_plugin=reload_plugins[repository], + stop_plugin=stop_plugins[repository], + connect_plugin=connect_plugins[repository], + display_plugin=display_plugins[repository], + new_cluster_config=new_cluster_config, + new_clients=new_ssh_clients, + rollback=True, + bootstrap_plugin=bootstrap_plugins[repository], + repository_dir_map=repository_dir_map, ): deploy_config.update_component(cluster_config) self._call_stdio('stop_loading', 'succeed') return False - def redeploy_cluster(self, name, opt=Values(), search_repo=True): + def redeploy_cluster(self, name, search_repo=True): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -2116,11 +2288,13 @@ def redeploy_cluster(self, name, opt=Values(), search_repo=True): self._call_stdio('start_loading', 'Get local repositories') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) self._call_stdio('stop_loading', 'succeed') self._call_stdio('verbose', 'Check deploy status') if deploy_info.status in [DeployStatus.STATUS_RUNNING, DeployStatus.STATUS_UPRADEING]: - if not self._stop_cluster(deploy, repositories, options=Values({'force': True})): + obd = self.fork(options=Values({'force': True})) + if not obd._stop_cluster(deploy, repositories): return False elif deploy_info.status not in [DeployStatus.STATUS_STOPPED, DeployStatus.STATUS_DEPLOYED]: self._call_stdio('error', 'Deploy "%s" is %s. You could not destroy an undeployed cluster' % ( @@ -2129,7 +2303,7 @@ def redeploy_cluster(self, name, opt=Values(), search_repo=True): # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) - if not self._destroy_cluster(deploy, repositories, opt): + if not self._destroy_cluster(deploy, repositories): return False if search_repo: if deploy_info.config_status != DeployConfigStatus.UNCHNAGE and not deploy.apply_temp_deploy_config(): @@ -2140,11 +2314,13 @@ def redeploy_cluster(self, name, opt=Values(), search_repo=True): repositories, install_plugins = self.search_components_from_mirrors_and_install(deploy_config) if not repositories or not install_plugins: return False - return self._deploy_cluster(deploy, repositories, opt) and self._start_cluster(deploy, repositories) + self.set_repositories(repositories) + return self._deploy_cluster(deploy, repositories) and self._start_cluster(deploy, repositories) - def destroy_cluster(self, name, opt=Values()): + def destroy_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -2159,11 +2335,13 @@ def destroy_cluster(self, name, opt=Values()): self._call_stdio('start_loading', 'Get local repositories') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) self._call_stdio('stop_loading', 'succeed') self._call_stdio('verbose', 'Check deploy status') if deploy_info.status in [DeployStatus.STATUS_RUNNING, DeployStatus.STATUS_UPRADEING]: - if not self._stop_cluster(deploy, repositories, Values({'force': True})): + obd = self.fork(options=Values({'force': True})) + if not obd._stop_cluster(deploy, repositories): return False elif deploy_info.status not in [DeployStatus.STATUS_STOPPED, DeployStatus.STATUS_DEPLOYED]: self._call_stdio('error', 'Deploy "%s" is %s. You could not destroy an undeployed cluster' % (name, deploy_info.status.value)) @@ -2171,22 +2349,22 @@ def destroy_cluster(self, name, opt=Values()): # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) - return self._destroy_cluster(deploy, repositories, opt) + return self._destroy_cluster(deploy, repositories) - def _destroy_cluster(self, deploy, repositories, opt=Values()): + def _destroy_cluster(self, deploy, repositories): deploy_config = deploy.deploy_config self._call_stdio('start_loading', 'Search plugins') # Get the repository - plugins = self.search_py_script_plugin(repositories, 'destroy') + destroy_plugins = self.search_py_script_plugin(repositories, 'destroy') self._call_stdio('stop_loading', 'succeed') # Get the client ssh_clients = self.get_clients(deploy_config, repositories) # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 1: - if getattr(opt, 'force_kill', False): + if getattr(self.options, 'force_kill', False): self._call_stdio('verbose', 'Try to stop cluster') status = deploy.deploy_info.status deploy.update_deploy_status(DeployStatus.STATUS_RUNNING) @@ -2205,20 +2383,18 @@ def _destroy_cluster(self, deploy, repositories, opt=Values()): return False for repository in repositories: - cluster_config = deploy_config.components[repository.name] + self.call_plugin(destroy_plugins[repository], repository) - self._call_stdio('verbose', 'Call %s for %s' % (plugins[repository], repository)) - plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) - self._call_stdio('verbose', 'Set %s deploy status to destroyed' % deploy.name) if deploy.update_deploy_status(DeployStatus.STATUS_DESTROYED): self._call_stdio('print', '%s destroyed' % deploy.name) return True return False - def reinstall(self, name, options=Values()): + def reinstall(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -2229,8 +2405,8 @@ def reinstall(self, name, options=Values()): self._call_stdio('error', 'Deploy "%s" is %s' % (name, deploy_info.status.value)) return False - component = getattr(options, 'component') - usable = getattr(options, 'hash') + component = getattr(self.options, 'component') + usable = getattr(self.options, 'hash') if not component: self._call_stdio('error', 'Specify the components you want to reinstall.') return False @@ -2246,6 +2422,7 @@ def reinstall(self, name, options=Values()): for current_repository in repositories: if current_repository.name == component: break + self.set_repositories(repositories) stop_plugins = self.search_py_script_plugin([current_repository], 'stop') start_plugins = self.search_py_script_plugin([current_repository], 'start') @@ -2296,29 +2473,28 @@ def reinstall(self, name, options=Values()): if need_restart: # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, [current_repository], component_status) + cluster_status = self.cluster_status_check([current_repository], component_status) if cluster_status is False or cluster_status == 1: - self._call_stdio('verbose', 'Call %s for %s' % (stop_plugins[current_repository], current_repository)) - if not stop_plugins[current_repository](deploy_config.components.keys(), ssh_clients, current_cluster_config, [], options, self.stdio): + if not self.call_plugin(stop_plugins[current_repository], current_repository): return False # install repo to remote servers if need_change_repo: - if not self.install_repositories_to_servers(deploy_config, [dest_repository, ], install_plugins, ssh_clients, options): + if not self.install_repositories_to_servers(deploy_config, [dest_repository, ], install_plugins, ssh_clients, self.options): return False sync_repositories = [dest_repository] repository = dest_repository # sync runtime dependencies - if not self.sync_runtime_dependencies(deploy_config, sync_repositories, ssh_clients, options): + if not self.sync_runtime_dependencies(deploy_config, sync_repositories, ssh_clients, self.options): return False # start cluster if needed if need_restart and deploy_info.status == DeployStatus.STATUS_RUNNING: - self._call_stdio('verbose', 'Call %s for %s' % (start_plugins[current_repository], repository)) - setattr(options, 'without_parameter', True) - if not start_plugins[current_repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, self.home_path, repository.repository_dir, deploy_name=deploy.name) and getattr(options, 'force', False) is False: - self.install_repositories_to_servers(deploy_config, [current_repository, ], install_plugins, ssh_clients, options) + setattr(self.options, 'without_parameter', True) + obd = self.fork(options=self.options) + if not obd.call_plugin(start_plugins[current_repository], current_repository, home_path=self.home_path) and getattr(self.options, 'force', False) is False: + self.install_repositories_to_servers(deploy_config, [current_repository, ], install_plugins, ssh_clients, self.options) return False # update deploy info @@ -2326,9 +2502,10 @@ def reinstall(self, name, options=Values()): deploy.use_model(dest_repository.name, dest_repository) return True - def upgrade_cluster(self, name, options=Values()): + def upgrade_cluster(self, name): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -2344,6 +2521,7 @@ def upgrade_cluster(self, name, options=Values()): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins @@ -2352,10 +2530,10 @@ def upgrade_cluster(self, name, options=Values()): self._call_stdio('stop_loading', 'succeed') if deploy_info.status == DeployStatus.STATUS_RUNNING: - component = getattr(options, 'component') - version = getattr(options, 'version') - usable = getattr(options, 'usable', '') - disable = getattr(options, 'disable', '') + component = getattr(self.options, 'component') + version = getattr(self.options, 'version') + usable = getattr(self.options, 'usable', '') + disable = getattr(self.options, 'disable', '') if component: if component not in deploy_info.components: @@ -2431,7 +2609,7 @@ def upgrade_cluster(self, name, options=Values()): use_images = [] upgrade_route_plugins = self.search_py_script_plugin([current_repository], 'upgrade_route', no_found_act='warn') if current_repository in upgrade_route_plugins: - ret = upgrade_route_plugins[current_repository](deploy_config.components.keys(), ssh_clients, cluster_config, {}, options, self.stdio, current_repository, dest_repository) + ret = self.call_plugin(upgrade_route_plugins[current_repository], current_repository , current_repository=current_repository, dest_repository=dest_repository) route = ret.get_return('route') if not route: return False @@ -2479,29 +2657,26 @@ def upgrade_cluster(self, name, options=Values()): return False upgrade_repositories.append(dest_repository) + self.set_repositories(upgrade_repositories) upgrade_check_plugins = self.search_py_script_plugin(upgrade_repositories, 'upgrade_check', no_found_act='warn') if current_repository in upgrade_check_plugins: connect_plugin = self.search_py_script_plugin(upgrade_repositories, 'connect')[current_repository] db = None cursor = None - self._call_stdio('verbose', 'Call %s for %s' % (connect_plugin, current_repository)) - ret = connect_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio) + ret = self.call_plugin(connect_plugin, current_repository) if ret: db = ret.get_return('connect') cursor = ret.get_return('cursor') if not db: return False - self._call_stdio('verbose', 'Call %s for %s' % (upgrade_check_plugins[current_repository], current_repository)) - if not upgrade_check_plugins[current_repository]( - deploy_config.components.keys(), ssh_clients, cluster_config, {}, options, self.stdio, + if not self.call_plugin( + upgrade_check_plugins[current_repository], current_repository, current_repository=current_repository, - repositories=upgrade_repositories, route=route, cursor=cursor - ): + ): return False cursor.close() - db.close() self._call_stdio( 'print_list', @@ -2544,26 +2719,26 @@ def upgrade_cluster(self, name, options=Values()): if not install_plugins: return False - if not self.install_repositories_to_servers(deploy_config, upgrade_repositories[1:], install_plugins, ssh_clients, options): + if not self.install_repositories_to_servers(deploy_config, upgrade_repositories[1:], install_plugins, ssh_clients, self.options): return False n = len(upgrade_repositories) while upgrade_ctx['index'] < n: - repository = upgrade_repositories[upgrade_ctx['index'] - 1] + repository = upgrade_repositories[upgrade_ctx['index']] repositories = [repository] upgrade_plugin = self.search_py_script_plugin(repositories, 'upgrade')[repository] - - ret = upgrade_plugin( - deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, - search_py_script_plugin=self.search_py_script_plugin, - local_home_path=self.home_path, - current_repository=current_repository, - upgrade_repositories=upgrade_repositories, - apply_param_plugin=lambda repository: self.search_param_plugin_and_apply([repository], deploy_config), - upgrade_ctx=upgrade_ctx, - install_repository_to_servers=self.install_repository_to_servers, - unuse_lib_repository=deploy_config.unuse_lib_repository - ) + self.set_repositories(repositories) + ret = self.call_plugin( + upgrade_plugin, repository, + search_py_script_plugin=self.search_py_script_plugin, + local_home_path=self.home_path, + current_repository=current_repository, + upgrade_repositories=upgrade_repositories, + apply_param_plugin=lambda repository: self.search_param_plugin_and_apply([repository], deploy_config), + upgrade_ctx=upgrade_ctx, + install_repository_to_servers=self.install_repository_to_servers, + unuse_lib_repository=deploy_config.unuse_lib_repository + ) deploy.update_upgrade_ctx(**upgrade_ctx) if not ret: return False @@ -2572,10 +2747,10 @@ def upgrade_cluster(self, name, options=Values()): return True - def create_repository(self, options): - force = getattr(options, 'force', False) + def create_repository(self): + force = getattr(self.options, 'force', False) necessary = ['name', 'version', 'path'] - attrs = options.__dict__ + attrs = self.options.__dict__ success = True for key in necessary: if key not in attrs or not attrs[key]: @@ -2614,7 +2789,7 @@ def create_repository(self, options): self._call_stdio('start_loading', 'Package') try: - pkg = LocalPackage(repo_path, attrs['name'], attrs['version'], files, getattr(options, 'release', None), getattr(options, 'arch', None)) + pkg = LocalPackage(repo_path, attrs['name'], attrs['version'], files, getattr(self.options, 'release', None), getattr(self.options, 'arch', None)) self._call_stdio('stop_loading', 'succeed') except: self._call_stdio('exception', 'Package failed') @@ -2638,7 +2813,9 @@ def create_repository(self, options): self._call_stdio('error', 'Repository(%s) existed' % tag_repository.repository_dir) return True - def _test_optimize_init(self, opts, test_name, deploy_config, cluster_config): + def _test_optimize_init(self, test_name, repository): + opts = self.options + deploy_config = self.deploy.deploy_config optimize_config_path = getattr(opts, 'optimize_config', None) if optimize_config_path: self._call_stdio('verbose', 'load optimize config {}'.format(optimize_config_path)) @@ -2651,11 +2828,11 @@ def _test_optimize_init(self, opts, test_name, deploy_config, cluster_config): self._call_stdio('verbose', 'Get optimize config') optimize_config = self.optimize_manager.optimize_config check_options_plugin = self.plugin_manager.get_best_py_script_plugin('check_options', 'optimize', '0.1') - self._call_stdio('verbose', 'Call check options plugin for optimize') - return check_options_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio, optimize_config=optimize_config) + return self.call_plugin(check_options_plugin, repository, optimize_config=optimize_config) @staticmethod - def _get_first_db_and_cursor_from_connect(connect_ret): + def _get_first_db_and_cursor_from_connect(namespace): + connect_ret = namespace.get_return('connect') dbs = connect_ret.get_return('connect') cursors = connect_ret.get_return('cursor') if not dbs or not cursors: @@ -2668,22 +2845,10 @@ def _get_first_db_and_cursor_from_connect(connect_ret): else: return dbs, cursors - def _test_optimize_operation(self, deploy, optimize_envs, connect_context, stage=None, opts=None, operation='optimize'): + def _test_optimize_operation(self, repository, ob_repository, optimize_envs, connect_namespaces, connect_plugin, stage=None, operation='optimize'): """ - - :param deploy: :param stage: optimize stage :param optimize_envs: envs for optimize plugin - :param connect_context: { - "": { - "db": db, - "cursor": cursor, - "connect_kwargs": { - "component": , - "target_server": "server1" # kwargs for connect plugin - } - } - } :param operation: "optimize" or "recover" :return: """ @@ -2693,73 +2858,55 @@ def _test_optimize_operation(self, deploy, optimize_envs, connect_context, stage self._call_stdio('verbose', 'Recover the optimizes') else: raise Exception("Invalid optimize operation!") - deploy_config = deploy.deploy_config ob_cursor = None odp_cursor = None - cluster_config = None - for component in connect_context.keys(): - self._call_stdio('verbose', 'get cursor for component {}'.format(component)) - connect_context[component] = connect_context.get(component, {}) - cursor = connect_context[component].get('cursor') - db = connect_context[component].get('db') - if not cursor or not db: - self._call_stdio('verbose', 'cursor not found for component {}, try to connect'.format(component)) - connect_kwargs = connect_context[component].get('connect_kwargs', {}) - ret = self._get_connect(deploy, **connect_kwargs) - db, cursor = self._get_first_db_and_cursor_from_connect(ret) - connect_context[component]['db'] = db - cursor = connect_context[component]['cursor'] = cursor - if component in ['oceanbase', 'oceanbase-ce']: - ob_cursor = cursor - elif component in ['obproxy', 'obproxy-ce']: - odp_cursor = cursor - cluster_config = deploy_config.components[component] + for namespace in connect_namespaces: + db, cursor = self._get_first_db_and_cursor_from_connect(namespace) + if not db or not cursor: + if not self.call_plugin(connect_plugin, repository, spacename=namespace.spacename): + raise Exception('call connect plugin for {} failed'.format(namespace.spacename)) + if namespace.spacename in ['oceanbase', 'oceanbase-ce']: + ob_db, ob_cursor = db, cursor + elif namespace.spacename in ['obproxy', 'obproxy-ce']: + odp_db, odp_cursor = db, cursor operation_plugin = self.plugin_manager.get_best_py_script_plugin(operation, 'optimize', '0.1') optimize_config = self.optimize_manager.optimize_config - kwargs = dict(optimize_config=optimize_config, stage=stage, ob_cursor=ob_cursor, odp_cursor=odp_cursor, optimize_envs=optimize_envs) - self._call_stdio('verbose', 'Call {} plugin.'.format(operation)) - ret = operation_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio, **kwargs) + ret = self.call_plugin(operation_plugin, repository, + optimize_config=optimize_config, stage=stage, + ob_cursor=ob_cursor, odp_cursor=odp_cursor, optimize_envs=optimize_envs) if ret: restart_components = ret.get_return('restart_components') else: return False if restart_components: self._call_stdio('verbose', 'Components {} need restart.'.format(','.join(restart_components))) - for component in restart_components: - self._call_stdio('verbose', 'close cursor for {}'.format(component)) - connect_context[component]['cursor'].close() - connect_context[component]['db'].close() - ret = self._restart_cluster_for_optimize(deploy.name, restart_components) + for namespace in connect_namespaces: + db, cursor = self._get_first_db_and_cursor_from_connect(namespace) + if cursor: + cursor.close() + ret = self._restart_cluster_for_optimize(self.deploy.name, restart_components) if not ret: return False if operation == 'optimize': - for component, connect_item in connect_context.items(): - connect_kwargs = connect_item['connect_kwargs'] - self._call_stdio('verbose', 'reconnect {} by kwargs {}'.format(component, connect_kwargs)) - if connect_kwargs['component_name'] in restart_components: - ret = self._get_connect(deploy, **connect_kwargs) - if not ret: + for namespace in connect_namespaces: + if not self.call_plugin(connect_plugin, repository, spacename=namespace.spacename): + raise Exception('call connect plugin for {} failed'.format(namespace.spacename)) + if namespace.spacename == ob_repository.name and ob_repository.name in restart_components: + self._call_stdio('verbose', '{}: major freeze for component ready'.format(ob_repository.name)) + self._call_stdio('start_loading', 'Waiting for {} ready'.format(ob_repository.name)) + db, cursor = self._get_first_db_and_cursor_from_connect(namespace) + if not self._major_freeze(repository=ob_repository, cursor=cursor, tenant=optimize_envs.get('tenant')): + self._call_stdio('stop_loading', 'fail') return False - db, cursor = self._get_first_db_and_cursor_from_connect(ret) - connect_context[component]['db'] = db - connect_context[component]['cursor'] = cursor - for component in restart_components: - self._call_stdio('verbose', '{}: major freeze for component ready'.format(component)) - self._call_stdio('start_loading', 'Waiting for {} ready'.format(component)) - cursor = connect_context[component]['cursor'] - if not self._major_freeze(deploy_config, component, cursor=cursor, tenant=optimize_envs.get('tenant')): - self._call_stdio('stop_loading', 'fail') - return False self._call_stdio('stop_loading', 'succeed') return True - def _major_freeze(self, deploy_config, component, **kwargs): - cluster_config = deploy_config.components[component] - major_freeze_plugin = self.plugin_manager.get_best_py_script_plugin('major_freeze', component, cluster_config.version) + def _major_freeze(self, repository, **kwargs): + major_freeze_plugin = self.plugin_manager.get_best_py_script_plugin('major_freeze', repository.name, repository.version) if not major_freeze_plugin: - self._call_stdio('verbose', 'no major freeze plugin for component {}, skip.'.format(component)) + self._call_stdio('verbose', 'no major freeze plugin for component {}, skip.'.format(repository.name)) return True - return major_freeze_plugin(deploy_config.components.keys(), [], cluster_config, [], {}, self.stdio, **kwargs) + return self.call_plugin(major_freeze_plugin, repository, **kwargs) def _restart_cluster_for_optimize(self, deploy_name, components): self._call_stdio('start_loading', 'Restart cluster') @@ -2769,43 +2916,30 @@ def _restart_cluster_for_optimize(self, deploy_name, components): stdio = None obd = ObdHome(self.home_path, self.dev_mode, stdio=stdio) obd.lock_manager.set_try_times(-1) - option = Values({'components': ','.join(components), 'without_parameter': True}) - if obd.stop_cluster(name=deploy_name, options=option) and \ - obd.start_cluster(name=deploy_name, options=option) and obd.display_cluster(name=deploy_name): + obd.set_options(Values({'components': ','.join(components), 'without_parameter': True})) + if obd.stop_cluster(name=deploy_name) and \ + obd.start_cluster(name=deploy_name) and obd.display_cluster(name=deploy_name): self._call_stdio('stop_loading', 'succeed') return True else: self._call_stdio('stop_loading', 'fail') return False - def _get_connect(self, deploy, component_name, **kwargs): - deploy_config = deploy.deploy_config - cluster_config = deploy_config.components[component_name] - connect_plugin = self.plugin_manager.get_best_py_script_plugin('connect', component_name, cluster_config.version) - ret = connect_plugin(deploy_config.components.keys(), [], cluster_config, [], {}, self.stdio, **kwargs) - if not ret or not ret.get_return('connect'): - return None - return ret - - def create_mysqltest_snap(self, deploy, ssh_clients, repositories, create_snap_plugin, start_plugins, stop_plugins, options, snap_configs, env={}): - deploy_config = deploy.deploy_config + def create_mysqltest_snap(self, repositories, create_snap_plugin, start_plugins, stop_plugins, snap_configs, env={}): for repository in repositories: if repository in snap_configs: - cluster_config = deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (stop_plugins[repository], repository)) - if not stop_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio): + if not self.call_plugin(stop_plugins[repository], repository): return False - self._call_stdio('verbose', 'Call %s for %s' % (create_snap_plugin, repository)) - if not create_snap_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env=env, snap_config=snap_configs[repository]): + if not self.call_plugin(create_snap_plugin, repository, env=env, snap_config=snap_configs[repository]): return False - self._call_stdio('verbose', 'Call %s for %s' % (start_plugins[repository], repository)) - if not start_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], options, self.stdio, self.home_path, repository.repository_dir, deploy_name=deploy.name): + if not self.call_plugin(start_plugins[repository], repository, home_path=self.home_path): return False return True def mysqltest(self, name, opts): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -2860,6 +2994,7 @@ def mysqltest(self, name, opts): # Get the repository # repositories = self.get_local_repositories({opts.component: deploy_config.components[opts.component]}) repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) target_repository = None ob_repository = None for repository in repositories: @@ -2886,19 +3021,22 @@ def mysqltest(self, name, opts): # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) for repository in component_status: cluster_status = component_status[repository] for server in cluster_status: if cluster_status[server] == 0: self._call_stdio('print', '%s %s is stopped' % (server, repository.name)) return False + namespace = self.get_namespace(target_repository.name) + namespace.set_variable('target_server', opts.test_server) + namespace.set_variable('connect_proxysys', False) connect_plugin = self.search_py_script_plugin(repositories, 'connect')[target_repository] - ret = connect_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, target_server=opts.test_server, sys_root=False) + ret = self.call_plugin(connect_plugin, target_repository) if not ret or not ret.get_return('connect'): return False db = ret.get_return('connect') @@ -2908,6 +3046,7 @@ def mysqltest(self, name, opts): env['host'] = opts.test_server.ip env['port'] = db.port + namespace.set_variable('env', env) mysqltest_init_plugin = self.plugin_manager.get_best_py_script_plugin('init', 'mysqltest', ob_repository.version) mysqltest_check_opt_plugin = self.plugin_manager.get_best_py_script_plugin('check_opt', 'mysqltest', ob_repository.version) mysqltest_check_test_plugin = self.plugin_manager.get_best_py_script_plugin('check_test', 'mysqltest', ob_repository.version) @@ -2924,13 +3063,11 @@ def mysqltest(self, name, opts): snap_check_plugin = self.plugin_manager.get_best_py_script_plugin('snap_check', 'general', '0.1') snap_configs = self.search_plugins(repositories, PluginType.SNAP_CONFIG, no_found_exit=False) - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_check_opt_plugin, target_repository)) - ret = mysqltest_check_opt_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env) + ret = self.call_plugin(mysqltest_check_opt_plugin, target_repository) if not ret: return False if not env['init_only']: - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_check_test_plugin, target_repository)) - ret = mysqltest_check_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env) + ret = self.call_plugin(mysqltest_check_test_plugin, target_repository) if not ret: self._call_stdio('error', 'Failed to get test set') return False @@ -2940,15 +3077,13 @@ def mysqltest(self, name, opts): use_snap = False if env['need_init'] or env['init_only']: - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_init_plugin, target_repository)) - if not mysqltest_init_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env): + if not self.call_plugin(mysqltest_init_plugin, target_repository, env=env): self._call_stdio('error', 'Failed to init for mysqltest') return False if fast_reboot: - if not self.create_mysqltest_snap(deploy, ssh_clients, repositories, create_snap_plugin, start_plugins, stop_plugins, opts, snap_configs, env): + if not self.create_mysqltest_snap(repositories, create_snap_plugin, start_plugins, stop_plugins, snap_configs, env): return False - connect_plugin = self.search_py_script_plugin(repositories, 'connect')[target_repository] - ret = connect_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, target_server=opts.test_server, sys_root=False) + ret = self.call_plugin(connect_plugin, target_repository) if not ret or not ret.get_return('connect'): return False db = ret.get_return('connect') @@ -2958,8 +3093,7 @@ def mysqltest(self, name, opts): env['port'] = db.port self._call_stdio('start_loading', 'Check init') env['load_snap'] = True - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_init_plugin, target_repository)) - mysqltest_init_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env) + self.call_plugin(mysqltest_init_plugin, target_repository) env['load_snap'] = False self._call_stdio('stop_loading', 'succeed') use_snap = True @@ -2970,15 +3104,13 @@ def mysqltest(self, name, opts): if fast_reboot and use_snap is False: self._call_stdio('start_loading', 'Check init') env['load_snap'] = True - mysqltest_init_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env) + self.call_plugin(mysqltest_init_plugin, target_repository) env['load_snap'] = False self._call_stdio('stop_loading', 'succeed') snap_num = 0 for repository in repositories: if repository in snap_configs: - cluster_config = deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (snap_check_plugin, repository)) - if not snap_check_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env=env, snap_config=snap_configs[repository]): + if not self.call_plugin(snap_check_plugin, repository, env=env, snap_config=snap_configs[repository]): break snap_num += 1 use_snap = len(snap_configs) == snap_num @@ -2988,18 +3120,14 @@ def mysqltest(self, name, opts): self._call_stdio('verbose', 'total: {}'.format(len(env['test_set']))) reboot_success = True while True: - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_run_test_plugin, target_repository)) - ret = mysqltest_run_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env) + ret = self.call_plugin(mysqltest_run_test_plugin, target_repository) if not ret: break - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_collect_log_plugin, target_repository)) - mysqltest_collect_log_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, - self.stdio, env) + self.call_plugin(mysqltest_collect_log_plugin, target_repository) if ret.get_return('finished'): break if ret.get_return('reboot') and not env['disable_reboot']: cursor.close() - db.close() if getattr(self.stdio, 'sub_io'): stdio = self.stdio.sub_io(msg_lv=MsgLevel.ERROR) else: @@ -3015,33 +3143,28 @@ def mysqltest(self, name, opts): for repository in repositories: if repository in snap_configs: cluster_config = deploy_config.components[repository.name] - self._call_stdio('verbose', 'Call %s for %s' % (stop_plugins[repository], repository)) - if not stop_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, stdio): + if not self.call_plugin(stop_plugins[repository]): self._call_stdio('stop_loading', 'fail') continue - self._call_stdio('verbose', 'Call %s for %s' % (load_snap_plugin, repository)) - if not load_snap_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, stdio, env=env, snap_config=snap_configs[repository]): + if not self.call_plugin(load_snap_plugin, repository, env=env, snap_config=snap_configs[repository]): self._call_stdio('stop_loading', 'fail') continue - if not start_plugins[repository](deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, stdio, self.home_path, repository.repository_dir, deploy_name=deploy.name): + if not self.call_plugin(start_plugins[repository], repository, home_path=self.home_path): self._call_stdio('stop_loading', 'fail') continue else: self._call_stdio('start_loading', 'Reboot') obd = ObdHome(self.home_path, self.dev_mode, stdio=stdio) obd.lock_manager.set_try_times(-1) - if not obd.redeploy_cluster( - name, - opt=Values({'force_kill': True, 'force': True, 'force_delete': True}), search_repo=False): + obd.set_options(Values({'force_kill': True, 'force': True, 'force_delete': True})) + if not obd.redeploy_cluster(name, search_repo=False): self._call_stdio('stop_loading', 'fail') continue obd.lock_manager.set_try_times(6000) obd = None self._call_stdio('stop_loading', 'succeed') - connect_plugin = self.search_py_script_plugin(repositories, 'connect')[target_repository] - ret = connect_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, - self.stdio, target_server=opts.test_server, sys_root=False) + ret = self.call_plugin(connect_plugin, target_repository) if not ret or not ret.get_return('connect'): self._call_stdio('error', 'Failed to connect server') continue @@ -3049,30 +3172,25 @@ def mysqltest(self, name, opts): cursor = ret.get_return('cursor') env['cursor'] = cursor - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_init_plugin, target_repository)) - if mysqltest_init_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, - self.stdio, env): + if self.call_plugin(mysqltest_init_plugin, target_repository): if fast_reboot and use_snap is False: - if not self.create_mysqltest_snap(deploy, ssh_clients, repositories, create_snap_plugin, start_plugins, stop_plugins, opts, snap_configs, env): + if not self.create_mysqltest_snap(repositories, create_snap_plugin, start_plugins, stop_plugins, snap_configs, env): return False use_snap = True - connect_plugin = self.search_py_script_plugin(repositories, 'connect')[target_repository] - ret = connect_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, - self.stdio, target_server=opts.test_server, sys_root=False) + ret = self.call_plugin(connect_plugin, target_repository) if not ret or not ret.get_return('connect'): self._call_stdio('error', 'Failed to connect server') continue db = ret.get_return('connect') cursor = ret.get_return('cursor') env['cursor'] = cursor - self._call_stdio('verbose', 'Call %s for %s' % (mysqltest_init_plugin, target_repository)) - mysqltest_init_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env) + self.call_plugin(mysqltest_init_plugin, target_repository) reboot_success = True else: self._call_stdio('error', 'Failed to prepare for mysqltest') if not reboot_success: env['collect_log'] = True - mysqltest_collect_log_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], {}, self.stdio, env, test_name='reboot_failed') + self.call_plugin(mysqltest_collect_log_plugin, target_repository, test_name='reboot_failed') break result = env.get('case_results', []) passcnt = len(list(filter(lambda x: x["ret"] == 0, result))) @@ -3097,6 +3215,7 @@ def mysqltest(self, name, opts): def sysbench(self, name, opts): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -3148,21 +3267,20 @@ def sysbench(self, name, opts): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) + self.get_clients(deploy_config, repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) self._call_stdio('stop_loading', 'succeed') - # Get the client - ssh_clients = self.get_clients(deploy_config, repositories) - # Check the status for the deployed cluster if not getattr(opts, 'skip_cluster_status_check', False): component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) for repository in component_status: cluster_status = component_status[repository] for server in cluster_status: @@ -3172,46 +3290,39 @@ def sysbench(self, name, opts): ob_repository = None repository = None - env = {'sys_root': False} - odp_db = None - odp_cursor = None - ob_component = None - connect_context = {} + connect_namespaces = [] for tmp_repository in repositories: if tmp_repository.name in ["oceanbase", "oceanbase-ce"]: ob_repository = tmp_repository - ob_component = tmp_repository.name if tmp_repository.name == opts.component: repository = tmp_repository - if tmp_repository.name in ['obproxy', 'obproxy-ce']: - odp_component = tmp_repository.name - allow_components = ['oceanbase', 'oceanbase-ce'] - for component_name in deploy_config.components: - if component_name in allow_components: - config = deploy_config.components[component_name] - env['user'] = 'root' - env['password'] = config.get_global_conf().get('root_password', '') - env['target_server'] = opts.test_server - break - connect_kwargs = dict(component_name=odp_component, target_server=opts.test_server) - ret = self._get_connect(deploy, **connect_kwargs) - if not ret or not ret.get_return('connect'): - return False - odp_db, odp_cursor = self._get_first_db_and_cursor_from_connect(ret) - connect_context[tmp_repository.name] = {'connect_kwargs': connect_kwargs, 'db': odp_db, - 'cursor': odp_cursor} if not ob_repository: self._call_stdio('error', 'Deploy {} must contain the component oceanbase or oceanbase-ce.'.format(deploy.name)) return False + sys_namespace = self.get_namespace(ob_repository.name) + connect_plugin = self.plugin_manager.get_best_py_script_plugin('connect', repository.name, repository.version) + if repository.name in ['obproxy', 'obproxy-ce']: + for component_name in deploy_config.components: + if component_name in ['oceanbase', 'oceanbase-ce']: + ob_cluster_config = deploy_config.components[component_name] + sys_namespace.set_variable("connect_proxysys", False) + sys_namespace.set_variable("user", "root") + sys_namespace.set_variable("password", ob_cluster_config.get_global_conf().get('root_password', '')) + sys_namespace.set_variable("target_server", opts.test_server) + break + proxysys_namespace = self.get_namespace(repository.name) + proxysys_namespace.set_variable("component_name", repository) + proxysys_namespace.set_variable("target_server", opts.test_server) + ret = self.call_plugin(connect_plugin, repository, spacename=proxysys_namespace.spacename) + if not ret or not ret.get_return('connect'): + return False + connect_namespaces.append(proxysys_namespace) plugin_version = ob_repository.version if ob_repository else repository.version - - connect_kwargs = dict(component_name=repository.name, **env) - ret = self._get_connect(deploy=deploy, **connect_kwargs) + ret = self.call_plugin(connect_plugin, repository, spacename=sys_namespace.spacename) if not ret or not ret.get_return('connect'): return False - db, cursor = self._get_first_db_and_cursor_from_connect(ret) - connect_context[ob_component] = {'connect_kwargs': connect_kwargs, 'db': db, 'cursor': cursor} - + connect_namespaces.append(sys_namespace) + db, cursor = self._get_first_db_and_cursor_from_connect(namespace=sys_namespace) pre_test_plugin = self.plugin_manager.get_best_py_script_plugin('pre_test', 'sysbench', plugin_version) run_test_plugin = self.plugin_manager.get_best_py_script_plugin('run_test', 'sysbench', plugin_version) @@ -3220,31 +3331,29 @@ def sysbench(self, name, opts): optimization = getattr(opts, 'optimization', 0) - self._call_stdio('verbose', 'Call %s for %s' % (pre_test_plugin, repository)) - ret = pre_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, cursor=cursor) + ret = self.call_plugin(pre_test_plugin, repository, cursor=cursor) if not ret: return False kwargs = ret.kwargs optimization_init = False try: if optimization: - if not self._test_optimize_init(opts=opts, test_name='sysbench', deploy_config=deploy_config, cluster_config=cluster_config): + if not self._test_optimize_init(test_name='sysbench', repository=repository): return False optimization_init = True - if not self._test_optimize_operation(deploy=deploy, stage='test', opts=opts, connect_context=connect_context, optimize_envs=kwargs): + if not self._test_optimize_operation(repository=repository, ob_repository=ob_repository, stage='test', connect_namespaces=connect_namespaces, connect_plugin=connect_plugin, optimize_envs=kwargs): return False - self._call_stdio('verbose', 'Call %s for %s' % (run_test_plugin, repository)) - if run_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio): + if self.call_plugin(run_test_plugin, repository): return True - return False finally: if optimization and optimization_init: - self._test_optimize_operation(deploy=deploy, connect_context=connect_context, optimize_envs=kwargs, operation='recover') + self._test_optimize_operation(repository=repository, ob_repository=ob_repository, connect_namespaces=connect_namespaces, connect_plugin=connect_plugin, optimize_envs=kwargs, operation='recover') def tpch(self, name, opts): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -3288,7 +3397,7 @@ def tpch(self, name, opts): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.get_local_repositories({opts.component: deploy_config.components[opts.component]}) - repository = repositories[0] + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -3300,25 +3409,25 @@ def tpch(self, name, opts): if not getattr(opts, 'skip_cluster_status_check', False): # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) for repository in component_status: cluster_status = component_status[repository] for server in cluster_status: if cluster_status[server] == 0: self._call_stdio('print', '%s %s is stopped' % (server, repository.name)) return False - - connect_context = {} - connect_kwargs = dict(component_name=repository.name, target_server=opts.test_server) - ret = self._get_connect(deploy=deploy, **connect_kwargs) + repository = repositories[0] + namespace = self.get_namespace(repository.name) + namespace.set_variable('target_server', opts.test_server) + connect_plugin = self.plugin_manager.get_best_py_script_plugin('connect', repository.name, repository.version) + ret = self.call_plugin(connect_plugin, repository) if not ret or not ret.get_return('connect'): return False db = ret.get_return('connect') cursor = ret.get_return('cursor') - connect_context[repository.name] = {'connect_kwargs': connect_kwargs, 'db': db, 'cursor': cursor} pre_test_plugin = self.plugin_manager.get_best_py_script_plugin('pre_test', 'tpch', repository.version) run_test_plugin = self.plugin_manager.get_best_py_script_plugin('run_test', 'tpch', repository.version) @@ -3328,21 +3437,21 @@ def tpch(self, name, opts): optimization = getattr(opts, 'optimization', 0) - self._call_stdio('verbose', 'Call %s for %s' % (pre_test_plugin, repository)) - ret = pre_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, cursor=cursor) + ret = self.call_plugin(pre_test_plugin,repository, cursor=cursor) if not ret: return False kwargs = ret.kwargs optimization_init = False try: if optimization: - if not self._test_optimize_init(opts=opts, test_name='tpch', deploy_config=deploy_config, cluster_config=cluster_config): + if not self._test_optimize_init(test_name='tpch', repository=repository): return False optimization_init = True - if not self._test_optimize_operation(deploy=deploy, stage='test', opts=opts, connect_context=connect_context, optimize_envs=kwargs): + if not self._test_optimize_operation( + repository=repository, ob_repository=repository, stage='test', + connect_namespaces=[namespace], connect_plugin=connect_plugin, optimize_envs=kwargs): return False - self._call_stdio('verbose', 'Call %s for %s' % (run_test_plugin, repository)) - if run_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, db, cursor, **kwargs): + if self.call_plugin(run_test_plugin, repository, db=db, cursor=cursor, **kwargs): return True return False except Exception as e: @@ -3350,7 +3459,9 @@ def tpch(self, name, opts): return False finally: if optimization and optimization_init: - self._test_optimize_operation(deploy=deploy, connect_context=connect_context, optimize_envs=kwargs, operation='recover') + self._test_optimize_operation( + repository=repository, ob_repository=repository, connect_namespaces=[namespace], + connect_plugin=connect_plugin, optimize_envs=kwargs, operation='recover') def update_obd(self, version, install_prefix='/'): self._obd_update_lock() @@ -3375,6 +3486,7 @@ def update_obd(self, version, install_prefix='/'): def tpcds(self, name, opts): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -3413,6 +3525,7 @@ def tpcds(self, name, opts): # Get the repository # repositories = self.get_local_repositories({opts.component: deploy_config.components[opts.component]}) repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -3423,10 +3536,10 @@ def tpcds(self, name, opts): # Check the status for the deployed cluster component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) for repository in component_status: cluster_status = component_status[repository] for server in cluster_status: @@ -3451,19 +3564,21 @@ def tpcds(self, name, opts): check_opt_plugin = self.plugin_manager.get_best_py_script_plugin('check_opt', 'tpcds', db_cluster_config.version) load_data_plugin = self.plugin_manager.get_best_py_script_plugin('load_data', 'tpcds', cluster_config.version) run_test_plugin = self.plugin_manager.get_best_py_script_plugin('run_test', 'tpcds', cluster_config.version) + repository = None + for tmp_repository in repositories: + if tmp_repository.name == opts.component: + repository = tmp_repository - self._call_stdio('verbose', 'Call %s for %s' % (check_opt_plugin, cluster_config.name)) - if not check_opt_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, db_cluster_config=db_cluster_config): + if not self.call_plugin(check_opt_plugin, repository, db_cluster_config=db_cluster_config): return False - self._call_stdio('verbose', 'Call %s for %s' % (load_data_plugin, db_cluster_config.name)) - if not load_data_plugin(deploy_config.components.keys(), ssh_clients, db_cluster_config, [], opts, self.stdio): + if not self.call_plugin(load_data_plugin, repository): return False - self._call_stdio('verbose', 'Call %s for %s' % (run_test_plugin, cluster_config.name)) - return run_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio) + return self.call_plugin(run_test_plugin) def tpcc(self, name, opts): self._call_stdio('verbose', 'Get Deploy by name') deploy = self.deploy_manager.get_deploy_config(name) + self.set_deploy(deploy) if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False @@ -3507,6 +3622,7 @@ def tpcc(self, name, opts): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) @@ -3518,10 +3634,10 @@ def tpcc(self, name, opts): # Check the status for the deployed cluster if not getattr(opts, 'skip_cluster_status_check', False): component_status = {} - cluster_status = self.cluster_status_check(ssh_clients, deploy_config, repositories, component_status) + cluster_status = self.cluster_status_check(repositories, component_status) if cluster_status is False or cluster_status == 0: if self.stdio: - self._call_stdio('error', EC_SOME_SERVER_STOPED) + self._call_stdio('error', err.EC_SOME_SERVER_STOPED.format()) for repository in component_status: cluster_status = component_status[repository] for server in cluster_status: @@ -3531,44 +3647,42 @@ def tpcc(self, name, opts): ob_repository = None repository = None - env = {} odp_cursor = None - ob_component = None - odp_component = None - connect_context = {} + proxysys_namespace = None + connect_namespaces = [] for tmp_repository in repositories: if tmp_repository.name in ["oceanbase", "oceanbase-ce"]: ob_repository = tmp_repository - ob_component = tmp_repository.name if tmp_repository.name == opts.component: repository = tmp_repository - if tmp_repository.name in ['obproxy', 'obproxy-ce']: - odp_component = tmp_repository.name - allow_components = ['oceanbase', 'oceanbase-ce'] - for component in deploy_info.components: - if component in allow_components: - config = deploy_config.components[component] - env['user'] = 'root' - env['password'] = config.get_global_conf().get('root_password', '') - env['target_server'] = opts.test_server - break - connect_kwargs = dict(component_name=odp_component, target_server=opts.test_server) - ret = self._get_connect(deploy, **connect_kwargs) - if not ret or not ret.get_return('connect'): - return False - odp_db, odp_cursor = self._get_first_db_and_cursor_from_connect(ret) - connect_context[odp_component] = {'connect_kwargs': connect_kwargs, 'db': odp_db, 'cursor': odp_cursor} if not ob_repository: self._call_stdio('error', 'Deploy {} must contain the component oceanbase or oceanbase-ce.'.format(deploy.name)) return False + sys_namespace = self.get_namespace(ob_repository.name) + connect_plugin = self.plugin_manager.get_best_py_script_plugin('connect', repository.name, repository.version) + if repository.name in ['obproxy', 'obproxy-ce']: + for component_name in deploy_config.components: + if component_name in ['oceanbase', 'oceanbase-ce']: + ob_cluster_config = deploy_config.components[component_name] + sys_namespace.set_variable("connect_proxysys", False) + sys_namespace.set_variable("user", "root") + sys_namespace.set_variable("password", ob_cluster_config.get_global_conf().get('root_password', '')) + sys_namespace.set_variable("target_server", opts.test_server) + break + proxysys_namespace = self.get_namespace(repository.name) + proxysys_namespace.set_variable("component_name", repository) + proxysys_namespace.set_variable("target_server", opts.test_server) + ret = self.call_plugin(connect_plugin, repository, spacename=proxysys_namespace.spacename) + if not ret or not ret.get_return('connect'): + return False + odp_db, odp_cursor = self._get_first_db_and_cursor_from_connect(proxysys_namespace) + connect_namespaces.append(proxysys_namespace) plugin_version = ob_repository.version if ob_repository else repository.version - connect_kwargs = dict(component_name=repository.name, **env) - ret = self._get_connect(deploy=deploy, **connect_kwargs) + ret = self.call_plugin(connect_plugin, repository, spacename=sys_namespace.spacename) if not ret or not ret.get_return('connect'): return False - db, cursor = self._get_first_db_and_cursor_from_connect(ret) - connect_context[ob_component] = {'connect_kwargs': connect_kwargs, 'db': db, 'cursor': cursor} - + connect_namespaces.append(sys_namespace) + db, cursor = self._get_first_db_and_cursor_from_connect(namespace=sys_namespace) pre_test_plugin = self.plugin_manager.get_best_py_script_plugin('pre_test', 'tpcc', plugin_version) build_plugin = self.plugin_manager.get_best_py_script_plugin('build', 'tpcc', plugin_version) run_test_plugin = self.plugin_manager.get_best_py_script_plugin('run_test', 'tpcc', plugin_version) @@ -3582,37 +3696,34 @@ def tpcc(self, name, opts): test_only = getattr(opts, 'test_only', False) optimization_inited = False try: - self._call_stdio('verbose', 'Call %s for %s' % (pre_test_plugin, repository)) - ret = pre_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, - cursor=cursor, odp_cursor=odp_cursor, **kwargs) + ret = self.call_plugin(pre_test_plugin, repository, cursor=cursor, odp_cursor=odp_cursor, **kwargs) if not ret: return False else: kwargs.update(ret.kwargs) if optimization: - if not self._test_optimize_init(opts=opts, test_name='tpcc', deploy_config=deploy_config, cluster_config=cluster_config): + if not self._test_optimize_init(test_name='tpcc', repository=repository): return False optimization_inited = True - if not self._test_optimize_operation(deploy=deploy, stage='build', opts=opts, connect_context=connect_context, optimize_envs=kwargs): + if not self._test_optimize_operation(repository=repository, ob_repository=ob_repository, stage='build', + connect_namespaces=connect_namespaces, + connect_plugin=connect_plugin, optimize_envs=kwargs): return False if not test_only: - self._call_stdio('verbose', 'Call %s for %s' % (build_plugin, repository)) - cursor = connect_context[ob_component]['cursor'] - if odp_component: - odp_cursor = connect_context[odp_component]['cursor'] - ret = build_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, cursor, - odp_cursor, **kwargs) + db, cursor = self._get_first_db_and_cursor_from_connect(sys_namespace) + odp_db, odp_cursor = self._get_first_db_and_cursor_from_connect(proxysys_namespace) + ret = self.call_plugin(build_plugin, repository, cursor=cursor, odp_cursor=odp_cursor, **kwargs) if not ret: return False else: kwargs.update(ret.kwargs) if optimization: - ret = self._test_optimize_operation(deploy=deploy, stage='test', opts=opts, connect_context=connect_context, optimize_envs=kwargs) - if not ret: + if not self._test_optimize_operation(repository=repository, ob_repository=ob_repository, stage='test', + connect_namespaces=connect_namespaces, + connect_plugin=connect_plugin, optimize_envs=kwargs): return False - self._call_stdio('verbose', 'Call %s for %s' % (run_test_plugin, repository)) - cursor = connect_context[ob_component]['cursor'] - ret = run_test_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, cursor, **kwargs) + db, cursor = self._get_first_db_and_cursor_from_connect(sys_namespace) + ret = self.call_plugin(run_test_plugin, repository, cursor=cursor, **kwargs) if not ret: return False else: @@ -3623,7 +3734,9 @@ def tpcc(self, name, opts): return False finally: if optimization and optimization_inited: - self._test_optimize_operation(deploy=deploy, connect_context=connect_context, optimize_envs=kwargs, operation='recover') + self._test_optimize_operation(repository=repository, ob_repository=ob_repository, + connect_namespaces=connect_namespaces, + connect_plugin=connect_plugin, optimize_envs=kwargs, operation='recover') def db_connect(self, name, opts): self._call_stdio('verbose', 'Get Deploy by name') @@ -3631,7 +3744,7 @@ def db_connect(self, name, opts): if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False - + self.set_deploy(deploy) self._call_stdio('verbose', 'Get deploy configuration') deploy_config = deploy.deploy_config deploy_info = deploy.deploy_info @@ -3669,16 +3782,21 @@ def db_connect(self, name, opts): return False self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository - repositories = self.get_local_repositories({opts.component: deploy_config.components[opts.component]}) + repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) + repository = None + for tmp_repository in repositories: + if tmp_repository.name == opts.component: + repository = tmp_repository # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) self._call_stdio('stop_loading', 'succeed') sync_config_plugin = self.plugin_manager.get_best_py_script_plugin('sync_cluster_config', 'general', '0.1') - sync_config_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio) + self.call_plugin(sync_config_plugin, repository) db_connect_plugin = self.plugin_manager.get_best_py_script_plugin('db_connect', 'general', '0.1') - return db_connect_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio) + return self.call_plugin(db_connect_plugin, repository) def commands(self, name, cmd_name, opts): self._call_stdio('verbose', 'Get Deploy by name') @@ -3686,6 +3804,7 @@ def commands(self, name, cmd_name, opts): if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False + self.set_deploy(deploy) self._call_stdio('verbose', 'Get deploy configuration') deploy_config = deploy.deploy_config deploy_info = deploy.deploy_info @@ -3697,6 +3816,8 @@ def commands(self, name, cmd_name, opts): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + repositories = self.sort_repositories_by_depends(deploy_config, repositories) + self.set_repositories(repositories) # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) self._call_stdio('stop_loading', 'succeed') @@ -3704,23 +3825,24 @@ def commands(self, name, cmd_name, opts): check_opt_plugin = self.plugin_manager.get_best_py_script_plugin('check_opt', 'commands', '0.1') prepare_variables_plugin = self.plugin_manager.get_best_py_script_plugin('prepare_variables', 'commands', '0.1') commands_plugin = self.plugin_manager.get_best_py_script_plugin('commands', 'commands', '0.1') - ssh_clients = self.get_clients(deploy_config, repositories) sync_config_plugin = self.plugin_manager.get_best_py_script_plugin('sync_cluster_config', 'general', '0.1') - cluster_config = deploy_config.components[repositories[0].name] + repository = repositories[0] context = {} - sync_config_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio) - ret = check_opt_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, name=cmd_name, context=context) + self.call_plugin(sync_config_plugin, repository) + ret = self.call_plugin(check_opt_plugin, repository, name=cmd_name, context=context) if not ret: return for component in context['components']: - cluster_config = deploy_config.components[component] + for repository in repositories: + if repository.name == component: + break for server in context['servers']: - ret = prepare_variables_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, name=cmd_name, component=component, server=server, context=context) + ret = self.call_plugin(prepare_variables_plugin, repository, name=cmd_name, component=component, server=server, context=context) if not ret: return if not ret.get_return("skip"): - ret = commands_plugin(deploy_config.components.keys(), ssh_clients, cluster_config, [], opts, self.stdio, context=context) + ret = self.call_plugin(commands_plugin, repository, context=context) if context.get('interactive'): return bool(ret) results = context.get('results', []) @@ -3733,7 +3855,7 @@ def dooba(self, name, opts): if not deploy: self._call_stdio('error', 'No such deploy: %s.' % name) return False - + self.set_deploy(deploy) self._call_stdio('verbose', 'Get deploy configuration') deploy_config = deploy.deploy_config deploy_info = deploy.deploy_info @@ -3780,16 +3902,19 @@ def dooba(self, name, opts): self._call_stdio('start_loading', 'Get local repositories and plugins') # Get the repository repositories = self.load_local_repositories(deploy_info) + self.set_repositories(repositories) plugin_version = None + target_repository = None for repository in repositories: if repository.name in ['oceanbase', 'oceanbase-ce']: plugin_version = repository.version - break + if repository.name == opts.component: + target_repository = repository # Check whether the components have the parameter plugins and apply the plugins self.search_param_plugin_and_apply(repositories, deploy_config) self._call_stdio('stop_loading', 'succeed') sync_config_plugin = self.plugin_manager.get_best_py_script_plugin('sync_cluster_config', 'general', '0.1') - sync_config_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio) + self.call_plugin(sync_config_plugin, target_repository) dooba_plugin = self.plugin_manager.get_best_py_script_plugin('run', 'dooba', plugin_version) - return dooba_plugin(deploy_config.components.keys(), [], cluster_config, [], opts, self.stdio) \ No newline at end of file + return self.call_plugin(dooba_plugin, target_repository) diff --git a/example/all-components-min.yaml b/example/all-components-min.yaml index 19c5349..9d8e69f 100644 --- a/example/all-components-min.yaml +++ b/example/all-components-min.yaml @@ -24,14 +24,22 @@ oceanbase-ce: memory_limit: 6G # The maximum running memory for an observer system_memory: 1G # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. datafile_size: 20G # Size of the data file. - log_disk_size: 24G # The size of disk space used by the clog files. + log_disk_size: 15G # The size of disk space used by the clog files. cpu_count: 16 production_mode: false - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. # root_password: # root user password, can be empty + # ocp_meta_db: ocp_express # The database name of ocp express meta + # ocp_meta_username: meta # The username of ocp express meta + # ocp_meta_password: '' # The password of ocp express meta + # ocp_agent_monitor_password: '' # The password for obagent monitor user + ocp_meta_tenant: # The config for ocp express meta tenant + tenant_name: ocp + max_cpu: 1 + memory_size: 2G + log_disk_size: 7680M # The recommend value is (4608 + (expect node num + expect tenant num) * 512) M. server1: mysql_port: 2881 # External port for OceanBase Database. The default value is 2881. DO NOT change this value after the cluster is started. rpc_port: 2882 # Internal port for OceanBase Database. The default value is 2882. DO NOT change this value after the cluster is started. @@ -68,7 +76,7 @@ obproxy-ce: depends: - oceanbase-ce servers: - - 192.168.1.5 + - 172.19.33.6 global: listen_port: 2883 # External port. The default value is 2883. prometheus_listen_port: 2884 # The Prometheus port. The default value is 2884. @@ -96,19 +104,17 @@ obagent: ip: 172.19.33.4 global: home_path: /root/obagent - ob_monitor_status: active -prometheus: - depends: +ocp-express: + depeneds: + - oceanbase-ce + - obproxy-ce - obagent servers: - - 192.168.1.5 - global: - home_path: /root/prometheus -grafana: - depends: - - prometheus - servers: - - 192.168.1.5 + - 172.19.33.5 global: - home_path: /root/grafana - login_password: oceanbase \ No newline at end of file + # The working directory for prometheus. prometheus is started under this directory. This is a required field. + home_path: /root/ocp-server + # log_dir: /home/oceanbase/ocp-server/log # The log directory of ocp express server. The default value is {home_path}/log. + memory_size: 1G # The memory size of ocp-express server. The recommend value is 512MB * (expect node num + expect tenant num) * 60MB. + # logging_file_total_size_cap: 10G # The total log file size of ocp-express server + # logging_file_max_history: 1 # The maximum of retention days the log archive log files to keep. The default value is unlimited \ No newline at end of file diff --git a/example/all-components.yaml b/example/all-components.yaml index 17806bd..a993ed5 100644 --- a/example/all-components.yaml +++ b/example/all-components.yaml @@ -25,13 +25,21 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. skip_proxy_sys_private_check: true enable_strict_kernel_release: false # root_password: # root user password + # ocp_meta_db: ocp_express # The database name of ocp express meta + # ocp_meta_username: meta # The username of ocp express meta + # ocp_meta_password: '' # The password of ocp express meta + # ocp_agent_monitor_password: '' # The password for obagent monitor user + ocp_meta_tenant: # The config for ocp express meta tenant + tenant_name: ocp + max_cpu: 1 + memory_size: 2G + log_disk_size: 7680M # The recommend value is (4608 + (expect node num + expect tenant num) * 512) M. # In this example , support multiple ob process in single node, so different process use different ports. # If deploy ob cluster in multiple nodes, the port and path setting can be same. server1: @@ -70,7 +78,7 @@ obproxy-ce: depends: - oceanbase-ce servers: - - 192.168.1.5 + - 172.19.33.6 global: listen_port: 2883 # External port. The default value is 2883. prometheus_listen_port: 2884 # The Prometheus port. The default value is 2884. @@ -98,19 +106,17 @@ obagent: ip: 172.19.33.4 global: home_path: /root/obagent - ob_monitor_status: active -prometheus: - depends: +ocp-express: + depeneds: + - oceanbase-ce + - obproxy-ce - obagent servers: - - 192.168.1.5 - global: - home_path: /root/prometheus -grafana: - depends: - - prometheus - servers: - - 192.168.1.5 + - 172.19.33.5 global: - home_path: /root/grafana - login_password: oceanbase \ No newline at end of file + # The working directory for prometheus. prometheus is started under this directory. This is a required field. + home_path: /root/ocp-server + # log_dir: /home/oceanbase/ocp-server/log # The log directory of ocp express server. The default value is {home_path}/log. + memory_size: 1G # The memory size of ocp-express server. The recommend value is 512MB * (expect node num + expect tenant num) * 60MB. + # logging_file_total_size_cap: 10G # The total log file size of ocp-express server + # logging_file_max_history: 1 # The maximum of retention days the log archive log files to keep. The default value is unlimited \ No newline at end of file diff --git a/example/autodeploy/all-components.yaml b/example/autodeploy/all-components.yaml index 270cf60..d7ad5e0 100644 --- a/example/autodeploy/all-components.yaml +++ b/example/autodeploy/all-components.yaml @@ -41,8 +41,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. @@ -61,35 +61,15 @@ oceanbase-ce: zone: zone2 server3: zone: zone3 -obproxy-ce: - # Set dependent components for the component. - # When the associated configurations are not done, OBD will automatically get the these configurations from the dependent components. - depends: - - oceanbase-ce - servers: - - 192.168.1.5 - global: - listen_port: 2883 # External port. The default value is 2883. - prometheus_listen_port: 2884 # The Prometheus port. The default value is 2884. - home_path: /root/obproxy - # oceanbase root server list - # format: ip:mysql_port;ip:mysql_port. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. - # rs_list: 192.168.1.2:2881;192.168.1.3:2881;192.168.1.4:2881 - enable_cluster_checkout: false - # observer cluster name, consistent with oceanbase-ce's appname. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. - # cluster_name: obcluster - skip_proxy_sys_private_check: true - enable_strict_kernel_release: false - # obproxy_sys_password: # obproxy sys user password, can be empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. - # observer_sys_password: # proxyro user pasword, consistent with oceanbase-ce's proxyro_password, can be empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. obproxy-ce: depends: - oceanbase-ce servers: - - 192.168.1.5 + - 172.19.33.5 global: # The working directory for obproxy. Obproxy is started under this directory. This is a required field. home_path: /root/obproxy + enable_cluster_checkout: false skip_proxy_sys_private_check: true enable_strict_kernel_release: false # External port. The default value is 2883. @@ -123,40 +103,34 @@ obagent: global: # The working directory for obagent. obagent is started under this directory. This is a required field. home_path: /root/obagent - # The port that pulls and manages the metrics. The default port number is 8088. - # server_port: 8088 - # Debug port for pprof. The default port number is 8089. - # pprof_port: 8089 - # Log level. The default value is INFO. - # log_level: INFO + # The port of monitor agent. The default port number is 8088. + # monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + # mgragent_http_port: 8089 # Log path. The default value is log/monagent.log. # log_path: log/monagent.log - # Encryption method. OBD supports aes and plain. The default value is plain. - # crypto_method: plain - # Path to store the crypto key. The default value is conf/.config_secret.key. - # crypto_path: conf/.config_secret.key - # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. - # log_size: 30 - # Expiration time for logs. The default value is 7 days. - # log_expire_day: 7 - # The maximum number for log files. The default value is 10. - # log_file_count: 10 - # Whether to use local time for log files. The default value is true. - # log_use_localtime: true - # Whether to enable log compression. The default value is true. - # log_compress: true + # The log level of manager agent. + # mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + # mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + # mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + # mgragent_log_max_backups: 15 + # The log level of monitor agent. + # monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + # monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + # monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + # monagent_log_max_backups: 15 # Username for HTTP authentication. The default value is admin. # http_basic_auth_user: admin # Password for HTTP authentication. The default value is root. # http_basic_auth_password: root - # Username for debug service. The default value is admin. - # pprof_basic_auth_user: admin - # Password for debug service. The default value is root. - # pprof_basic_auth_password: root - # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. - # monitor_user: root - # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. - # monitor_password: + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. + # monitor_password: # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. # sql_port: 2881 # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. @@ -165,136 +139,40 @@ obagent: # cluster_name: obcluster # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. # cluster_id: 1 - # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. - # zone_name: zone1 + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + # ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + # ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + # ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + # observer_log_path: /root/observer/log # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. # ob_monitor_status: active - # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. - # host_monitor_status: active - # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. - # disable_http_basic_auth: false - # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. - # disable_pprof_basic_auth: false -prometheus: +ocp-server: servers: - - 192.168.1.5 - depends: - - obagent - global: - # The working directory for prometheus. prometheus is started under this directory. This is a required field. - home_path: /root/prometheus - # address: 0.0.0.0 # The ip address to bind to. Along with port, corresponds to the `web.listen-address` parameter. - # port: 9090 # The http port to use. Along with address, corresponds to the `web.listen-address` parameter. - # enable_lifecycle: true # Enable shutdown and reload via HTTP request. Corresponds to the `web.enable-lifecycle` parameter. - # data_dir: /root/prometheus/data # Base path for metrics storage. Corresponds to the `storage.tsdb.path` parameter. - # basic_auth_users: # Usernames and passwords that have full access to the web server via basic authentication. Corresponds to the `basic_auth_users` parameter. - # : # The format of `basic_auth_users` : the key is the user name and the value is the password. - # web_config: # Content of Prometheus web service config file. The format is consistent with the file. However, `basic_auth_users` cannot be set in it. Please set `basic_auth_users` above if needed. Corresponds to the `web.config.file` parameter. - # tls_server_config: - # # Certificate and key files for server to use to authenticate to client. - # cert_file: - # key_file: - # config: # Configuration of the Prometheus service. The format is consistent with the Prometheus config file. Corresponds to the `config.file` parameter. - # rule_files: - # - rules/*rules.yaml - # scrape_configs: - # - job_name: prometheus - # metrics_path: /metrics - # scheme: http - # static_configs: - # - targets: - # - localhost:9090 - # - job_name: node - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/node/host - # scheme: http - # file_sd_configs: # Set the targets to be collected by reading local files. The example is to collect targets corresponding to all yaml files in the 'targets' directory under $home_path. - # - files: - # - 'targets/*.yaml' - # - job_name: ob_basic - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/ob/basic - # scheme: http - # file_sd_configs: - # - files: - # - 'targets/*.yaml' - # - job_name: ob_extra - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/ob/extra - # scheme: http - # file_sd_configs: - # - files: - # - 'targets/*.yaml' - # - job_name: agent - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/stat - # scheme: http - # file_sd_configs: - # - files: - # - 'targets/*.yaml' - # additional_parameters: # Additional parameters for Prometheus service, among which `web.listen-address`, `web.enable-lifecycle`, `storage.tsdb.path`, `config.file` and `web.config.file` cannot be set. Please set them in the corresponding configuration above if needed. - # - log.level: debug -grafana: - servers: - - 192.168.1.5 - depends: - - prometheus + - name: server1 + ip: 192.168.1.1 global: - home_path: /root/grafana - login_password: oceanbase # Grafana login password. The default value is 'oceanbase'. - # data_dir: # Path to where grafana can store temp files, sessions, and the sqlite3 db (if that is used).$data_dir can be empty. The default value is $home_path/data. - # logs_dir: # Directory where grafana can store logs, can be empty. The default value is $data_dir/log. - # plugins_dir: # Directory where grafana will automatically scan and look for plugins, can be empty. The default value is $data_dir/plugins. - # provisioning_dir: # folder that contains provisioning config files that grafana will apply on startup and while running, can be empty. The default value is $home_path/conf/provisioning. - # temp_data_lifetime: # How long temporary images in data directory should be kept. Supported modifiers h (hours), m (minutes), Use 0 to never clean up temporary files, can be empty. The default value is 24h. - # log_max_days: # Expired days of log file(delete after max days), can be empty. The default value is 7. - # domian: # The public facing domain name used to access grafana from a browser, can be empty. The default value is $server.ip. - # port: # The http port to use, can be empty. The default value is 3000. - - # # list of datasources to insert/update depending on what's available in the database, can be empty. - # # For more parameter settings, please refer to https://grafana.com/docs/grafana/latest/administration/provisioning/#datasources - # datasources: - # name: # name of the datasource. Required and should not be 'OB-Prometheus' - # type: # datasource type. Required - # access: # access mode. direct or proxy. Required - # url: # the url of datasource - - # list of dashboards providers that load dashboards into Grafana from the local filesystem, can be empty. - # For more information, please refer to https://grafana.com/docs/grafana/latest/administration/provisioning/#dashboards - # providers: - # name: # an unique provider name. Required and should not be 'OceanBase Metrics' - # type: # provider type. Default to 'file' - # options: - # path: # path to dashboard files on disk. Required when using the 'file' type - - # # customize your Grafana instance by adding/modifying the custom configuration as follows - # # for more information, please refer to https://grafana.com/docs/grafana/latest/setup-grafana/configure-grafana/#configure-grafana - # # Here, setting parameters is required for format conversion. - # # For example, if the original grafana configuration format is - # # - # # [section1.section2] - # # key1 = value1 - # # key2 = value2 - # # - # # Then when writing the configuration below, you need to write it as - # # - # # section1: - # # section2: - # # key1: value1 - # # key2: value2 - # # - # # Here we only list one item, because there are more than 500 items. Please add them according to your own needs. - # customize_config: - # # original grafana configuration format is - # # [server] - # # protocol = http - # server: - # protocol: http + # The working directory for ocp express. ocp express is started under this directory. This is a required field. + home_path: /root/ocp-server + # log_dir: /root/ocp-server/log # The log directory of ocp express server. The default value is {home_path}/log. + # memory_size: 1G # The memory size of ocp-express server. The recommend value is 512MB * (expect node num + expect tenant num) * 60MB. + # jdbc_url: jdbc:oceanbase://192.168.1.1:2881/meta_db # jdbc connection string to connect to the meta db + # jdbc_username: username # username to connect to meta db + # jdbc_password: '' # password to connect to meta db + # port: 8080 # The http port to use. + # cluster_name: obcluster # the cluster name of oceanbase cluster. Refer to the configuration item appname of oceanbase + # ob_cluster_id: 1 # the cluster id of oceanbase cluster. Refer to the configuration item cluster_id of oceanbase + # root_sys_password: # the pass of oceanbase cluster. Refer to the configuration item cluster_id of oceanbase + # agent_username: # The username of obagent + # agent_password: # The password of obagent + # # logging_file_total_size_cap: 10G # The total log file size of ocp-express server + # # logging_file_max_history: 1 # The maximum of retention days the log archive log files to keep. The default value is unlimited + # server_addresses: # The cluster info for oceanbase cluster + # - address: 127.0.0.1 # The address of oceanbase server + # svrPort: 2882 # The rpc port of oceanbase server + # sqlPort: 2881 # The mysql port of oceanbase server + # withRootServer: true # Is the oceanbase server a root server of cluster. + # agentMgrPort: 62888 # The port of obagent manager process + # agentMonPort: 62889 # The port of obagent monitor process diff --git a/example/autodeploy/default-example.yaml b/example/autodeploy/default-example.yaml index fe92596..d7ad5e0 100644 --- a/example/autodeploy/default-example.yaml +++ b/example/autodeploy/default-example.yaml @@ -41,8 +41,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. @@ -103,40 +103,34 @@ obagent: global: # The working directory for obagent. obagent is started under this directory. This is a required field. home_path: /root/obagent - # The port that pulls and manages the metrics. The default port number is 8088. - # server_port: 8088 - # Debug port for pprof. The default port number is 8089. - # pprof_port: 8089 - # Log level. The default value is INFO. - # log_level: INFO + # The port of monitor agent. The default port number is 8088. + # monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + # mgragent_http_port: 8089 # Log path. The default value is log/monagent.log. # log_path: log/monagent.log - # Encryption method. OBD supports aes and plain. The default value is plain. - # crypto_method: plain - # Path to store the crypto key. The default value is conf/.config_secret.key. - # crypto_path: conf/.config_secret.key - # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. - # log_size: 30 - # Expiration time for logs. The default value is 7 days. - # log_expire_day: 7 - # The maximum number for log files. The default value is 10. - # log_file_count: 10 - # Whether to use local time for log files. The default value is true. - # log_use_localtime: true - # Whether to enable log compression. The default value is true. - # log_compress: true + # The log level of manager agent. + # mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + # mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + # mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + # mgragent_log_max_backups: 15 + # The log level of monitor agent. + # monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + # monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + # monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + # monagent_log_max_backups: 15 # Username for HTTP authentication. The default value is admin. # http_basic_auth_user: admin # Password for HTTP authentication. The default value is root. # http_basic_auth_password: root - # Username for debug service. The default value is admin. - # pprof_basic_auth_user: admin - # Password for debug service. The default value is root. - # pprof_basic_auth_password: root - # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. - # monitor_user: root - # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. - # monitor_password: + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. + # monitor_password: # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. # sql_port: 2881 # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. @@ -145,136 +139,40 @@ obagent: # cluster_name: obcluster # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. # cluster_id: 1 - # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. - # zone_name: zone1 + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + # ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + # ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + # ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + # observer_log_path: /root/observer/log # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. # ob_monitor_status: active - # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. - # host_monitor_status: active - # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. - # disable_http_basic_auth: false - # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. - # disable_pprof_basic_auth: false -prometheus: +ocp-server: servers: - - 172.19.33.5 - depends: - - obagent - global: - # The working directory for prometheus. prometheus is started under this directory. This is a required field. - home_path: /root/prometheus - # address: 0.0.0.0 # The ip address to bind to. Along with port, corresponds to the `web.listen-address` parameter. - # port: 9090 # The http port to use. Along with address, corresponds to the `web.listen-address` parameter. - # enable_lifecycle: true # Enable shutdown and reload via HTTP request. Corresponds to the `web.enable-lifecycle` parameter. - # data_dir: /root/prometheus/data # Base path for metrics storage. Corresponds to the `storage.tsdb.path` parameter. - # basic_auth_users: # Usernames and passwords that have full access to the web server via basic authentication. Corresponds to the `basic_auth_users` parameter. - # : # The format of `basic_auth_users` : the key is the user name and the value is the password. - # web_config: # Content of Prometheus web service config file. The format is consistent with the file. However, `basic_auth_users` cannot be set in it. Please set `basic_auth_users` above if needed. Corresponds to the `web.config.file` parameter. - # tls_server_config: - # # Certificate and key files for server to use to authenticate to client. - # cert_file: - # key_file: - # config: # Configuration of the Prometheus service. The format is consistent with the Prometheus config file. Corresponds to the `config.file` parameter. - # rule_files: - # - rules/*rules.yaml - # scrape_configs: - # - job_name: prometheus - # metrics_path: /metrics - # scheme: http - # static_configs: - # - targets: - # - localhost:9090 - # - job_name: node - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/node/host - # scheme: http - # file_sd_configs: # Set the targets to be collected by reading local files. The example is to collect targets corresponding to all yaml files in the 'targets' directory under $home_path. - # - files: - # - 'targets/*.yaml' - # - job_name: ob_basic - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/ob/basic - # scheme: http - # file_sd_configs: - # - files: - # - 'targets/*.yaml' - # - job_name: ob_extra - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/ob/extra - # scheme: http - # file_sd_configs: - # - files: - # - 'targets/*.yaml' - # - job_name: agent - # basic_auth: - # username: admin - # password: root - # metrics_path: /metrics/stat - # scheme: http - # file_sd_configs: - # - files: - # - 'targets/*.yaml' - # additional_parameters: # Additional parameters for Prometheus service, among which `web.listen-address`, `web.enable-lifecycle`, `storage.tsdb.path`, `config.file` and `web.config.file` cannot be set. Please set them in the corresponding configuration above if needed. - # - log.level: debug -grafana: - servers: - - 172.19.33.5 - depends: - - prometheus + - name: server1 + ip: 192.168.1.1 global: - home_path: /root/grafana - login_password: oceanbase # Grafana login password. The default value is 'oceanbase'. - # data_dir: # Path to where grafana can store temp files, sessions, and the sqlite3 db (if that is used).$data_dir can be empty. The default value is $home_path/data. - # logs_dir: # Directory where grafana can store logs, can be empty. The default value is $data_dir/log. - # plugins_dir: # Directory where grafana will automatically scan and look for plugins, can be empty. The default value is $data_dir/plugins. - # provisioning_dir: # folder that contains provisioning config files that grafana will apply on startup and while running, can be empty. The default value is $home_path/conf/provisioning. - # temp_data_lifetime: # How long temporary images in data directory should be kept. Supported modifiers h (hours), m (minutes), Use 0 to never clean up temporary files, can be empty. The default value is 24h. - # log_max_days: # Expired days of log file(delete after max days), can be empty. The default value is 7. - # domian: # The public facing domain name used to access grafana from a browser, can be empty. The default value is $server.ip. - # port: # The http port to use, can be empty. The default value is 3000. - - # # list of datasources to insert/update depending on what's available in the database, can be empty. - # # For more parameter settings, please refer to https://grafana.com/docs/grafana/latest/administration/provisioning/#datasources - # datasources: - # name: # name of the datasource. Required and should not be 'OB-Prometheus' - # type: # datasource type. Required - # access: # access mode. direct or proxy. Required - # url: # the url of datasource - - # list of dashboards providers that load dashboards into Grafana from the local filesystem, can be empty. - # For more information, please refer to https://grafana.com/docs/grafana/latest/administration/provisioning/#dashboards - # providers: - # name: # an unique provider name. Required and should not be 'OceanBase Metrics' - # type: # provider type. Default to 'file' - # options: - # path: # path to dashboard files on disk. Required when using the 'file' type - - # # customize your Grafana instance by adding/modifying the custom configuration as follows - # # for more information, please refer to https://grafana.com/docs/grafana/latest/setup-grafana/configure-grafana/#configure-grafana - # # Here, setting parameters is required for format conversion. - # # For example, if the original grafana configuration format is - # # - # # [section1.section2] - # # key1 = value1 - # # key2 = value2 - # # - # # Then when writing the configuration below, you need to write it as - # # - # # section1: - # # section2: - # # key1: value1 - # # key2: value2 - # # - # # Here we only list one item, because there are more than 500 items. Please add them according to your own needs. - # customize_config: - # # original grafana configuration format is - # # [server] - # # protocol = http - # server: - # protocol: http + # The working directory for ocp express. ocp express is started under this directory. This is a required field. + home_path: /root/ocp-server + # log_dir: /root/ocp-server/log # The log directory of ocp express server. The default value is {home_path}/log. + # memory_size: 1G # The memory size of ocp-express server. The recommend value is 512MB * (expect node num + expect tenant num) * 60MB. + # jdbc_url: jdbc:oceanbase://192.168.1.1:2881/meta_db # jdbc connection string to connect to the meta db + # jdbc_username: username # username to connect to meta db + # jdbc_password: '' # password to connect to meta db + # port: 8080 # The http port to use. + # cluster_name: obcluster # the cluster name of oceanbase cluster. Refer to the configuration item appname of oceanbase + # ob_cluster_id: 1 # the cluster id of oceanbase cluster. Refer to the configuration item cluster_id of oceanbase + # root_sys_password: # the pass of oceanbase cluster. Refer to the configuration item cluster_id of oceanbase + # agent_username: # The username of obagent + # agent_password: # The password of obagent + # # logging_file_total_size_cap: 10G # The total log file size of ocp-express server + # # logging_file_max_history: 1 # The maximum of retention days the log archive log files to keep. The default value is unlimited + # server_addresses: # The cluster info for oceanbase cluster + # - address: 127.0.0.1 # The address of oceanbase server + # svrPort: 2882 # The rpc port of oceanbase server + # sqlPort: 2881 # The mysql port of oceanbase server + # withRootServer: true # Is the oceanbase server a root server of cluster. + # agentMgrPort: 62888 # The port of obagent manager process + # agentMonPort: 62889 # The port of obagent monitor process diff --git a/example/autodeploy/distributed-example.yaml b/example/autodeploy/distributed-example.yaml index 2eec916..3999dde 100644 --- a/example/autodeploy/distributed-example.yaml +++ b/example/autodeploy/distributed-example.yaml @@ -41,8 +41,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. diff --git a/example/autodeploy/distributed-with-obproxy-and-obagent-example.yaml b/example/autodeploy/distributed-with-obproxy-and-obagent-example.yaml index 37d0d53..f9691b4 100644 --- a/example/autodeploy/distributed-with-obproxy-and-obagent-example.yaml +++ b/example/autodeploy/distributed-with-obproxy-and-obagent-example.yaml @@ -41,8 +41,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. @@ -103,40 +103,34 @@ obagent: global: # The working directory for obagent. obagent is started under this directory. This is a required field. home_path: /root/obagent - # The port that pulls and manages the metrics. The default port number is 8088. - # server_port: 8088 - # Debug port for pprof. The default port number is 8089. - # pprof_port: 8089 - # Log level. The default value is INFO. - # log_level: INFO + # The port of monitor agent. The default port number is 8088. + # monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + # mgragent_http_port: 8089 # Log path. The default value is log/monagent.log. # log_path: log/monagent.log - # Encryption method. OBD supports aes and plain. The default value is plain. - # crypto_method: plain - # Path to store the crypto key. The default value is conf/.config_secret.key. - # crypto_path: conf/.config_secret.key - # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. - # log_size: 30 - # Expiration time for logs. The default value is 7 days. - # log_expire_day: 7 - # The maximum number for log files. The default value is 10. - # log_file_count: 10 - # Whether to use local time for log files. The default value is true. - # log_use_localtime: true - # Whether to enable log compression. The default value is true. - # log_compress: true + # The log level of manager agent. + # mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + # mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + # mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + # mgragent_log_max_backups: 15 + # The log level of monitor agent. + # monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + # monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + # monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + # monagent_log_max_backups: 15 # Username for HTTP authentication. The default value is admin. # http_basic_auth_user: admin # Password for HTTP authentication. The default value is root. # http_basic_auth_password: root - # Username for debug service. The default value is admin. - # pprof_basic_auth_user: admin - # Password for debug service. The default value is root. - # pprof_basic_auth_password: root - # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. - # monitor_user: root - # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. - # monitor_password: + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. + # monitor_password: # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. # sql_port: 2881 # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. @@ -145,13 +139,12 @@ obagent: # cluster_name: obcluster # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. # cluster_id: 1 - # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. - # zone_name: zone1 - # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. - # ob_monitor_status: active - # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. - # host_monitor_status: active - # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. - # disable_http_basic_auth: false - # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. - # disable_pprof_basic_auth: false + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + # ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + # ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + # ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + # observer_log_path: /root/observer/log + # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. \ No newline at end of file diff --git a/example/autodeploy/distributed-with-obproxy-example.yaml b/example/autodeploy/distributed-with-obproxy-example.yaml index 6b53be4..203e9d8 100644 --- a/example/autodeploy/distributed-with-obproxy-example.yaml +++ b/example/autodeploy/distributed-with-obproxy-example.yaml @@ -41,8 +41,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. diff --git a/example/autodeploy/single-example.yaml b/example/autodeploy/single-example.yaml index 1526ba8..7a94311 100644 --- a/example/autodeploy/single-example.yaml +++ b/example/autodeploy/single-example.yaml @@ -36,8 +36,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. diff --git a/example/autodeploy/single-with-obproxy-example.yaml b/example/autodeploy/single-with-obproxy-example.yaml index 8c0b846..185688d 100644 --- a/example/autodeploy/single-with-obproxy-example.yaml +++ b/example/autodeploy/single-with-obproxy-example.yaml @@ -36,8 +36,8 @@ oceanbase-ce: # datafile_size: 200G # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. # log_disk_size: 66G - # System log level. The default value is INFO. - # syslog_level: INFO + # System log level. The default value is WDIAG. + # syslog_level: WDIAG # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. # enable_syslog_wf: false # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. diff --git a/example/distributed-example.yaml b/example/distributed-example.yaml index ebc0528..b6acd56 100644 --- a/example/distributed-example.yaml +++ b/example/distributed-example.yaml @@ -25,7 +25,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/distributed-with-obproxy-example.yaml b/example/distributed-with-obproxy-example.yaml index f8d556f..447237a 100644 --- a/example/distributed-with-obproxy-example.yaml +++ b/example/distributed-with-obproxy-example.yaml @@ -25,7 +25,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/grafana/all-components-with-prometheus-and-grafana.yaml b/example/grafana/all-components-with-prometheus-and-grafana.yaml new file mode 100644 index 0000000..f48d675 --- /dev/null +++ b/example/grafana/all-components-with-prometheus-and-grafana.yaml @@ -0,0 +1,300 @@ +## Only need to configure when remote login is required +# user: +# username: your username +# password: your password if need +# key_file: your ssh-key file path if need +# port: your ssh port, default 22 +# timeout: ssh connection timeout (second), default 30 +oceanbase-ce: + servers: + - name: server1 + # Please don't use hostname, only IP can be supported + ip: 172.19.33.2 + - name: server2 + ip: 172.19.33.3 + - name: server3 + ip: 172.19.33.4 + global: + # The working directory for OceanBase Database. OceanBase Database is started under this directory. This is a required field. + home_path: /root/observer + # The directory for data storage. The default value is $home_path/store. + # data_dir: /data + # The directory for clog, ilog, and slog. The default value is the same as the data_dir value. + # redo_dir: /redo + # Please set devname as the network adaptor's name whose ip is in the setting of severs. + # if set severs as "127.0.0.1", please set devname as "lo" + # if current ip is 192.168.1.10, and the ip's network adaptor's name is "eth0", please use "eth0" + # devname: eth0 + # External port for OceanBase Database. The default value is 2881. DO NOT change this value after the cluster is started. + # mysql_port: 2881 + # Internal port for OceanBase Database. The default value is 2882. DO NOT change this value after the cluster is started. + # rpc_port: 2882 + # Defines the zone for an observer. The default value is zone1. + # zone: zone1 + # The maximum running memory for an observer. When ignored, autodeploy calculates this value based on the current server available resource. + # memory_limit: 58G + # The percentage of the maximum available memory to the total memory. This value takes effect only when memory_limit is 0. The default value is 80. + # memory_limit_percentage: 80 + # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. Autodeploy calculates this value based on the current server available resource. + # system_memory: 22G + # The size of a data file. When ignored, autodeploy calculates this value based on the current server available resource. + # datafile_size: 200G + # The size of disk space used by the clog files. When ignored, autodeploy calculates this value based on the current server available resource. + # log_disk_size: 66G + # System log level. The default value is WDIAG. + # syslog_level: WDIAG + # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. The default value for autodeploy mode is false. + # enable_syslog_wf: false + # Enable auto system log recycling or not. The default value is false. The default value for autodeploy mode is on. + # enable_syslog_recycle: true + # The maximum number of reserved log files before enabling auto recycling. When set to 0, no logs are deleted. The default value for autodeploy mode is 4. + # max_syslog_file_count: 4 + # Cluster name for OceanBase Database. The default value is obcluster. When you deploy OceanBase Database and obproxy, this value must be the same as the cluster_name for obproxy. + # appname: obcluster + # Password for root. The default value is empty. + # root_password: + # Password for proxyro. proxyro_password must be the same as observer_sys_password. The default value is empty. + # proxyro_password: + server1: + zone: zone1 + server2: + zone: zone2 + server3: + zone: zone3 +obproxy-ce: + # Set dependent components for the component. + # When the associated configurations are not done, OBD will automatically get the these configurations from the dependent components. + depends: + - oceanbase-ce + servers: + - 192.168.1.5 + global: + listen_port: 2883 # External port. The default value is 2883. + prometheus_listen_port: 2884 # The Prometheus port. The default value is 2884. + home_path: /root/obproxy + # oceanbase root server list + # format: ip:mysql_port;ip:mysql_port. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. + # rs_list: 192.168.1.2:2881;192.168.1.3:2881;192.168.1.4:2881 + enable_cluster_checkout: false + # observer cluster name, consistent with oceanbase-ce's appname. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. + # cluster_name: obcluster + skip_proxy_sys_private_check: true + enable_strict_kernel_release: false + # obproxy_sys_password: # obproxy sys user password, can be empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. + # observer_sys_password: # proxyro user pasword, consistent with oceanbase-ce's proxyro_password, can be empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. +obproxy-ce: + depends: + - oceanbase-ce + servers: + - 192.168.1.5 + global: + # The working directory for obproxy. Obproxy is started under this directory. This is a required field. + home_path: /root/obproxy + skip_proxy_sys_private_check: true + enable_strict_kernel_release: false + # External port. The default value is 2883. + # listen_port: 2883 + # The Prometheus port. The default value is 2884. + # prometheus_listen_port: 2884 + # rs_list is the root server list for observers. The default root server is the first server in the zone. + # The format for rs_list is observer_ip:observer_mysql_port;observer_ip:observer_mysql_port. + # Ignore this value in autodeploy mode. + # rs_list: 127.0.0.1:2881 + # Cluster name for the proxy OceanBase Database. The default value is obcluster. This value must be set to the same with the appname for OceanBase Database. + # cluster_name: obcluster + # Password for obproxy system tenant. The default value is empty. + # obproxy_sys_password: + # Password for proxyro. proxyro_password must be the same with proxyro_password. The default value is empty. + # observer_sys_password: +obagent: + # Set dependent components for the component. + # When the associated configurations are not done, OBD will automatically get the these configurations from the dependent components. + depends: + - oceanbase-ce + # The list of servers to be monitored. This list is consistent with the servers in oceanbase-ce. + servers: + - name: server1 + # Please don't use hostname, only IP is supported. + ip: 172.19.33.2 + - name: server2 + ip: 172.19.33.3 + - name: server3 + ip: 172.19.33.4 + global: + # The working directory for obagent. obagent is started under this directory. This is a required field. + home_path: /root/obagent + # The port that pulls and manages the metrics. The default port number is 8088. + # server_port: 8088 + # Debug port for pprof. The default port number is 8089. + # pprof_port: 8089 + # Log level. The default value is INFO. + # log_level: INFO + # Log path. The default value is log/monagent.log. + # log_path: log/monagent.log + # Encryption method. OBD supports aes and plain. The default value is plain. + # crypto_method: plain + # Path to store the crypto key. The default value is conf/.config_secret.key. + # crypto_path: conf/.config_secret.key + # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. + # log_size: 30 + # Expiration time for logs. The default value is 7 days. + # log_expire_day: 7 + # The maximum number for log files. The default value is 10. + # log_file_count: 10 + # Whether to use local time for log files. The default value is true. + # log_use_localtime: true + # Whether to enable log compression. The default value is true. + # log_compress: true + # Username for HTTP authentication. The default value is admin. + # http_basic_auth_user: admin + # Password for HTTP authentication. The default value is root. + # http_basic_auth_password: root + # Username for debug service. The default value is admin. + # pprof_basic_auth_user: admin + # Password for debug service. The default value is root. + # pprof_basic_auth_password: root + # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. + # monitor_user: root + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. + # monitor_password: + # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. + # sql_port: 2881 + # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. + # rpc_port: 2882 + # Cluster name for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the appname in oceanbase-ce. + # cluster_name: obcluster + # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. + # cluster_id: 1 + # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. + # zone_name: zone1 + # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. + # ob_monitor_status: active + # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. + # host_monitor_status: active + # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. + # disable_http_basic_auth: false + # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. + # disable_pprof_basic_auth: false +prometheus: + servers: + - 192.168.1.5 + depends: + - obagent + global: + # The working directory for prometheus. prometheus is started under this directory. This is a required field. + home_path: /root/prometheus + # address: 0.0.0.0 # The ip address to bind to. Along with port, corresponds to the `web.listen-address` parameter. + # port: 9090 # The http port to use. Along with address, corresponds to the `web.listen-address` parameter. + # enable_lifecycle: true # Enable shutdown and reload via HTTP request. Corresponds to the `web.enable-lifecycle` parameter. + # data_dir: /root/prometheus/data # Base path for metrics storage. Corresponds to the `storage.tsdb.path` parameter. + # basic_auth_users: # Usernames and passwords that have full access to the web server via basic authentication. Corresponds to the `basic_auth_users` parameter. + # : # The format of `basic_auth_users` : the key is the user name and the value is the password. + # web_config: # Content of Prometheus web service config file. The format is consistent with the file. However, `basic_auth_users` cannot be set in it. Please set `basic_auth_users` above if needed. Corresponds to the `web.config.file` parameter. + # tls_server_config: + # # Certificate and key files for server to use to authenticate to client. + # cert_file: + # key_file: + # config: # Configuration of the Prometheus service. The format is consistent with the Prometheus config file. Corresponds to the `config.file` parameter. + # rule_files: + # - rules/*rules.yaml + # scrape_configs: + # - job_name: prometheus + # metrics_path: /metrics + # scheme: http + # static_configs: + # - targets: + # - localhost:9090 + # - job_name: node + # basic_auth: + # username: admin + # password: root + # metrics_path: /metrics/node/host + # scheme: http + # file_sd_configs: # Set the targets to be collected by reading local files. The example is to collect targets corresponding to all yaml files in the 'targets' directory under $home_path. + # - files: + # - 'targets/*.yaml' + # - job_name: ob_basic + # basic_auth: + # username: admin + # password: root + # metrics_path: /metrics/ob/basic + # scheme: http + # file_sd_configs: + # - files: + # - 'targets/*.yaml' + # - job_name: ob_extra + # basic_auth: + # username: admin + # password: root + # metrics_path: /metrics/ob/extra + # scheme: http + # file_sd_configs: + # - files: + # - 'targets/*.yaml' + # - job_name: agent + # basic_auth: + # username: admin + # password: root + # metrics_path: /metrics/stat + # scheme: http + # file_sd_configs: + # - files: + # - 'targets/*.yaml' + # additional_parameters: # Additional parameters for Prometheus service, among which `web.listen-address`, `web.enable-lifecycle`, `storage.tsdb.path`, `config.file` and `web.config.file` cannot be set. Please set them in the corresponding configuration above if needed. + # - log.level: debug +grafana: + servers: + - 192.168.1.5 + depends: + - prometheus + global: + home_path: /root/grafana + login_password: oceanbase # Grafana login password. The default value is 'oceanbase'. + # data_dir: # Path to where grafana can store temp files, sessions, and the sqlite3 db (if that is used).$data_dir can be empty. The default value is $home_path/data. + # logs_dir: # Directory where grafana can store logs, can be empty. The default value is $data_dir/log. + # plugins_dir: # Directory where grafana will automatically scan and look for plugins, can be empty. The default value is $data_dir/plugins. + # provisioning_dir: # folder that contains provisioning config files that grafana will apply on startup and while running, can be empty. The default value is $home_path/conf/provisioning. + # temp_data_lifetime: # How long temporary images in data directory should be kept. Supported modifiers h (hours), m (minutes), Use 0 to never clean up temporary files, can be empty. The default value is 24h. + # log_max_days: # Expired days of log file(delete after max days), can be empty. The default value is 7. + # domian: # The public facing domain name used to access grafana from a browser, can be empty. The default value is $server.ip. + # port: # The http port to use, can be empty. The default value is 3000. + + # # list of datasources to insert/update depending on what's available in the database, can be empty. + # # For more parameter settings, please refer to https://grafana.com/docs/grafana/latest/administration/provisioning/#datasources + # datasources: + # name: # name of the datasource. Required and should not be 'OB-Prometheus' + # type: # datasource type. Required + # access: # access mode. direct or proxy. Required + # url: # the url of datasource + + # list of dashboards providers that load dashboards into Grafana from the local filesystem, can be empty. + # For more information, please refer to https://grafana.com/docs/grafana/latest/administration/provisioning/#dashboards + # providers: + # name: # an unique provider name. Required and should not be 'OceanBase Metrics' + # type: # provider type. Default to 'file' + # options: + # path: # path to dashboard files on disk. Required when using the 'file' type + + # # customize your Grafana instance by adding/modifying the custom configuration as follows + # # for more information, please refer to https://grafana.com/docs/grafana/latest/setup-grafana/configure-grafana/#configure-grafana + # # Here, setting parameters is required for format conversion. + # # For example, if the original grafana configuration format is + # # + # # [section1.section2] + # # key1 = value1 + # # key2 = value2 + # # + # # Then when writing the configuration below, you need to write it as + # # + # # section1: + # # section2: + # # key1: value1 + # # key2: value2 + # # + # # Here we only list one item, because there are more than 500 items. Please add them according to your own needs. + # customize_config: + # # original grafana configuration format is + # # [server] + # # protocol = http + # server: + # protocol: http \ No newline at end of file diff --git a/example/local-example.yaml b/example/local-example.yaml index ea254f1..dd15e63 100644 --- a/example/local-example.yaml +++ b/example/local-example.yaml @@ -22,7 +22,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/mini-distributed-example.yaml b/example/mini-distributed-example.yaml index e366c85..3cf262e 100644 --- a/example/mini-distributed-example.yaml +++ b/example/mini-distributed-example.yaml @@ -24,10 +24,9 @@ oceanbase-ce: memory_limit: 6G # The maximum running memory for an observer system_memory: 1G # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. datafile_size: 20G # Size of the data file. - log_disk_size: 24G # The size of disk space used by the clog files. + log_disk_size: 15G # The size of disk space used by the clog files. cpu_count: 16 production_mode: false - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/mini-distributed-with-obproxy-example.yaml b/example/mini-distributed-with-obproxy-example.yaml index 7d4b097..3301a88 100644 --- a/example/mini-distributed-with-obproxy-example.yaml +++ b/example/mini-distributed-with-obproxy-example.yaml @@ -24,10 +24,9 @@ oceanbase-ce: memory_limit: 6G # The maximum running memory for an observer system_memory: 1G # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. datafile_size: 20G # Size of the data file. - log_disk_size: 24G # The size of disk space used by the clog files. + log_disk_size: 15G # The size of disk space used by the clog files. cpu_count: 16 production_mode: false - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/mini-local-example.yaml b/example/mini-local-example.yaml index 4a6bef8..9d8055a 100755 --- a/example/mini-local-example.yaml +++ b/example/mini-local-example.yaml @@ -21,10 +21,9 @@ oceanbase-ce: memory_limit: 6G # The maximum running memory for an observer system_memory: 1G # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. datafile_size: 20G # Size of the data file. - log_disk_size: 24G # The size of disk space used by the clog files. + log_disk_size: 15G # The size of disk space used by the clog files. cpu_count: 16 production_mode: false - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/mini-single-example.yaml b/example/mini-single-example.yaml index d96c623..24faa42 100755 --- a/example/mini-single-example.yaml +++ b/example/mini-single-example.yaml @@ -28,10 +28,9 @@ oceanbase-ce: memory_limit: 6G # The maximum running memory for an observer system_memory: 1G # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. datafile_size: 20G # Size of the data file. - log_disk_size: 24G # The size of disk space used by the clog files. + log_disk_size: 15G # The size of disk space used by the clog files. cpu_count: 16 production_mode: false - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/mini-single-with-obproxy-example.yaml b/example/mini-single-with-obproxy-example.yaml index da967aa..dd5ee0e 100644 --- a/example/mini-single-with-obproxy-example.yaml +++ b/example/mini-single-with-obproxy-example.yaml @@ -28,10 +28,9 @@ oceanbase-ce: memory_limit: 6G # The maximum running memory for an observer system_memory: 1G # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. datafile_size: 20G # Size of the data file. - log_disk_size: 24G # The size of disk space used by the clog files. + log_disk_size: 15G # The size of disk space used by the clog files. cpu_count: 16 production_mode: false - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/obagent/distributed-with-obproxy-and-obagent-example.yaml b/example/obagent/distributed-with-obproxy-and-obagent-example.yaml index b0cebce..55d1b30 100644 --- a/example/obagent/distributed-with-obproxy-and-obagent-example.yaml +++ b/example/obagent/distributed-with-obproxy-and-obagent-example.yaml @@ -25,7 +25,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. @@ -100,40 +99,34 @@ obagent: global: # The working directory for obagent. obagent is started under this directory. This is a required field. home_path: /root/obagent - # The port that pulls and manages the metrics. The default port number is 8088. - server_port: 8088 - # Debug port for pprof. The default port number is 8089. - pprof_port: 8089 - # Log level. The default value is INFO. - log_level: INFO + # The port of monitor agent. The default port number is 8088. + monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + mgragent_http_port: 8089 # Log path. The default value is log/monagent.log. log_path: log/monagent.log - # Encryption method. OBD supports aes and plain. The default value is plain. - crypto_method: plain - # Path to store the crypto key. The default value is conf/.config_secret.key. - # crypto_path: conf/.config_secret.key - # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. - log_size: 30 - # Expiration time for logs. The default value is 7 days. - log_expire_day: 7 - # The maximum number for log files. The default value is 10. - log_file_count: 10 - # Whether to use local time for log files. The default value is true. - # log_use_localtime: true - # Whether to enable log compression. The default value is true. - # log_compress: true + # The log level of manager agent. + mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + mgragent_log_max_backups: 15 + # The log level of monitor agent. + monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + monagent_log_max_backups: 15 # Username for HTTP authentication. The default value is admin. http_basic_auth_user: admin # Password for HTTP authentication. The default value is root. http_basic_auth_password: root - # Username for debug service. The default value is admin. - pprof_basic_auth_user: admin - # Password for debug service. The default value is root. - pprof_basic_auth_password: root - # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. - # monitor_user: root - # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. - # monitor_password: + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. + # monitor_password: # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. # sql_port: 2881 # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. @@ -142,13 +135,24 @@ obagent: # cluster_name: obcluster # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. # cluster_id: 1 - # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. - # zone_name: zone1 + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + # ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + # ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + # ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + # observer_log_path: /root/observer/log # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. ob_monitor_status: active - # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. - host_monitor_status: active - # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. - disable_http_basic_auth: false - # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. - disable_pprof_basic_auth: false \ No newline at end of file + # Synchronize the obagent-related information to the specified path of the remote host, as the targets specified by `file_sd_config` in the Prometheus configuration. + # For prometheus that depends on obagent, it can be specified to $home_path/targets of prometheus. + # For independently deployed prometheus, specify the files to be collected by setting `config` -> `scrape_configs` -> `file_sd_configs` -> `files`. For details, please refer to prometheus-only-example.yaml. + # target_sync_configs: + # - host: 192.168.1.1 + # target_dir: /root/prometheus/targets + # username: your username + # password: your password if need + # key_file: your ssh-key file path if need + # port: your ssh port, default 22 + # timeout: ssh connection timeout (second), default 30 \ No newline at end of file diff --git a/example/obagent/obagent-only-1.2.0-example.yaml b/example/obagent/obagent-only-1.2.0-example.yaml new file mode 100644 index 0000000..d6a95b4 --- /dev/null +++ b/example/obagent/obagent-only-1.2.0-example.yaml @@ -0,0 +1,84 @@ +## Only need to configure when remote login is required +# user: +# username: your username +# password: your password if need +# key_file: your ssh-key file path if need +# port: your ssh port, default 22 +# timeout: ssh connection timeout (second), default 30 +obagent: + servers: + # Please don't use hostname, only IP can be supported + - 192.168.1.2 + - 192.168.1.3 + - 192.168.1.4 + global: + # The working directory for obagent. obagent is started under this directory. This is a required field. + home_path: /root/obagent + # The port that pulls and manages the metrics. The default port number is 8088. + server_port: 8088 + # Debug port for pprof. The default port number is 8089. + pprof_port: 8089 + # Log path. The default value is log/monagent.log. + log_path: log/monagent.log + # Encryption method. OBD supports aes and plain. The default value is plain. + crypto_method: plain + # Path to store the crypto key. The default value is conf/.config_secret.key. + # crypto_path: conf/.config_secret.key + # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. + log_size: 30 + # Expiration time for logs. The default value is 7 days. + log_expire_day: 7 + # The maximum number for log files. The default value is 10. + log_file_count: 10 + # Whether to use local time for log files. The default value is true. + # log_use_localtime: true + # Whether to enable log compression. The default value is true. + # log_compress: true + # Username for HTTP authentication. The default value is admin. + http_basic_auth_user: admin + # Password for HTTP authentication. The default value is root. + http_basic_auth_password: root + # Username for debug service. The default value is admin. + pprof_basic_auth_user: admin + # Password for debug service. The default value is root. + pprof_basic_auth_password: root + # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. + monitor_user: root + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. + monitor_password: + # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. + sql_port: 2881 + # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. + rpc_port: 2882 + # Cluster name for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the appname in oceanbase-ce. + cluster_name: obcluster + # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. + cluster_id: 1 + # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. + ob_monitor_status: active + # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. + host_monitor_status: active + # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. + disable_http_basic_auth: false + # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. + disable_pprof_basic_auth: false + # Synchronize the obagent-related information to the specified path of the remote host, as the targets specified by `file_sd_config` in the Prometheus configuration. + # For prometheus that depends on obagent, it can be specified to $home_path/targets of prometheus. + # For independently deployed prometheus, specify the files to be collected by setting `config` -> `scrape_configs` -> `file_sd_configs` -> `files`. For details, please refer to prometheus-only-example.yaml. + # target_sync_configs: + # - host: 192.168.1.1 + # target_dir: /root/prometheus/targets + # username: your username + # password: your password if need + # key_file: your ssh-key file path if need + # port: your ssh port, default 22 + # timeout: ssh connection timeout (second), default 30 + 192.168.1.2: + # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. + zone_name: zone1 + 192.168.1.3: + # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. + zone_name: zone2 + 192.168.1.4: + # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. + zone_name: zone3 \ No newline at end of file diff --git a/example/obagent/obagent-only-example.yaml b/example/obagent/obagent-only-example.yaml index 3419f51..fc45667 100644 --- a/example/obagent/obagent-only-example.yaml +++ b/example/obagent/obagent-only-example.yaml @@ -14,39 +14,33 @@ obagent: global: # The working directory for obagent. obagent is started under this directory. This is a required field. home_path: /root/obagent - # The port that pulls and manages the metrics. The default port number is 8088. - server_port: 8088 - # Debug port for pprof. The default port number is 8089. - pprof_port: 8089 - # Log level. The default value is INFO. - log_level: INFO + # The port of monitor agent. The default port number is 8088. + monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + mgragent_http_port: 8089 # Log path. The default value is log/monagent.log. log_path: log/monagent.log - # Encryption method. OBD supports aes and plain. The default value is plain. - crypto_method: plain - # Path to store the crypto key. The default value is conf/.config_secret.key. - # crypto_path: conf/.config_secret.key - # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. - log_size: 30 - # Expiration time for logs. The default value is 7 days. - log_expire_day: 7 - # The maximum number for log files. The default value is 10. - log_file_count: 10 - # Whether to use local time for log files. The default value is true. - # log_use_localtime: true - # Whether to enable log compression. The default value is true. - # log_compress: true + # The log level of manager agent. + mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + mgragent_log_max_backups: 15 + # The log level of monitor agent. + monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + monagent_log_max_backups: 15 # Username for HTTP authentication. The default value is admin. http_basic_auth_user: admin # Password for HTTP authentication. The default value is root. http_basic_auth_password: root - # Username for debug service. The default value is admin. - pprof_basic_auth_user: admin - # Password for debug service. The default value is root. - pprof_basic_auth_password: root - # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. - monitor_user: root - # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. monitor_password: # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. sql_port: 2881 @@ -56,14 +50,16 @@ obagent: cluster_name: obcluster # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. cluster_id: 1 + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + observer_log_path: /root/observer/log # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. ob_monitor_status: active - # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. - host_monitor_status: active - # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. - disable_http_basic_auth: false - # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. - disable_pprof_basic_auth: false # Synchronize the obagent-related information to the specified path of the remote host, as the targets specified by `file_sd_config` in the Prometheus configuration. # For prometheus that depends on obagent, it can be specified to $home_path/targets of prometheus. # For independently deployed prometheus, specify the files to be collected by setting `config` -> `scrape_configs` -> `file_sd_configs` -> `files`. For details, please refer to prometheus-only-example.yaml. diff --git a/example/obproxy/distributed-with-obproxy-example.yaml b/example/obproxy/distributed-with-obproxy-example.yaml index f8d556f..447237a 100644 --- a/example/obproxy/distributed-with-obproxy-example.yaml +++ b/example/obproxy/distributed-with-obproxy-example.yaml @@ -25,7 +25,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/oceanbase-3.x/distributed-with-obproxy-and-obagent-example.yaml b/example/oceanbase-3.x/distributed-with-obproxy-and-obagent-example.yaml new file mode 100644 index 0000000..300c798 --- /dev/null +++ b/example/oceanbase-3.x/distributed-with-obproxy-and-obagent-example.yaml @@ -0,0 +1,159 @@ +## Only need to configure when remote login is required +# user: +# username: your username +# password: your password if need +# key_file: your ssh-key file path if need +# port: your ssh port, default 22 +# timeout: ssh connection timeout (second), default 30 +oceanbase-ce: + servers: + - name: server1 + # Please don't use hostname, only IP can be supported + ip: 172.19.33.2 + - name: server2 + ip: 172.19.33.3 + - name: server3 + ip: 172.19.33.4 + global: + # Please set devname as the network adaptor's name whose ip is in the setting of severs. + # if set severs as "127.0.0.1", please set devname as "lo" + # if current ip is 192.168.1.10, and the ip's network adaptor's name is "eth0", please use "eth0" + devname: eth0 + # if current hardware's memory capacity is smaller than 50G, please use the setting of "mini-single-example.yaml" and do a small adjustment. + memory_limit: 64G # The maximum running memory for an observer + # The reserved system memory. system_memory is reserved for general tenants. The default value is 30G. + system_memory: 30G + datafile_size: 192G # Size of the data file. + log_disk_size: 192G # The size of disk space used by the clog files. + enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. + enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. + max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. + # root_password: # root user password + # In this example , support multiple ob process in single node, so different process use different ports. + # If deploy ob cluster in multiple nodes, the port and path setting can be same. + server1: + mysql_port: 2881 # External port for OceanBase Database. The default value is 2881. DO NOT change this value after the cluster is started. + rpc_port: 2882 # Internal port for OceanBase Database. The default value is 2882. DO NOT change this value after the cluster is started. + # The working directory for OceanBase Database. OceanBase Database is started under this directory. This is a required field. + home_path: /root/observer + # The directory for data storage. The default value is $home_path/store. + # data_dir: /data + # The directory for clog, ilog, and slog. The default value is the same as the data_dir value. + # redo_dir: /redo + zone: zone1 + server2: + mysql_port: 2881 # External port for OceanBase Database. The default value is 2881. DO NOT change this value after the cluster is started. + rpc_port: 2882 # Internal port for OceanBase Database. The default value is 2882. DO NOT change this value after the cluster is started. + # The working directory for OceanBase Database. OceanBase Database is started under this directory. This is a required field. + home_path: /root/observer + # The directory for data storage. The default value is $home_path/store. + # data_dir: /data + # The directory for clog, ilog, and slog. The default value is the same as the data_dir value. + # redo_dir: /redo + zone: zone2 + server3: + mysql_port: 2881 # External port for OceanBase Database. The default value is 2881. DO NOT change this value after the cluster is started. + rpc_port: 2882 # Internal port for OceanBase Database. The default value is 2882. DO NOT change this value after the cluster is started. + # The working directory for OceanBase Database. OceanBase Database is started under this directory. This is a required field. + home_path: /root/observer + # The directory for data storage. The default value is $home_path/store. + # data_dir: /data + # The directory for clog, ilog, and slog. The default value is the same as the data_dir value. + # redo_dir: /redo + zone: zone3 +obproxy-ce: + servers: + - 192.168.1.5 + # Set dependent components for the component. + # When the associated configurations are not done, OBD will automatically get the these configurations from the dependent components. + depends: + - oceanbase-ce + global: + listen_port: 2883 # External port. The default value is 2883. + prometheus_listen_port: 2884 # The Prometheus port. The default value is 2884. + home_path: /root/obproxy + # oceanbase root server list + # format: ip:mysql_port;ip:mysql_port. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. + # rs_list: 192.168.1.2:2881;192.168.1.3:2881;192.168.1.4:2881 + enable_cluster_checkout: false + # observer cluster name, consistent with oceanbase-ce's appname. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. + # cluster_name: obcluster + skip_proxy_sys_private_check: true + enable_strict_kernel_release: false + # obproxy_sys_password: # obproxy sys user password, can be empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. + # observer_sys_password: # proxyro user pasword, consistent with oceanbase-ce's proxyro_password, can be empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. +obagent: + version: 1.2.0 + # The list of servers to be monitored. This list is consistent with the servers in oceanbase-ce. + servers: + - name: server1 + # Please don't use hostname, only IP can be supported + ip: 172.19.33.2 + - name: server2 + ip: 172.19.33.3 + - name: server3 + ip: 172.19.33.4 + # Set dependent components for the component. + # When the associated configurations are not done, OBD will automatically get the these configurations from the dependent components. + depends: + - oceanbase-ce + global: + # The working directory for obagent. obagent is started under this directory. This is a required field. + home_path: /root/obagent + # The port of monitor agent. The default port number is 8088. + monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + mgragent_http_port: 8089 + # Log path. The default value is log/monagent.log. + log_path: log/monagent.log + # The log level of manager agent. + mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + mgragent_log_max_backups: 15 + # The log level of monitor agent. + monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + monagent_log_max_backups: 15 + # Username for HTTP authentication. The default value is admin. + http_basic_auth_user: admin + # Password for HTTP authentication. The default value is root. + http_basic_auth_password: root + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. + # monitor_password: + # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. + # sql_port: 2881 + # The RPC port for observer. The default value is 2882. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the rpc_port in oceanbase-ce. + # rpc_port: 2882 + # Cluster name for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the appname in oceanbase-ce. + # cluster_name: obcluster + # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. + # cluster_id: 1 + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + # ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + # ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + # ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + # observer_log_path: /root/observer/log + # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. + ob_monitor_status: active + # Synchronize the obagent-related information to the specified path of the remote host, as the targets specified by `file_sd_config` in the Prometheus configuration. + # For prometheus that depends on obagent, it can be specified to $home_path/targets of prometheus. + # For independently deployed prometheus, specify the files to be collected by setting `config` -> `scrape_configs` -> `file_sd_configs` -> `files`. For details, please refer to prometheus-only-example.yaml. + # target_sync_configs: + # - host: 192.168.1.1 + # target_dir: /root/prometheus/targets + # username: your username + # password: your password if need + # key_file: your ssh-key file path if need + # port: your ssh port, default 22 + # timeout: ssh connection timeout (second), default 30 \ No newline at end of file diff --git a/example/ocp-express/ocp-express-only.yaml b/example/ocp-express/ocp-express-only.yaml new file mode 100644 index 0000000..3608cc2 --- /dev/null +++ b/example/ocp-express/ocp-express-only.yaml @@ -0,0 +1,34 @@ +## Only need to configure when remote login is required +# user: +# username: your username +# password: your password if need +# key_file: your ssh-key file path if need +# port: your ssh port, default 22 +# timeout: ssh connection timeout (second), default 30 +ocp-server: + servers: + - name: server1 + ip: 192.168.1.1 + global: + # The working directory for ocp express. ocp express is started under this directory. This is a required field. + home_path: /root/ocp-server + # log_dir: /root/ocp-server/log # The log directory of ocp express server. The default value is {home_path}/log. + memory_size: 1G # The memory size of ocp-express server. The recommend value is 512MB * (expect node num + expect tenant num) * 60MB. + jdbc_url: jdbc:oceanbase://192.168.1.1:2881/meta_db # jdbc connection string to connect to the meta db + jdbc_username: username # username to connect to meta db + jdbc_password: '' # password to connect to meta db + port: 8080 # The http port to use. + cluster_name: obcluster # the cluster name of oceanbase cluster. Refer to the configuration item appname of oceanbase + ob_cluster_id: 1 # the cluster id of oceanbase cluster. Refer to the configuration item cluster_id of oceanbase + root_sys_password: # the pass of oceanbase cluster. Refer to the configuration item cluster_id of oceanbase + agent_username: # The username of obagent + agent_password: # The password of obagent + # logging_file_total_size_cap: 10G # The total log file size of ocp-express server + # logging_file_max_history: 1 # The maximum of retention days the log archive log files to keep. The default value is unlimited + server_addresses: # The cluster info for oceanbase cluster + - address: 127.0.0.1 # The address of oceanbase server + svrPort: 2882 # The rpc port of oceanbase server + sqlPort: 2881 # The mysql port of oceanbase server + withRootServer: true # Is the oceanbase server a root server of cluster. + agentMgrPort: 62888 # The port of obagent manager process + agentMonPort: 62889 # The port of obagent monitor process diff --git a/example/prometheus/distributed-with-obagent-and-prometheus-example.yaml b/example/prometheus/distributed-with-obagent-and-prometheus-example.yaml index fa4fea9..1eff5af 100644 --- a/example/prometheus/distributed-with-obagent-and-prometheus-example.yaml +++ b/example/prometheus/distributed-with-obagent-and-prometheus-example.yaml @@ -25,7 +25,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. @@ -79,39 +78,33 @@ obagent: global: # The working directory for obagent. obagent is started under this directory. This is a required field. home_path: /root/obagent - # The port that pulls and manages the metrics. The default port number is 8088. - server_port: 8088 - # Debug port for pprof. The default port number is 8089. - pprof_port: 8089 - # Log level. The default value is INFO. - log_level: INFO + # The port of monitor agent. The default port number is 8088. + monagent_http_port: 8088 + # The port of manager agent. The default port number is 8089. + mgragent_http_port: 8089 # Log path. The default value is log/monagent.log. log_path: log/monagent.log - # Encryption method. OBD supports aes and plain. The default value is plain. - crypto_method: plain - # Path to store the crypto key. The default value is conf/.config_secret.key. - # crypto_path: conf/.config_secret.key - # Size for a single log file. Log size is measured in Megabytes. The default value is 30M. - log_size: 30 - # Expiration time for logs. The default value is 7 days. - log_expire_day: 7 - # The maximum number for log files. The default value is 10. - log_file_count: 10 - # Whether to use local time for log files. The default value is true. - # log_use_localtime: true - # Whether to enable log compression. The default value is true. - # log_compress: true + # The log level of manager agent. + mgragent_log_level: info + # The total size of manager agent.Log size is measured in Megabytes. The default value is 30M. + mgragent_log_max_size: 30 + # Expiration time for manager agent logs. The default value is 30 days. + mgragent_log_max_days: 30 + # The maximum number for manager agent log files. The default value is 15. + mgragent_log_max_backups: 15 + # The log level of monitor agent. + monagent_log_level: info + # The total size of monitor agent.Log size is measured in Megabytes. The default value is 200M. + monagent_log_max_size: 200 + # Expiration time for monitor agent logs. The default value is 30 days. + monagent_log_max_days: 30 + # The maximum number for monitor agent log files. The default value is 15. + monagent_log_max_backups: 15 # Username for HTTP authentication. The default value is admin. http_basic_auth_user: admin # Password for HTTP authentication. The default value is root. http_basic_auth_password: root - # Username for debug service. The default value is admin. - pprof_basic_auth_user: admin - # Password for debug service. The default value is root. - pprof_basic_auth_password: root - # Monitor username for OceanBase Database. The user must have read access to OceanBase Database as a system tenant. The default value is root. - # monitor_user: root - # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the root_password in oceanbase-ce. + # Monitor password for OceanBase Database. The default value is empty. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the ocp_agent_monitor_password in oceanbase-ce. # monitor_password: # The SQL port for observer. The default value is 2881. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the mysql_port in oceanbase-ce. # sql_port: 2881 @@ -121,16 +114,16 @@ obagent: # cluster_name: obcluster # Cluster ID for OceanBase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the cluster_id in oceanbase-ce. # cluster_id: 1 - # Zone name for your observer. The default value is zone1. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the zone name in oceanbase-ce. - # zone_name: zone1 + # The redo dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the redo_dir in oceanbase-ce. + # ob_log_path: /root/observer/store + # The data dir for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the data_dir in oceanbase-ce. + # ob_data_path: /root/observer/store + # The work directory for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the home_path in oceanbase-ce. + # ob_install_path: /root/observer + # The log path for Oceanbase Database. When a depends exists, OBD gets this value from the oceanbase-ce of the depends. The value is the same as the {home_path}/log in oceanbase-ce. + # observer_log_path: /root/observer/log # Monitor status for OceanBase Database. Active is to enable. Inactive is to disable. The default value is active. When you deploy an cluster automatically, OBD decides whether to enable this parameter based on depends. ob_monitor_status: active - # Monitor status for your host. Active is to enable. Inactive is to disable. The default value is active. - host_monitor_status: active - # Whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. The default value is false. - disable_http_basic_auth: false - # Whether to disable the basic authentication for the debug interface. True is to disable. False is to enable. The default value is false. - disable_pprof_basic_auth: false prometheus: servers: - 192.168.1.5 diff --git a/example/single-example.yaml b/example/single-example.yaml index 5e00e6e..4c5cecf 100644 --- a/example/single-example.yaml +++ b/example/single-example.yaml @@ -29,7 +29,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/example/single-with-obproxy-example.yaml b/example/single-with-obproxy-example.yaml index fe041f7..b39a4ac 100644 --- a/example/single-with-obproxy-example.yaml +++ b/example/single-with-obproxy-example.yaml @@ -29,7 +29,6 @@ oceanbase-ce: system_memory: 30G datafile_size: 192G # Size of the data file. log_disk_size: 192G # The size of disk space used by the clog files. - syslog_level: INFO # System log level. The default value is INFO. enable_syslog_wf: false # Print system logs whose levels are higher than WARNING to a separate log file. The default value is true. enable_syslog_recycle: true # Enable auto system log recycling or not. The default value is false. max_syslog_file_count: 4 # The maximum number of reserved log files before enabling auto recycling. The default value is 0. diff --git a/optimize/obproxy/4.1.0/sysbench.yaml b/optimize/obproxy/4.1.0/sysbench.yaml new file mode 100644 index 0000000..41eef71 --- /dev/null +++ b/optimize/obproxy/4.1.0/sysbench.yaml @@ -0,0 +1,10 @@ +test: + system_config: + - name: proxy_mem_limited + value: 4G + - name: enable_compression_protocol + value: false + need_restart: true + value_type: BOOL + - name: enable_qos + value: false \ No newline at end of file diff --git a/optimize/obproxy/4.1.0/tpcc.yaml b/optimize/obproxy/4.1.0/tpcc.yaml new file mode 100644 index 0000000..c7cb45c --- /dev/null +++ b/optimize/obproxy/4.1.0/tpcc.yaml @@ -0,0 +1,10 @@ +build: + system_config: + - name: proxy_mem_limited + value: 4G + - name: enable_compression_protocol + value: false + need_restart: true + value_type: BOOL + - name: enable_qos + value: false \ No newline at end of file diff --git a/optimize/oceanbase-ce/4.1.0/optimizer.yaml b/optimize/oceanbase-ce/4.1.0/optimizer.yaml new file mode 100644 index 0000000..2d83eac --- /dev/null +++ b/optimize/oceanbase-ce/4.1.0/optimizer.yaml @@ -0,0 +1,14 @@ +system_config: + default: + query_sql: select value from oceanbase.__all_virtual_sys_parameter_stat where name="{name}" + modify_sql: "alter system set {name}={value}" + tenant: + query_sql: select * from oceanbase.__all_virtual_tenant_parameter_info where name like "{name}" and tenant_id={tenant_id} + modify_sql: "alter system set {name}={value} {tenant_where}" +variables: + default: + query_sql: "select value from oceanbase.__all_virtual_sys_variable where tenant_id = {tenant_id} and name = '{name}'" + modify_sql: "ALTER TENANT {tenant} SET VARIABLES {name} = {value}" +exec_sql: + clean_cache: + modify_sql: "alter system flush plan cache global" \ No newline at end of file diff --git a/optimize/oceanbase-ce/4.1.0/sysbench.yaml b/optimize/oceanbase-ce/4.1.0/sysbench.yaml new file mode 100644 index 0000000..3edbfc0 --- /dev/null +++ b/optimize/oceanbase-ce/4.1.0/sysbench.yaml @@ -0,0 +1,61 @@ +test: + variables: + - name: binlog_row_image + value: MINIMAL + - name: auto_increment_cache_size + value: 10000000 + system_config: + - name: enable_sql_audit + value: 'false' + - name: sleep + value: 3 + optimizer: sleep + - name: enable_early_lock_release + value: 'false' + optimizer: tenant + - name: syslog_level + value: 'ERROR' + - name: enable_perf_event + value: false + value_type: BOOL + - name: _enable_defensive_check + value: false + - name: _rowsets_enabled + value: 'false' + optimizer: tenant + - name: _enable_newsort + value: 'false' + - name: _trace_control_info + value: '' + optimizer: tenant + - name: _lcl_op_interval + value: 0ms + - name: writing_throttling_trigger_percentage + value: 100 + optimizer: tenant + - name: default_auto_increment_mode + value: 'NOORDER' + optimizer: tenant + - name: enable_monotonic_weak_read + value: 'false' + optimizer: tenant + - name: _enable_adaptive_compaction + value: 'false' + optimizer: tenant + - name: enable_record_trace_log + value: 'false' + - name: cpu_quota_concurrency + value: 2 + optimizer: tenant + - name: _ob_enable_prepared_statement + value: 'true' + - name: _pushdown_storage_level + value: 0 + optimizer: tenant + - name: ignore_replay_checksum_error + value: 'true' + - name: weak_read_version_refresh_interval + value: 2s + - name: freeze_trigger_percentage + value: 40 + optimizer: tenant \ No newline at end of file diff --git a/optimize/oceanbase-ce/4.1.0/tpcc.yaml b/optimize/oceanbase-ce/4.1.0/tpcc.yaml new file mode 100644 index 0000000..b72d27a --- /dev/null +++ b/optimize/oceanbase-ce/4.1.0/tpcc.yaml @@ -0,0 +1,63 @@ +build: + variables: + - name: binlog_row_image + value: MINIMAL + - name: auto_increment_cache_size + value: 10000000 + system_config: + - name: enable_sql_audit + value: 'false' + - name: sleep + value: 5 + optimizer: sleep + - name: enable_early_lock_release + value: 'false' + optimizer: tenant + - name: syslog_level + value: 'ERROR' + - name: enable_perf_event + value: false + value_type: BOOL + - name: _enable_defensive_check + value: false + - name: _rowsets_enabled + value: false + optimizer: tenant + - name: _enable_newsort + value: false + - name: _trace_control_info + value: '' + optimizer: tenant + - name: _lcl_op_interval + value: 0ms + - name: default_auto_increment_mode + value: 'NOORDER' + optimizer: tenant + - name: enable_monotonic_weak_read + value: 'false' + optimizer: tenant + - name: _enable_adaptive_compaction + value: 'false' + optimizer: tenant + - name: enable_record_trace_log + value: 'false' + - name: cpu_quota_concurrency + value: 2 + optimizer: tenant + - name: _ob_enable_prepared_statement + value: 'true' + - name: _pushdown_storage_level + value: 0 + optimizer: tenant + - name: ignore_replay_checksum_error + value: 'true' + - name: weak_read_version_refresh_interval + value: 2s + - name: freeze_trigger_percentage + value: 40 + optimizer: tenant +test: + system_config: + - name: writing_throttling_trigger_percentage + value: 100 + optimizer: tenant \ No newline at end of file diff --git a/optimize/oceanbase/4.1.0/optimizer.yaml b/optimize/oceanbase/4.1.0/optimizer.yaml new file mode 100644 index 0000000..2d83eac --- /dev/null +++ b/optimize/oceanbase/4.1.0/optimizer.yaml @@ -0,0 +1,14 @@ +system_config: + default: + query_sql: select value from oceanbase.__all_virtual_sys_parameter_stat where name="{name}" + modify_sql: "alter system set {name}={value}" + tenant: + query_sql: select * from oceanbase.__all_virtual_tenant_parameter_info where name like "{name}" and tenant_id={tenant_id} + modify_sql: "alter system set {name}={value} {tenant_where}" +variables: + default: + query_sql: "select value from oceanbase.__all_virtual_sys_variable where tenant_id = {tenant_id} and name = '{name}'" + modify_sql: "ALTER TENANT {tenant} SET VARIABLES {name} = {value}" +exec_sql: + clean_cache: + modify_sql: "alter system flush plan cache global" \ No newline at end of file diff --git a/optimize/oceanbase/4.1.0/sysbench.yaml b/optimize/oceanbase/4.1.0/sysbench.yaml new file mode 100644 index 0000000..3edbfc0 --- /dev/null +++ b/optimize/oceanbase/4.1.0/sysbench.yaml @@ -0,0 +1,61 @@ +test: + variables: + - name: binlog_row_image + value: MINIMAL + - name: auto_increment_cache_size + value: 10000000 + system_config: + - name: enable_sql_audit + value: 'false' + - name: sleep + value: 3 + optimizer: sleep + - name: enable_early_lock_release + value: 'false' + optimizer: tenant + - name: syslog_level + value: 'ERROR' + - name: enable_perf_event + value: false + value_type: BOOL + - name: _enable_defensive_check + value: false + - name: _rowsets_enabled + value: 'false' + optimizer: tenant + - name: _enable_newsort + value: 'false' + - name: _trace_control_info + value: '' + optimizer: tenant + - name: _lcl_op_interval + value: 0ms + - name: writing_throttling_trigger_percentage + value: 100 + optimizer: tenant + - name: default_auto_increment_mode + value: 'NOORDER' + optimizer: tenant + - name: enable_monotonic_weak_read + value: 'false' + optimizer: tenant + - name: _enable_adaptive_compaction + value: 'false' + optimizer: tenant + - name: enable_record_trace_log + value: 'false' + - name: cpu_quota_concurrency + value: 2 + optimizer: tenant + - name: _ob_enable_prepared_statement + value: 'true' + - name: _pushdown_storage_level + value: 0 + optimizer: tenant + - name: ignore_replay_checksum_error + value: 'true' + - name: weak_read_version_refresh_interval + value: 2s + - name: freeze_trigger_percentage + value: 40 + optimizer: tenant \ No newline at end of file diff --git a/optimize/oceanbase/4.1.0/tpcc.yaml b/optimize/oceanbase/4.1.0/tpcc.yaml new file mode 100644 index 0000000..b72d27a --- /dev/null +++ b/optimize/oceanbase/4.1.0/tpcc.yaml @@ -0,0 +1,63 @@ +build: + variables: + - name: binlog_row_image + value: MINIMAL + - name: auto_increment_cache_size + value: 10000000 + system_config: + - name: enable_sql_audit + value: 'false' + - name: sleep + value: 5 + optimizer: sleep + - name: enable_early_lock_release + value: 'false' + optimizer: tenant + - name: syslog_level + value: 'ERROR' + - name: enable_perf_event + value: false + value_type: BOOL + - name: _enable_defensive_check + value: false + - name: _rowsets_enabled + value: false + optimizer: tenant + - name: _enable_newsort + value: false + - name: _trace_control_info + value: '' + optimizer: tenant + - name: _lcl_op_interval + value: 0ms + - name: default_auto_increment_mode + value: 'NOORDER' + optimizer: tenant + - name: enable_monotonic_weak_read + value: 'false' + optimizer: tenant + - name: _enable_adaptive_compaction + value: 'false' + optimizer: tenant + - name: enable_record_trace_log + value: 'false' + - name: cpu_quota_concurrency + value: 2 + optimizer: tenant + - name: _ob_enable_prepared_statement + value: 'true' + - name: _pushdown_storage_level + value: 0 + optimizer: tenant + - name: ignore_replay_checksum_error + value: 'true' + - name: weak_read_version_refresh_interval + value: 2s + - name: freeze_trigger_percentage + value: 40 + optimizer: tenant +test: + system_config: + - name: writing_throttling_trigger_percentage + value: 100 + optimizer: tenant \ No newline at end of file diff --git a/optimize/optimize_parser/0.1/optimize_parser.py b/optimize/optimize_parser/0.1/optimize_parser.py index 3022d05..0ecb4fc 100644 --- a/optimize/optimize_parser/0.1/optimize_parser.py +++ b/optimize/optimize_parser/0.1/optimize_parser.py @@ -148,7 +148,7 @@ def _format(self): else: r = re.match('^(\d+)(\w)B?$', self._origin.upper()) n, u = r.groups() - unit = self.UNITS.get(u.upper()) + unit = self.UNITS.get(u.upper()) if unit: self._value = int(n) * unit else: @@ -540,14 +540,8 @@ def query(self, cursor, stdio=None, *args, **kwargs): if not self.query_sql: stdio.verbose('no query sql') return - try: - sql = self.query_sql.format(**kwargs) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - ret = cursor.fetchone() - return ret - except: - stdio.exception("") + sql = self.query_sql.format(**kwargs) + return cursor.fetchone(sql) @property def modify_sql(self): @@ -559,20 +553,13 @@ def modify(self, cursor, stdio=None, value=None, *args, **kwargs): if not self.modify_sql: stdio.verbose('no modify sql') return - try: - sql = self.modify_sql.format(**kwargs) - cursor_args = None - if value is not None: - cursor_args = (value, ) - stdio.verbose('execute sql: %s' % (sql % cursor_args)) - else: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql, cursor_args) - cursor.fetchone() - return True - except: - stdio.exception("") + sql = self.modify_sql.format(**kwargs) + cursor_args = None + if value is not None: + cursor_args = (value, ) + if cursor.fetchone(sql, cursor_args) is False: return False + return True class OptimizeParser(SafeStdio): diff --git a/plugins/commands/0.1/command_template.yaml b/plugins/commands/0.1/command_template.yaml index 0b3f8d5..c413c9b 100644 --- a/plugins/commands/0.1/command_template.yaml +++ b/plugins/commands/0.1/command_template.yaml @@ -2,14 +2,14 @@ variables: ssh: - name: host config_key: host - components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce'] + components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce', 'obagent', 'ocp-express', 'grafana', 'prometheus'] - name: user config_key: username - components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce'] + components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce', 'obagent', 'ocp-express', 'grafana', 'prometheus'] server: - name: home_path config_key: home_path - components: ['oceanbase', 'oceanbase-ce', 'obproxy', 'obproxy-ce'] + components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce', 'obagent', 'ocp-express', 'grafana', 'prometheus'] - name: mysql_port config_key: mysql_port components: ['oceanbase', 'oceanbase-ce'] @@ -31,8 +31,13 @@ wrappers: commands: - name: ssh - components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce'] - command: "cd {home_path}/log;bash --login" + components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce', 'obagent', 'ocp-express', 'grafana', 'prometheus'] + command: "cd {home_path}/log; echo 'ssh {user}@{host}'; bash --login" + wrapper: "ssh" + interactive: true + - name: log + components: ['oceanbase', 'obproxy', 'oceanbase-ce', 'obproxy-ce', 'obagent', 'ocp-express', 'grafana', 'prometheus'] + command: "cd {home_path}/log; echo 'ssh {user}@{host}'; ls -l; bash --login" wrapper: "ssh" interactive: true - name: less diff --git a/plugins/general/0.1/install_repo.py b/plugins/general/0.1/install_repo.py index 51ce519..d173005 100644 --- a/plugins/general/0.1/install_repo.py +++ b/plugins/general/0.1/install_repo.py @@ -93,7 +93,7 @@ def install_to_home_path(): stdio.verbose('%s %s install check' % (server, install_repository)) try: yaml_loader = YamlLoader(stdio=stdio) - data = yaml_loader.load(remote_repository_data) + data = yaml_loader.loads(remote_repository_data) if not data: stdio.verbose('%s %s need to be installed ' % (server, install_repository)) elif data == install_repository: diff --git a/plugins/grafana/7.5.17/display.py b/plugins/grafana/7.5.17/display.py index 425de37..4a7373a 100644 --- a/plugins/grafana/7.5.17/display.py +++ b/plugins/grafana/7.5.17/display.py @@ -19,7 +19,8 @@ from __future__ import absolute_import, division, print_function -import socket +from tool import NetUtil + stdio = None @@ -36,16 +37,16 @@ def display(plugin_context, cursor, *args, **kwargs): ip = server.ip if ip == '127.0.0.1': - hostname = socket.gethostname() - ip = socket.gethostbyname(hostname) + ip = NetUtil.get_host_ip() user = api_cursor.user protocol = api_cursor.protocol if 'prometheus' in cluster_config.depends: url = '%s://%s:%s/d/oceanbase' % (protocol, ip, server_config['port']) else: url = '%s://%s:%s' % (protocol, ip, server_config['port']) - stdio.verbose('type: %s'% type(server.ip)) results.append({ + 'ip': ip, + 'port': server_config['port'], 'user': user, 'password': api_cursor.password, 'url': url, @@ -53,4 +54,8 @@ def display(plugin_context, cursor, *args, **kwargs): }) stdio.print_list(results, [ 'url', 'user', 'password', 'status'], lambda x: [x['url'], x['user'], x['password'], x['status']], title='grafana') - return plugin_context.return_true() \ No newline at end of file + active_result = [r for r in results if r['status'] == 'active'] + info_dict = active_result[0] if len(active_result) > 0 else None + if info_dict is not None: + info_dict['type'] = 'web' + return plugin_context.return_true(info=info_dict) diff --git a/plugins/grafana/7.5.17/generate_config.py b/plugins/grafana/7.5.17/generate_config.py index 46d4175..22b5a14 100644 --- a/plugins/grafana/7.5.17/generate_config.py +++ b/plugins/grafana/7.5.17/generate_config.py @@ -21,35 +21,34 @@ from __future__ import absolute_import, division, print_function -def generate_config(plugin_context, deploy_config, auto_depend=False, *args, **kwargs): +def generate_config(plugin_context, auto_depend=False, generate_check=True, return_generate_keys=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=['login_password']) + cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio success = True have_depend = False depend = 'prometheus' - server_depends = {} + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) stdio.start_loading('Generate grafana configuration') global_config = cluster_config.get_original_global_conf() login_password = global_config.get('login_password') if login_password: - login_password = str(login_password) - if len(login_password) < 5: - stdio.error("Grafana : the length of configuration 'login_password' cannot be less than 5") - success = False - elif login_password == "admin": - stdio.error("Grafana : configuration 'login_password' in configuration file should not be 'admin'") - success = False + if generate_check: + login_password = str(login_password) + if len(login_password) < 5: + stdio.error("Grafana : the length of configuration 'login_password' cannot be less than 5") + success = False + elif login_password == "admin": + stdio.error("Grafana : configuration 'login_password' in configuration file should not be 'admin'") + success = False else: + generate_configs['global']['login_password'] = 'oceanbase' cluster_config.update_global_conf('login_password', 'oceanbase', False) - for server in cluster_config.servers: - server_depends[server] = [] - server_config = cluster_config.get_server_conf(server) - if not server_config.get('home_path'): - stdio.error("%s grafana : missing configuration 'home_path' in configuration file" % server) - success = False - if not success: stdio.stop_loading('fail') return diff --git a/plugins/grafana/7.5.17/init.py b/plugins/grafana/7.5.17/init.py index 40ea1f2..febb2ba 100644 --- a/plugins/grafana/7.5.17/init.py +++ b/plugins/grafana/7.5.17/init.py @@ -32,7 +32,7 @@ def critical(*arg, **kwargs): stdio.error(*arg, **kwargs) -def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def init(plugin_context, *args, **kwargs): global stdio, force cluster_config = plugin_context.cluster_config clients = plugin_context.clients diff --git a/plugins/grafana/7.5.17/parameter.yaml b/plugins/grafana/7.5.17/parameter.yaml index a2c2986..9c08961 100644 --- a/plugins/grafana/7.5.17/parameter.yaml +++ b/plugins/grafana/7.5.17/parameter.yaml @@ -1,16 +1,22 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING need_redeploy: true description_en: working directory for grafana description_local: grafana工作目录 - name: login_password + name_local: 登录密码 require: true + essential: true type: STRING need_reload: true description_en: password for Grafana description_local: Grafana 登录密码 - name: data_dir + name_local: 数据目录 + essential: true type: STRING min_value: NULL max_value: NULL @@ -59,8 +65,10 @@ description_en: The ip address to bind to description_local: 要绑定的ip地址 - name: port - type: INT + name_local: HTTP 端口 require: true + essential: true + type: INT default: 3000 min_value: 1 max_value: 65535 diff --git a/plugins/grafana/7.5.17/reload.py b/plugins/grafana/7.5.17/reload.py index fecf1cd..6970449 100644 --- a/plugins/grafana/7.5.17/reload.py +++ b/plugins/grafana/7.5.17/reload.py @@ -27,7 +27,7 @@ yaml = YamlLoader() -def reload(plugin_context, cursor, repository_dir, new_cluster_config, deploy_name=None, *args, **kwargs): +def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config clients = plugin_context.clients diff --git a/plugins/grafana/7.5.17/restart.py b/plugins/grafana/7.5.17/restart.py index ddc8811..9606c54 100644 --- a/plugins/grafana/7.5.17/restart.py +++ b/plugins/grafana/7.5.17/restart.py @@ -26,11 +26,22 @@ class Restart(object): def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, bootstrap_plugin=None): self.local_home_path = local_home_path - self.plugin_context = plugin_context + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + self.components = plugin_context.components self.clients = plugin_context.clients self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context self.repository = repository self.start_plugin = start_plugin self.reload_plugin = reload_plugin @@ -44,11 +55,28 @@ def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, self.dbs = None self.cursors = None + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) + def connect(self, cluster_config): - self.stdio.verbose('Call %s for %s' % (self.connect_plugin, self.repository)) self.sub_io.start_loading('Connect to Grafana') - ret = self.connect_plugin(self.components, self.clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io) + ret = self.call_plugin(self.connect_plugin, cluster_config=cluster_config) if not ret: self.sub_io.stop_loading('fail') return False @@ -68,13 +96,10 @@ def restart(self): if self.new_cluster_config: if not self.connect(self.cluster_config): return False - self.stdio.verbose('Call %s for %s' % (self.reload_plugin, self.repository)) - if not self.reload_plugin(self.components, self.clients, self.cluster_config, [], {}, self.sub_io, cursor=self.cursors, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir): + if not self.call_plugin(self.reload_plugin, clients=clients, cursor=self.cursors, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir): return False - self.stdio.verbose('Call %s for %s' % (self.stop_plugin, self.repository)) - if not self.stop_plugin(self.components, clients, self.cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io): + if not self.call_plugin(self.stop_plugin): self.stdio.stop_loading('stop_loading', 'fail') return False @@ -82,7 +107,7 @@ def restart(self): self.stdio.verbose('use new clients') for server in self.cluster_config.servers: new_client = self.new_clients[server] - server_config = self.cluster_config.get_serves_conf(server) + server_config = self.cluster_config.get_server_conf(server) chown_cmd = 'sudo chown -R %s:' % new_client.config.username for key in ['home_path', 'data_dir', 'logs_dir', 'plugins_dir', 'provisioning_dir']: if key in server_config: @@ -94,26 +119,20 @@ def restart(self): clients = self.new_clients cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config - self.stdio.verbose('Call %s for %s' % (self.start_plugin, self.repository)) - need_bootstrap = self.bootstrap_plugin is not None - if not self.start_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io, local_home_path=self.local_home_path, - repository_dir=self.repository.repository_dir, need_bootstrap=need_bootstrap): + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, need_bootstrap=False): self.rollback() self.stdio.stop_loading('stop_loading', 'fail') return False if self.connect(cluster_config): - self.stdio.verbose('Call %s for %s' % (self.display_plugin, self.repository)) - ret = self.display_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io, cursor=self.cursors) + ret = self.call_plugin(self.display_plugin, clients=clients, cluster_config=cluster_config, cursor=self.cursors) return ret return False def rollback(self): if self.new_clients: self.stdio.start_loading('Rollback') - self.stop_plugin(self.components, self.new_clients, self.new_cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io) + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.new_clients, cluster_config=cluster_config) for server in self.cluster_config.servers: client = self.clients[server] new_client = self.new_clients[server] @@ -127,8 +146,9 @@ def rollback(self): def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, - repository, new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, *args, + new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, *args, **kwargs): + repository = kwargs.get('repository') task = Restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config, new_clients) call = task.rollback if rollback else task.restart diff --git a/plugins/grafana/7.5.17/start.py b/plugins/grafana/7.5.17/start.py index 5703040..5def012 100644 --- a/plugins/grafana/7.5.17/start.py +++ b/plugins/grafana/7.5.17/start.py @@ -74,7 +74,7 @@ def confirm_port(client, pid, port, stdio): return False -def start(plugin_context, local_home_path, repository_dir, need_bootstrap=False, repository_dir_map=None, *args, **kwargs): +def start(plugin_context, *args, **kwargs): def spear_dict(di_, con_s='.'): def prefix_dict(di_, prefix_s=''): @@ -207,6 +207,7 @@ def generate_provider_yaml(): cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio + options = plugin_context.options global _grafana_conf_from_prometheus _grafana_conf_from_prometheus = None need_bootstrap = True @@ -234,45 +235,61 @@ def generate_provider_yaml(): client = clients[server] server_config = cluster_config.get_server_conf(server) home_path = server_config['home_path'] + + remote_pid_path = os.path.join(home_path, 'run/grafana.pid') + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + servers_pid[server] = [remote_pid] + stdio.verbose('%s is runnning in %s, skip' % (server, remote_pid)) + continue - ini_dict = generate_ini() - fail_message = check_parameter(ini_dict) - if fail_message: - for msg in fail_message: - stdio.warn('%s: %s' %(server, msg)) - stdio.stop_loading('fail') - return False + config_flag = os.path.join(home_path, '.configured') + if getattr(options, 'without_parameter', False) and client.execute_command('ls %s' % config_flag): + use_parameter = False + else: + use_parameter = True - prometheus_server_config = {} - prometheus_server = [] - if 'prometheus' in cluster_config.depends: - prometheus_servers = cluster_config.get_depend_servers('prometheus') - prometheus_server.append(prometheus_servers[0]) - prometheus_server_config = cluster_config.get_depend_config('prometheus', prometheus_servers[0]) - - key_map = {'port': 'server.http_port', - 'domain': 'server.domain', - 'log_max_days': 'log.file.max_days', - 'temp_data_lifetime': 'paths.temp_data_lifetime'} - for key in key_map: - if key in server_config: - ini_dict[key_map[key]] = server_config[key] - - stdio.verbose('%s generate obd-grafana ini file' % server) - if not generate_ini_file(home_path, ini_dict): - stdio.verbose('%s obd-grafana ini file generate failed' % server) - stdio.stop_loading('fail') - return False - stdio.verbose('%s generate datasources yaml' % server) - if not generate_datasource_yaml(prometheus_server, prometheus_server_config): - stdio.verbose('%s grafana datasources yaml generate failed' % server) - stdio.stop_loading('fail') - return False - stdio.verbose('%s generate providers yaml' % server) - if not generate_provider_yaml(): - stdio.verbose('%s grafana providers yaml generate failed' % server) - stdio.stop_loading('fail') - return False + if use_parameter: + ini_dict = generate_ini() + fail_message = check_parameter(ini_dict) + if fail_message: + for msg in fail_message: + stdio.warn('%s: %s' %(server, msg)) + stdio.stop_loading('fail') + return False + + prometheus_server_config = {} + prometheus_server = [] + if 'prometheus' in cluster_config.depends: + prometheus_servers = cluster_config.get_depend_servers('prometheus') + prometheus_server.append(prometheus_servers[0]) + prometheus_server_config = cluster_config.get_depend_config('prometheus', prometheus_servers[0]) + + key_map = {'port': 'server.http_port', + 'domain': 'server.domain', + 'log_max_days': 'log.file.max_days', + 'temp_data_lifetime': 'paths.temp_data_lifetime'} + for key in key_map: + if key in server_config: + ini_dict[key_map[key]] = server_config[key] + + stdio.verbose('%s generate obd-grafana ini file' % server) + if not generate_ini_file(home_path, ini_dict): + stdio.verbose('%s obd-grafana ini file generate failed' % server) + stdio.stop_loading('fail') + return False + stdio.verbose('%s generate datasources yaml' % server) + if not generate_datasource_yaml(prometheus_server, prometheus_server_config): + stdio.verbose('%s grafana datasources yaml generate failed' % server) + stdio.stop_loading('fail') + return False + stdio.verbose('%s generate providers yaml' % server) + if not generate_provider_yaml(): + stdio.verbose('%s grafana providers yaml generate failed' % server) + stdio.stop_loading('fail') + return False + client.execute_command('touch %s' % config_flag) ini_path = os.path.join(server_config["home_path"], 'conf/obd-grafana.ini') grafana_pid_path = os.path.join(server_config["home_path"], 'run/grafana.pid') diff --git a/plugins/grafana/7.5.17/start_check.py b/plugins/grafana/7.5.17/start_check.py index dfc8c0b..0b01aa1 100644 --- a/plugins/grafana/7.5.17/start_check.py +++ b/plugins/grafana/7.5.17/start_check.py @@ -19,8 +19,10 @@ from __future__ import absolute_import, division, print_function -from _errno import EC_CONFIG_CONFLICT_PORT import os +from tool import OrderedDict +import _errno as err + stdio = None success = True @@ -35,38 +37,128 @@ def get_port_socket_inode(client, port): return res.stdout.strip().split('\n') -def start_check(plugin_context, strict_check=False, *args, **kwargs): - - def critical(*arg, **kwargs): +def start_check(plugin_context, init_check_status=False, work_dir_check=False, work_dir_empty_check=True, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) + check_fail(item, error, suggests) + stdio.error(error) - global stdio + global stdio, success + success = True cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio servers_port = {} - stdio.start_loading('Check before start grafana') + servers_dirs = {} + servers_check_dirs = {} + check_status = {} + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + 'password': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + stdio.start_loading('Check before start grafana') for server in cluster_config.servers: - server_config = cluster_config.get_server_conf(server) + server_config = cluster_config.get_server_conf_with_default(server) home_path = server_config['home_path'] ip = server.ip client = clients[server] + if not precheck: + remote_pid_path = os.path.join(home_path, 'run/grafana.pid') + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + original_server_conf = cluster_config.get_server_conf(server) + + if not server_config.get('data_dir'): + server_config['data_dir'] = '%s/data' % home_path + if not server_config.get('logs_dir'): + server_config['logs_dir'] = '%s/log' % server_config['data_dir'] + if not server_config.get('plugins_dir'): + server_config['plugins_dir'] = '%s/plugins' % server_config['data_dir'] + if not server_config.get('provisioning_dir'): + server_config['provisioning_dir'] = '%s/conf/provisioning' % home_path + + keys = ['home_path', 'data_dir','logs_dir', 'plugins_dir', 'provisioning_dir'] + for key in keys: + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + if key not in original_server_conf: + continue + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False if server_config['login_password'] == 'admin': - critical("%s grafana admin password should not be 'admin'" % server) - continue - if len(str(server_config['login_password'])) < 5: - critical("%s grafana admin password length should not be less than 5" % server) - continue - - remote_pid_path = os.path.join(home_path, 'run/grafana.pid') - remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s' % remote_pid): - continue + critical('password', err.EC_GRAFANA_DEFAULT_PWD.format(server=server), [err.SUG_GRAFANA_PWD.format()]) + elif len(str(server_config['login_password'])) < 5: + critical('password', err.EC_GRAFANA_PWD_LESS_5.format(server=server), [err.SUG_GRAFANA_PWD.format()]) + else: + check_pass('password') + if ip not in servers_port: servers_port[ip] = {} ports = servers_port[ip] @@ -74,15 +166,28 @@ def critical(*arg, **kwargs): stdio.verbose('%s port check' % server) port = server_config['port'] if port in ports: - critical(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], - key=ports[port]['key'])) - continue - ports[port] = { - 'server': server, - 'key': port - } - if get_port_socket_inode(client, port): - critical('%s:%s port is already used' % (ip, port)) + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) + else: + ports[port] = { + 'server': server, + 'key': port + } + if get_port_socket_inode(client, port): + critical( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + continue + check_pass('port') + + for server in cluster_config.servers: + wait_2_pass() + if success: stdio.stop_loading('succeed') plugin_context.return_true() diff --git a/plugins/grafana/7.5.17/stop.py b/plugins/grafana/7.5.17/stop.py index 193ec67..9c3f9a7 100644 --- a/plugins/grafana/7.5.17/stop.py +++ b/plugins/grafana/7.5.17/stop.py @@ -69,7 +69,7 @@ def stop(plugin_context, *args, **kwargs): } else: stdio.verbose('failed to stop grafana[pid:{}] in {}, permission deny'.format(grafana_pid, server)) - success = True + success = False else: stdio.verbose('{} grafana is not running'.format(server)) if not success: diff --git a/plugins/mysqltest/3.1.0/check_opt.py b/plugins/mysqltest/3.1.0/check_opt.py index cd33d15..5bb0561 100644 --- a/plugins/mysqltest/3.1.0/check_opt.py +++ b/plugins/mysqltest/3.1.0/check_opt.py @@ -25,7 +25,8 @@ from ssh import LocalClient -def check_opt(plugin_context, opt, *args, **kwargs): +def check_opt(plugin_context, env, *args, **kwargs): + opt = env stdio = plugin_context.stdio server = opt['test_server'] obclient_bin = opt['obclient_bin'] @@ -96,15 +97,12 @@ def check_opt(plugin_context, opt, *args, **kwargs): stdio.verbose('load engine from config') opt['_enable_static_typing_engine'] = global_config['_enable_static_typing_engine'] else: - try: - sql = "select value from oceanbase.__all_virtual_sys_parameter_stat where name like '_enable_static_typing_engine';" - stdio.verbose('execute sql: {}'.format(sql)) - cursor.execute(sql) - ret = cursor.fetchone() - stdio.verbose('query engine ret: {}'.format(ret)) - if ret: - opt['_enable_static_typing_engine'] = ret.get('value') - except: - stdio.exception('') + sql = "select value from oceanbase.__all_virtual_sys_parameter_stat where name like '_enable_static_typing_engine';" + ret = cursor.fetchone(sql) + if ret is False: + return + stdio.verbose('query engine ret: {}'.format(ret)) + if ret: + opt['_enable_static_typing_engine'] = ret.get('value') stdio.verbose('_enable_static_typing_engine: {}'.format(opt['_enable_static_typing_engine'])) return plugin_context.return_true() diff --git a/plugins/mysqltest/3.1.0/check_test.py b/plugins/mysqltest/3.1.0/check_test.py index 555adfb..4501249 100644 --- a/plugins/mysqltest/3.1.0/check_test.py +++ b/plugins/mysqltest/3.1.0/check_test.py @@ -87,7 +87,8 @@ def test_name(test_file): return base_name -def check_test(plugin_context, opt, *args, **kwargs): +def check_test(plugin_context, env, *args, **kwargs): + opt = env stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config tags = [] @@ -151,15 +152,12 @@ def check_test(plugin_context, opt, *args, **kwargs): else: test_zone = cluster_config.get_server_conf(opt['test_server'])['zone'] query = 'select zone, count(*) as a from oceanbase.__all_virtual_zone_stat group by region order by a desc limit 1' - try: - stdio.verbose('execute sql: {}'.format(query)) - cursor = opt['cursor'] - cursor.execute(query) - ret = cursor.fetchone() - except: - msg = 'execute sql exception: %s' % query - raise Exception(msg) - primary_zone = ret.get('zone', '') + cursor = opt['cursor'] + ret = cursor.fetchone(query) + if ret is False: + return + if ret: + primary_zone = ret.get('zone', '') if test_zone != primary_zone: opt["filter"] = 'slave' if regress_suites: diff --git a/plugins/mysqltest/3.1.0/init.py b/plugins/mysqltest/3.1.0/init.py index 0ab0593..6ddfe8e 100644 --- a/plugins/mysqltest/3.1.0/init.py +++ b/plugins/mysqltest/3.1.0/init.py @@ -43,16 +43,14 @@ def parse_size(size): def get_memory_limit(cursor, client): try: - cursor.execute('show parameters where name = \'memory_limit\'') - memory_limit = cursor.fetchone() + memory_limit = cursor.fetchone('show parameters where name = \'memory_limit\'') if memory_limit and 'value' in memory_limit and memory_limit['value']: return parse_size(memory_limit['value']) ret = client.execute_command('free -b') if ret: ret = client.execute_command("cat /proc/meminfo | grep 'MemTotal:' | awk -F' ' '{print $2}'") total_memory = int(ret.stdout) * 1024 - cursor.execute('show parameters where name = \'memory_limit_percentage\'') - memory_limit_percentage = cursor.fetchone() + memory_limit_percentage = cursor.fetchone('show parameters where name = \'memory_limit_percentage\'') if memory_limit_percentage and 'value' in memory_limit_percentage and memory_limit_percentage['value']: total_memory = total_memory * memory_limit_percentage['value'] / 100 return total_memory @@ -65,8 +63,7 @@ def init(plugin_context, env, *args, **kwargs): def get_root_server(cursor): while True: try: - cursor.execute('select * from oceanbase.__all_server where status = \'active\' and with_rootserver=1') - return cursor.fetchone() + return cursor.fetchone('select * from oceanbase.__all_server where status = \'active\' and with_rootserver=1', raise_exception=True) except: if load_snap: time.sleep(0.1) diff --git a/plugins/mysqltest/3.1.0/run_test.py b/plugins/mysqltest/3.1.0/run_test.py index 6a2777c..c407f06 100644 --- a/plugins/mysqltest/3.1.0/run_test.py +++ b/plugins/mysqltest/3.1.0/run_test.py @@ -205,7 +205,6 @@ def return_true(**kw): need_reboot = env.get('need_reboot', False) collect_all = env.get('collect_all', False) collect_log = False - total_test_count = len(test_set) while index < total_test_count: test = test_set[index] @@ -369,13 +368,8 @@ def return_true(**kw): opt['connector'] = 'ob' if opt['_enable_static_typing_engine'] is not None: - ret = None - try: - sql = "select value from oceanbase.__all_virtual_sys_parameter_stat where name like '_enable_static_typing_engine';" - cursor.execute(sql) - ret = cursor.fetchone() - except: - pass + sql = "select value from oceanbase.__all_virtual_sys_parameter_stat where name like '_enable_static_typing_engine';" + ret = cursor.fetchone(sql) if ret and str(ret.get('value')).lower() != str(opt['_enable_static_typing_engine']).lower(): LocalClient.execute_command('%s "alter system set _enable_static_typing_engine = %s;select sleep(2);"' % (exec_sql_cmd, opt['_enable_static_typing_engine']), stdio=stdio) diff --git a/plugins/mysqltest/4.0.0.0/check_test.py b/plugins/mysqltest/4.0.0.0/check_test.py index b87dd34..7f2ed99 100644 --- a/plugins/mysqltest/4.0.0.0/check_test.py +++ b/plugins/mysqltest/4.0.0.0/check_test.py @@ -89,7 +89,8 @@ def test_name(test_file): return base_name -def check_test(plugin_context, opt, *args, **kwargs): +def check_test(plugin_context, env, *args, **kwargs): + opt = env stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config tags = [] @@ -153,15 +154,12 @@ def check_test(plugin_context, opt, *args, **kwargs): else: test_zone = cluster_config.get_server_conf(opt['test_server'])['zone'] query = 'select zone, count(*) as a from oceanbase.DBA_OB_ZONES group by region order by a desc limit 1' - try: - stdio.verbose('execute sql: {}'.format(query)) - cursor = opt['cursor'] - cursor.execute(query) - ret = cursor.fetchone() - except: - msg = 'execute sql exception: %s' % query - raise Exception(msg) - primary_zone = ret.get('zone', '') + cursor = opt['cursor'] + ret = cursor.fetchone(query) + if ret is False: + return + if ret: + primary_zone = ret.get('zone', '') if test_zone != primary_zone: opt["filter"] = 'slave' if regress_suites: diff --git a/plugins/mysqltest/4.0.0.0/init.py b/plugins/mysqltest/4.0.0.0/init.py index ff25721..2b77011 100644 --- a/plugins/mysqltest/4.0.0.0/init.py +++ b/plugins/mysqltest/4.0.0.0/init.py @@ -45,8 +45,7 @@ def init(plugin_context, env, *args, **kwargs): def get_root_server(cursor): while True: try: - cursor.execute('select * from oceanbase.__all_server where status = \'active\' and with_rootserver=1') - return cursor.fetchone() + return cursor.fetchone('select * from oceanbase.__all_server where status = \'active\' and with_rootserver=1', raise_exception=True) except: if load_snap: time.sleep(0.1) diff --git a/plugins/obagent/0.1/display.py b/plugins/obagent/0.1/display.py index 1d973e8..f04f76d 100644 --- a/plugins/obagent/0.1/display.py +++ b/plugins/obagent/0.1/display.py @@ -19,7 +19,8 @@ from __future__ import absolute_import, division, print_function -import socket +from tool import NetUtil + def display(plugin_context, cursor, *args, **kwargs): stdio = plugin_context.stdio @@ -37,8 +38,7 @@ def display(plugin_context, cursor, *args, **kwargs): cmd = '''curl %s -H "Content-Type:application/json" -L "http://%s:%s/metrics/stat"''' % (auth, server.ip, config['server_port']) ip = server.ip if ip == '127.0.0.1': - hostname = socket.gethostname() - ip = socket.gethostbyname(hostname) + ip = NetUtil.get_host_ip() result.append({ 'ip': ip, 'status': 'active' if client.execute_command(cmd) else 'inactive', diff --git a/plugins/obagent/0.1/generate_config.py b/plugins/obagent/0.1/generate_config.py index 61c1bbf..e46b1a1 100644 --- a/plugins/obagent/0.1/generate_config.py +++ b/plugins/obagent/0.1/generate_config.py @@ -21,49 +21,44 @@ from __future__ import absolute_import, division, print_function -def generate_config(plugin_context, deploy_config, auto_depend=False, *args, **kwargs): +def generate_config(plugin_context, auto_depend=False, return_generate_keys=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=['ob_monitor_status']) + cluster_config = plugin_context.cluster_config - clients = plugin_context.clients stdio = plugin_context.stdio - success = True have_depend = False depends = ['oceanbase', 'oceanbase-ce'] server_depends = {} + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) stdio.start_loading('Generate obagent configuration') - for server in cluster_config.servers: - server_depends[server] = [] - server_config = cluster_config.get_server_conf(server) - if not server_config.get('home_path'): - stdio.error("obagent %s: missing configuration 'home_path' in configuration file" % server) - success = False - continue - if not success: - stdio.stop_loading('fail') - return - for comp in cluster_config.depends: if comp in depends: have_depend = True for server in cluster_config.servers: + server_depends[server] = [] obs_config = cluster_config.get_depend_config(comp, server) if obs_config is not None: server_depends[server].append(comp) - + if have_depend: - server_num = len(cluster_config.servers) for server in cluster_config.servers: for comp in depends: if comp in server_depends[server]: break else: cluster_config.update_server_conf(server, 'ob_monitor_status', 'inactive', False) + generate_configs[server]['ob_monitor_status'] = 'inactive' else: cluster_config.update_global_conf('ob_monitor_status', 'inactive', False) + generate_configs['global']['ob_monitor_status'] = 'inactive' if auto_depend: for depend in depends: if cluster_config.add_depend_component(depend): cluster_config.update_global_conf('ob_monitor_status', 'active', False) + generate_configs['global']['ob_monitor_status'] = 'active' break stdio.stop_loading('succeed') diff --git a/plugins/obagent/0.1/init.py b/plugins/obagent/0.1/init.py index 14456ec..379dd1e 100644 --- a/plugins/obagent/0.1/init.py +++ b/plugins/obagent/0.1/init.py @@ -23,7 +23,7 @@ from _errno import EC_FAIL_TO_INIT_PATH, InitDirFailedErrorMessage -def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def init(plugin_context, *args, **kwargs): cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio @@ -35,8 +35,6 @@ def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): server_config = cluster_config.get_server_conf(server) client = clients[server] home_path = server_config['home_path'] - remote_home_path = client.execute_command('echo ${OBD_HOME:-"$HOME"}/.obd').stdout.strip() - remote_repository_dir = repository_dir.replace(local_home_path, remote_home_path) stdio.verbose('%s init cluster work home', server) need_clean = force if clean and not force: diff --git a/plugins/obagent/0.1/parameter.yaml b/plugins/obagent/0.1/parameter.yaml index ddd9806..eb35d07 100644 --- a/plugins/obagent/0.1/parameter.yaml +++ b/plugins/obagent/0.1/parameter.yaml @@ -1,11 +1,15 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING - need_restart: true + need_redeploy: true description_en: working directory for obagent description_local: Obagent工作目录 - name: server_port + name_local: 服务端口 require: true + essential: true type: INT default: 8088 min_value: 1025 @@ -14,7 +18,9 @@ description_en: port number for pulling metrics and management description_local: 提供拉取 metrics 和管理的端口 - name: pprof_port + name_local: 调试端口 require: true + essential: true type: INT default: 8089 min_value: 1025 @@ -41,7 +47,9 @@ description_en: log path description_local: 日志路径 - name: crypto_method + name_local: 加密方式 require: true + essential: true type: STRING default: plain min_value: NULL @@ -51,6 +59,8 @@ description_local: 加密方式,仅支持 aes 和 plain - name: crypto_path require: true + name_local: 秘钥路径 + essential: true type: STRING default: conf/.config_secret.key min_value: NULL @@ -104,7 +114,9 @@ description_en: whether to enable log compression description_local: 是否开启日志压缩 - name: http_basic_auth_user + name_local: 用户名 require: true + essential: true type: STRING default: admin min_value: NULL @@ -113,7 +125,9 @@ description_en: username for HTTP authentication description_local: HTTP 服务认证用户名 - name: http_basic_auth_password + name_local: 密码 require: false + essential: true type: STRING default: root min_value: NULL @@ -122,7 +136,9 @@ description_en: password for HTTP authentication description_local: HTTP 服务认证密码 - name: pprof_basic_auth_user + name_local: 调试用户名 require: true + essential: true type: STRING default: admin min_value: NULL @@ -131,7 +147,9 @@ description_en: username for debug service description_local: debug 接口认证用户名 - name: pprof_basic_auth_password + name_local: 调试密码 require: false + essential: true type: STRING default: root min_value: NULL @@ -203,7 +221,9 @@ description_en: zone name for your observer description_local: observer 所在的 zone 名字 - name: ob_monitor_status + name_local: OceanBase 指标监控采集 require: true + essential: true type: STRING default: active min_value: NULL @@ -212,7 +232,9 @@ description_en: monitor status for OceanBase Database. Active is to enable. Inactive is to disable. description_local: OceanBase 监控指标采集状态,active 表示开启,inactive 表示关闭 - name: host_monitor_status + name_local: 主机指标监控采集 require: true + essential: true type: STRING default: active min_value: NULL @@ -221,7 +243,9 @@ description_en: monitor status for your host. Active is to enable. Inactive is to disable. description_local: 主机监控指标采集状态, active 表示开启, inactive 表示关闭 - name: disable_http_basic_auth + name_local: 禁用 HTTP 服务的basic auth 认证 require: true + essential: true type: BOOL default: false min_value: NULL @@ -230,7 +254,9 @@ description_en: whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. description_local: 是否禁用 HTTP 服务的basic auth 认证,true 表示禁用,false 表示不禁用 - name: disable_pprof_basic_auth + name_local: 禁用 debug 接口的basic auth 认证 require: true + essential: true type: BOOL default: false min_value: NULL diff --git a/plugins/obagent/0.1/reload.py b/plugins/obagent/0.1/reload.py index 0034d3c..fa3f9c1 100644 --- a/plugins/obagent/0.1/reload.py +++ b/plugins/obagent/0.1/reload.py @@ -29,11 +29,17 @@ from _errno import * -def reload(plugin_context, repository_dir, new_cluster_config, *args, **kwargs): +def reload(plugin_context, new_cluster_config, *args, **kwargs): stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config clients = plugin_context.clients servers = cluster_config.servers + + for repository in plugin_context.repositories: + if repository.name == cluster_config.name: + break + repository_dir = repository.repository_dir + yaml = YamlLoader(stdio) config_map = { "monitor_password": "root_password", diff --git a/plugins/obagent/0.1/restart.py b/plugins/obagent/0.1/restart.py index 79cbbe5..fc6815f 100644 --- a/plugins/obagent/0.1/restart.py +++ b/plugins/obagent/0.1/restart.py @@ -28,11 +28,22 @@ class Restart(object): def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, deploy_name=None): self.local_home_path = local_home_path - self.plugin_context = plugin_context + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + self.components = plugin_context.components self.clients = plugin_context.clients self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context self.repository = repository self.start_plugin = start_plugin self.reload_plugin = reload_plugin @@ -49,11 +60,29 @@ def dir_read_check(self, client, path): dirpath, name = os.path.split(path) return self.dir_read_check(client, dirpath) and client.execute_command('sudo chmod +1 %s' % path) return True + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) def restart(self): clients = self.clients - self.stdio.verbose('Call %s for %s' % (self.stop_plugin, self.repository)) - if not self.stop_plugin(self.components, clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io): + if not self.call_plugin(self.stop_plugin, clients=clients): self.stdio.stop_loading('stop_loading', 'fail') return False @@ -70,16 +99,16 @@ def restart(self): clients = self.new_clients cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config - self.stdio.verbose('Call %s for %s' % (self.start_plugin, self.repository)) - if not self.start_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, local_home_path=self.local_home_path, repository_dir=self.repository.repository_dir, deploy_name=self.deploy_name): + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, repository=self.repository): self.rollback() self.stdio.stop_loading('stop_loading', 'fail') return False - return self.display_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, cursor=None) + return self.call_plugin(self.display_plugin, clients=clients, cluster_config=cluster_config, cursor=None) def rollback(self): if self.new_clients: - self.stop_plugin(self.components, self.new_clients, self.new_cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.new_clients, cluster_config=cluster_config) for server in self.cluster_config.servers: client = self.clients[server] new_client = self.new_clients[server] @@ -88,7 +117,9 @@ def rollback(self): new_client.execute_command('sudo chown -R %s: %s' % (client.config.username, home_path)) -def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, rollback=False, deploy_name=None, *args, **kwargs): +def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, new_cluster_config=None, new_clients=None, rollback=False, *args, **kwargs): + repository = kwargs.get('repository') + deploy_name = plugin_context.deploy_name task = Restart(plugin_context=plugin_context, local_home_path=local_home_path, start_plugin=start_plugin, reload_plugin=reload_plugin, stop_plugin=stop_plugin, connect_plugin=connect_plugin, display_plugin=display_plugin, repository=repository, new_cluster_config=new_cluster_config, diff --git a/plugins/obagent/0.1/start.py b/plugins/obagent/0.1/start.py index ede6e5f..ac4e42b 100644 --- a/plugins/obagent/0.1/start.py +++ b/plugins/obagent/0.1/start.py @@ -108,7 +108,7 @@ def encrypt(key, data): def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -137,12 +137,13 @@ def generate_aes_b64_key(): return base64.b64encode(key.encode('utf-8')) -def start(plugin_context, local_home_path, repository_dir, deploy_name=None, *args, **kwargs): +def start(plugin_context, *args, **kwargs): global stdio cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio options = plugin_context.options + deploy_name = plugin_context.deploy_name config_files = {} pid_path = {} targets = [] @@ -157,6 +158,11 @@ def start(plugin_context, local_home_path, repository_dir, deploy_name=None, *ar "zone_name": "zone", } + for repository in plugin_context.repositories: + if repository.name == cluster_config.name: + break + repository_dir = repository.repository_dir + stdio.start_loading('Start obagent') for server in cluster_config.servers: client = clients[server] @@ -286,21 +292,28 @@ def start(plugin_context, local_home_path, repository_dir, deploy_name=None, *ar stdio.start_loading('obagent program health check') time.sleep(1) failed = [] - fail_time = 0 - for server in cluster_config.servers: - client = clients[server] - server_config = cluster_config.get_server_conf(server) - stdio.verbose('%s program health check' % server) - pid = client.execute_command("cat %s" % pid_path[server]).stdout.strip() - if pid: - if confirm_port(client, pid, int(server_config["server_port"])): - stdio.verbose('%s obagent[pid: %s] started', server, pid) - client.execute_command('echo %s > %s' % (pid, pid_path[server])) + servers = cluster_config.servers + count = 20 + while servers and count: + count -= 1 + tmp_servers = [] + for server in servers: + client = clients[server] + server_config = cluster_config.get_server_conf(server) + stdio.verbose('%s program health check' % server) + pid = client.execute_command("cat %s" % pid_path[server]).stdout.strip() + if pid: + if confirm_port(client, pid, int(server_config["server_port"])): + stdio.verbose('%s obagent[pid: %s] started', server, pid) + elif count: + tmp_servers.append(server) + else: + failed.append('failed to start %s obagent' % server) else: - fail_time += 1 - else: - failed.append('failed to start %s obagent' % server) - + failed.append('failed to start %s obagent' % server) + servers = tmp_servers + if servers and count: + time.sleep(1) if failed: stdio.stop_loading('fail') for msg in failed: diff --git a/plugins/obagent/0.1/start_check.py b/plugins/obagent/0.1/start_check.py index d4d755c..ca62745 100644 --- a/plugins/obagent/0.1/start_check.py +++ b/plugins/obagent/0.1/start_check.py @@ -20,7 +20,8 @@ from __future__ import absolute_import, division, print_function -from _errno import EC_CONFIG_CONFLICT_PORT +import os +import _errno as err stdio = None @@ -29,7 +30,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -37,35 +38,112 @@ def get_port_socket_inode(client, port): return res.stdout.strip().split('\n') -def start_check(plugin_context, strict_check=False, *args, **kwargs): - def alert(*arg, **kwargs): +def start_check(plugin_context, init_check_status=False, strict_check=False, work_dir_check=False, work_dir_empty_check=True, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def alert(item, error, suggests=[]): global success if strict_check: success = False - stdio.error(*arg, **kwargs) + check_fail(item, error, suggests) + stdio.error(error) else: - stdio.warn(*arg, **kwargs) - def critical(*arg, **kwargs): + stdio.warn(error) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) - global stdio + check_fail(item, error, suggests) + stdio.error(error) + + global stdio, success + success = True cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio servers_port = {} + servers_dirs = {} + servers_check_dirs = {} + check_status = {} + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + stdio.start_loading('Check before start obagent') for server in cluster_config.servers: ip = server.ip client = clients[server] server_config = cluster_config.get_server_conf(server) - port = int(server_config["server_port"]) - prometheus_port = int(server_config["pprof_port"]) - remote_pid_path = "%s/run/obagent-%s-%s.pid" % (server_config['home_path'], server.ip, server_config["server_port"]) - remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s' % remote_pid): - continue + if not precheck: + remote_pid_path = "%s/run/obagent-%s-%s.pid" % (server_config['home_path'], server.ip, server_config["server_port"]) + remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + key = 'home_path' + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False if ip not in servers_port: servers_port[ip] = {} @@ -76,14 +154,25 @@ def critical(*arg, **kwargs): port = int(server_config[key]) alert_f = alert if key == 'pprof_port' else critical if port in ports: - alert_f(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key'])) + alert_f( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) continue ports[port] = { 'server': server, 'key': key } if get_port_socket_inode(client, port): - alert_f('%s:%s port is already used' % (ip, port)) + alert_f( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + + for server in cluster_config.servers: + wait_2_pass() if success: stdio.stop_loading('succeed') diff --git a/plugins/obagent/0.1/stop.py b/plugins/obagent/0.1/stop.py index 3f4fd9f..8f520d9 100644 --- a/plugins/obagent/0.1/stop.py +++ b/plugins/obagent/0.1/stop.py @@ -28,7 +28,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) inode = res.stdout.strip() if not res or not inode: diff --git a/plugins/obagent/0.1/upgrade.py b/plugins/obagent/0.1/upgrade.py index ded202c..17bf275 100644 --- a/plugins/obagent/0.1/upgrade.py +++ b/plugins/obagent/0.1/upgrade.py @@ -20,14 +20,26 @@ from __future__ import absolute_import, division, print_function +import os -def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, *args, **kwargs): + +def call_plugin(plugin, plugin_context, repositories, *args, **kwargs): + namespace = plugin_context.namespace + namespaces = plugin_context.namespaces + deploy_name = plugin_context.deploy_name components = plugin_context.components clients = plugin_context.clients cluster_config = plugin_context.cluster_config - cmd = plugin_context.cmd + cmds = plugin_context.cmds options = plugin_context.options stdio = plugin_context.stdio + return plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, + stdio, *args, **kwargs) + + +def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, *args, **kwargs): + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients upgrade_ctx = kwargs.get('upgrade_ctx') local_home_path = kwargs.get('local_home_path') @@ -44,14 +56,14 @@ def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, *args, display_plugin = search_py_script_plugin([dest_repository], 'display')[dest_repository] apply_param_plugin(cur_repository) - if not stop_plugin(components, clients, cluster_config, cmd, options, stdio, *args, **kwargs): + if not call_plugin(stop_plugin, plugin_context, [cur_repository], *args, **kwargs): return apply_param_plugin(dest_repository) - if not start_plugin(components, clients, cluster_config, cmd, options, stdio, *args, **kwargs): - return - - ret = connect_plugin(components, clients, cluster_config, cmd, options, stdio, *args, **kwargs) - if ret and display_plugin(components, clients, cluster_config, cmd, options, stdio, ret.get_return('cursor'), *args, **kwargs): + if not call_plugin(start_plugin, plugin_context, [dest_repository], *args, **kwargs): + return + + ret = call_plugin(connect_plugin, plugin_context, [dest_repository], *args, **kwargs) + if ret and call_plugin(display_plugin, plugin_context, [dest_repository], ret.get_return('cursor'), *args, **kwargs): upgrade_ctx['index'] = len(upgrade_repositories) return plugin_context.return_true() diff --git a/plugins/obagent/1.1.0/parameter.yaml b/plugins/obagent/1.1.0/parameter.yaml index bf88977..c54944a 100644 --- a/plugins/obagent/1.1.0/parameter.yaml +++ b/plugins/obagent/1.1.0/parameter.yaml @@ -1,11 +1,15 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING - need_restart: true + need_redeploy: true description_en: working directory for obagent description_local: Obagent工作目录 - name: server_port + name_local: 服务端口 require: true + essential: true type: INT default: 8088 min_value: 1025 @@ -14,7 +18,9 @@ description_en: port number for pulling metrics and management description_local: 提供拉取 metrics 和管理的端口 - name: pprof_port + name_local: 调试端口 require: true + essential: true type: INT default: 8089 min_value: 1025 @@ -41,7 +47,9 @@ description_en: log path description_local: 日志路径 - name: crypto_method + name_local: 加密方式 require: true + essential: true type: STRING default: plain min_value: NULL @@ -51,6 +59,8 @@ description_local: 加密方式,仅支持 aes 和 plain - name: crypto_path require: true + name_local: 秘钥路径 + essential: true type: STRING default: conf/.config_secret.key min_value: NULL @@ -104,7 +114,9 @@ description_en: whether to enable log compression description_local: 是否开启日志压缩 - name: http_basic_auth_user + name_local: 用户名 require: true + essential: true type: STRING default: admin min_value: NULL @@ -113,7 +125,9 @@ description_en: username for HTTP authentication description_local: HTTP 服务认证用户名 - name: http_basic_auth_password + name_local: 密码 require: false + essential: true type: STRING default: root min_value: NULL @@ -122,7 +136,9 @@ description_en: password for HTTP authentication description_local: HTTP 服务认证密码 - name: pprof_basic_auth_user + name_local: 调试用户名 require: true + essential: true type: STRING default: admin min_value: NULL @@ -131,7 +147,9 @@ description_en: username for debug service description_local: debug 接口认证用户名 - name: pprof_basic_auth_password + name_local: 调试密码 require: false + essential: true type: STRING default: root min_value: NULL @@ -203,7 +221,9 @@ description_en: zone name for your observer description_local: observer 所在的 zone 名字 - name: ob_monitor_status + name_local: OceanBase 指标监控采集 require: true + essential: true type: STRING default: active min_value: NULL @@ -212,7 +232,9 @@ description_en: monitor status for OceanBase Database. Active is to enable. Inactive is to disable. description_local: OceanBase 监控指标采集状态,active 表示开启,inactive 表示关闭 - name: host_monitor_status + name_local: 主机指标监控采集 require: true + essential: true type: STRING default: active min_value: NULL @@ -221,7 +243,9 @@ description_en: monitor status for your host. Active is to enable. Inactive is to disable. description_local: 主机监控指标采集状态, active 表示开启, inactive 表示关闭 - name: disable_http_basic_auth + name_local: 禁用 HTTP 服务的basic auth 认证 require: true + essential: true type: BOOL default: false min_value: NULL @@ -230,7 +254,9 @@ description_en: whether to disable the basic authentication for HTTP service. True is to disable. False is to enable. description_local: 是否禁用 HTTP 服务的basic auth 认证,true 表示禁用,false 表示不禁用 - name: disable_pprof_basic_auth + name_local: 禁用 debug 接口的basic auth 认证 require: true + essential: true type: BOOL default: false min_value: NULL @@ -248,7 +274,9 @@ description_en: status for OceanBase Database log alarm. Active is to enable. Inactive is to disable. description_local: OceanBase 日志报警状态,active 表示开启,inactive 表示关闭 - name: alertmanager_address + name_local: Alertmanager 地址 require: false + essential: true type: STRING default: '' min_value: NULL diff --git a/plugins/obagent/1.1.0/start.py b/plugins/obagent/1.1.0/start.py index d2fbef3..d156930 100644 --- a/plugins/obagent/1.1.0/start.py +++ b/plugins/obagent/1.1.0/start.py @@ -108,7 +108,7 @@ def encrypt(key, data): def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -137,10 +137,11 @@ def generate_aes_b64_key(): return base64.b64encode(key.encode('utf-8')) -def start(plugin_context, local_home_path, repository_dir, deploy_name=None, *args, **kwargs): +def start(plugin_context, local_home_path, *args, **kwargs): global stdio cluster_config = plugin_context.cluster_config clients = plugin_context.clients + deploy_name = plugin_context.deploy_name stdio = plugin_context.stdio options = plugin_context.options config_files = {} @@ -158,6 +159,11 @@ def start(plugin_context, local_home_path, repository_dir, deploy_name=None, *ar "ob_install_path": "home_path" } + for repository in plugin_context.repositories: + if repository.name == cluster_config.name: + break + repository_dir = repository.repository_dir + stdio.start_loading('Start obagent') for server in cluster_config.servers: client = clients[server] @@ -281,21 +287,28 @@ def start(plugin_context, local_home_path, repository_dir, deploy_name=None, *ar stdio.start_loading('obagent program health check') time.sleep(1) failed = [] - fail_time = 0 - for server in cluster_config.servers: - client = clients[server] - server_config = cluster_config.get_server_conf(server) - stdio.verbose('%s program health check' % server) - pid = client.execute_command("cat %s" % pid_path[server]).stdout.strip() - if pid: - if confirm_port(client, pid, int(server_config["server_port"])): - stdio.verbose('%s obagent[pid: %s] started', server, pid) - client.execute_command('echo %s > %s' % (pid, pid_path[server])) + servers = cluster_config.servers + count = 20 + while servers and count: + count -= 1 + tmp_servers = [] + for server in servers: + client = clients[server] + server_config = cluster_config.get_server_conf(server) + stdio.verbose('%s program health check' % server) + pid = client.execute_command("cat %s" % pid_path[server]).stdout.strip() + if pid: + if confirm_port(client, pid, int(server_config["server_port"])): + stdio.verbose('%s obagent[pid: %s] started', server, pid) + elif count: + tmp_servers.append(server) + else: + failed.append('failed to start %s obagent' % server) else: - fail_time += 1 - else: - failed.append('failed to start %s obagent' % server) - + failed.append('failed to start %s obagent' % server) + servers = tmp_servers + if servers and count: + time.sleep(1) if failed: stdio.stop_loading('fail') for msg in failed: diff --git a/plugins/obagent/1.1.1/start_check.py b/plugins/obagent/1.1.1/start_check.py index 3d83b34..e3e5ad6 100644 --- a/plugins/obagent/1.1.1/start_check.py +++ b/plugins/obagent/1.1.1/start_check.py @@ -20,7 +20,8 @@ from __future__ import absolute_import, division, print_function -from _errno import EC_CONFIG_CONFLICT_PORT +import os +import _errno as err stdio = None @@ -29,7 +30,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -37,36 +38,106 @@ def get_port_socket_inode(client, port): return res.stdout.strip().split('\n') -def start_check(plugin_context, strict_check=False, *args, **kwargs): - # def alert(*arg, **kwargs): - # global success - # if strict_check: - # success = False - # stdio.error(*arg, **kwargs) - # else: - # stdio.warn(*arg, **kwargs) - def critical(*arg, **kwargs): +def start_check(plugin_context, init_check_status=False, work_dir_check=False, work_dir_empty_check=True, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) - global stdio + check_fail(item, error, suggests) + stdio.error(error) + + global stdio, success + success = True cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio servers_port = {} + check_status = {} + servers_dirs = {} + servers_check_dirs = {} + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + stdio.start_loading('Check before start obagent') for server in cluster_config.servers: ip = server.ip client = clients[server] server_config = cluster_config.get_server_conf(server) port = int(server_config["server_port"]) - prometheus_port = int(server_config["pprof_port"]) - remote_pid_path = "%s/run/obagent-%s-%s.pid" % (server_config['home_path'], server.ip, server_config["server_port"]) - remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s' % remote_pid): - continue + if not precheck: + remote_pid_path = "%s/run/obagent-%s-%s.pid" % (server_config['home_path'], server.ip, server_config["server_port"]) + remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + key = 'home_path' + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False + if ip not in servers_port: servers_port[ip] = {} ports = servers_port[ip] @@ -76,14 +147,25 @@ def critical(*arg, **kwargs): port = int(server_config[key]) # alert_f = alert if key == 'pprof_port' else critical if port in ports: - critical(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key'])) + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) continue ports[port] = { 'server': server, 'key': key } if get_port_socket_inode(client, port): - critical('%s:%s port is already used' % (ip, port)) + critical( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + + for server in cluster_config.servers: + wait_2_pass() if success: stdio.stop_loading('succeed') diff --git a/plugins/obagent/1.3.0/connect.py b/plugins/obagent/1.3.0/connect.py new file mode 100644 index 0000000..eab4e28 --- /dev/null +++ b/plugins/obagent/1.3.0/connect.py @@ -0,0 +1,100 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import json +import requests +from requests.auth import HTTPBasicAuth + +from _errno import EC_FAIL_TO_CONNECT + + +class ObagentAPICursor(object): + cmd_template = '{protocol}://{ip}:{port}/{suffix}' + + def __init__(self, ip, port, username=None, password=None, ssl=False): + self.ip = ip + self.port = port + self.username = username + self.password = password + self.ssl = ssl + protocol = 'https' if ssl else 'http' + self.url_prefix = "{protocol}://{ip}:{port}/".format(protocol=protocol, ip=self.ip, port=self.port) + if self.username: + self.auth = HTTPBasicAuth(username=username, password=password) + else: + self.auth = None + + def connect(self, stdio=None): + return self._request('GET', 'api/v1/agent/status', stdio=stdio) + + def reload(self, data, stdio=None): + return self._request('POST', 'api/v1/module/config/update', data, stdio=stdio) + + def _request(self, method, api, data=None, stdio=None): + url = self.url_prefix + api + stdio.verbose('send http request method: {}, url: {}, data: {}'.format(method, url, data)) + try: + if data is not None: + data = json.dumps(data) + resp = requests.request(method, url, auth=self.auth, data=data, verify=False) + return_code = resp.status_code + content = resp.content + except Exception as e: + stdio.exception("") + return_code = 500 + content = str(e) + if return_code == 200: + return True + stdio.verbose("request obagent failed: %s" % content) + return False + + +def connect(plugin_context, target_server=None, *args, **kwargs): + cluster_config = plugin_context.cluster_config + stdio = plugin_context.stdio + if target_server: + servers = [target_server] + stdio.start_loading('Connect to Obagent ({})'.format(target_server)) + else: + servers = cluster_config.servers + stdio.start_loading('Connect to Obagent') + cursors = {} + for server in servers: + config = cluster_config.get_server_conf(server) + ssl = False + username = '' + password = '' + if config.get('http_basic_auth_user'): + username = config['http_basic_auth_user'] + if config.get('http_basic_auth_password'): + password = config['http_basic_auth_password'] + stdio.verbose('connect obagent ({}:{} by user {})'.format(server.ip, config['mgragent_http_port'], username)) + api_cursor = ObagentAPICursor(ip=server.ip, port=config['mgragent_http_port'], username=username, password=password, + ssl=ssl) + if api_cursor.connect(stdio=stdio): + cursors[server] = api_cursor + if not cursors: + stdio.error(EC_FAIL_TO_CONNECT.format(component=cluster_config.name)) + stdio.stop_loading('fail') + return plugin_context.return_false() + stdio.stop_loading('succeed') + return plugin_context.return_true(connect=cursors, cursor=cursors) diff --git a/plugins/obagent/1.3.0/display.py b/plugins/obagent/1.3.0/display.py new file mode 100644 index 0000000..2b82766 --- /dev/null +++ b/plugins/obagent/1.3.0/display.py @@ -0,0 +1,46 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from __future__ import absolute_import, division, print_function + +from tool import NetUtil + + +def display(plugin_context, cursor, *args, **kwargs): + stdio = plugin_context.stdio + clients = plugin_context.clients + cluster_config = plugin_context.cluster_config + servers = cluster_config.servers + result = [] + for server in servers: + api_cursor = cursor.get(server) + server_config = cluster_config.get_server_conf(server) + ip = server.ip + if ip == '127.0.0.1': + ip = NetUtil.get_host_ip() + result.append({ + 'ip': ip, + 'status': 'active' if api_cursor and api_cursor.connect(stdio) else 'inactive', + 'mgragent_http_port': server_config['mgragent_http_port'], + 'monagent_http_port': server_config['monagent_http_port'] + }) + + stdio.print_list(result, ['ip', 'mgragent_http_port', 'monagent_http_port', 'status'], + lambda x: [x['ip'], x['mgragent_http_port'], x['monagent_http_port'], x['status']], title='obagent') + plugin_context.return_true() diff --git a/plugins/obagent/1.3.0/file_map.yaml b/plugins/obagent/1.3.0/file_map.yaml new file mode 100644 index 0000000..03e84e0 --- /dev/null +++ b/plugins/obagent/1.3.0/file_map.yaml @@ -0,0 +1,29 @@ +# 运维 agent 二进制 +- src_path: ./home/admin/obagent/bin/ob_mgragent + target_path: bin/ob_mgragent + type: bin + mode: 755 +# 监控 agent 二进制 +- src_path: ./home/admin/obagent/bin/ob_monagent + target_path: bin/ob_monagent + type: bin + mode: 755 +# 守护进程二进制 +- src_path: ./home/admin/obagent/bin/ob_agentd + target_path: bin/ob_agentd + type: bin + mode: 755 +# 命令行工具二进制 +- src_path: ./home/admin/obagent/bin/ob_agentctl + target_path: bin/ob_agentctl + type: bin + mode: 755 +# agent 二进制目录 +- src_path: ./home/admin/obagent/bin + target_path: bin + type: dir +# 配置定义目录 +- src_path: ./home/admin/obagent/conf + target_path: conf + type: dir + install_method: cp \ No newline at end of file diff --git a/plugins/obagent/1.3.0/init.py b/plugins/obagent/1.3.0/init.py new file mode 100644 index 0000000..f92aff8 --- /dev/null +++ b/plugins/obagent/1.3.0/init.py @@ -0,0 +1,84 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +from _errno import EC_FAIL_TO_INIT_PATH, InitDirFailedErrorMessage + + +def init(plugin_context, *args, **kwargs): + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + global_ret = True + force = getattr(plugin_context.options, 'force', False) + clean = getattr(plugin_context.options, 'clean', False) + stdio.start_loading('Initializes obagent work home') + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + home_path = server_config['home_path'] + stdio.verbose('%s init cluster work home', server) + need_clean = force + if clean and not force: + if client.execute_command('bash -c \'if [[ "$(ls -d {0} 2>/dev/null)" != "" && ! -O {0} ]]; then exit 0; else exit 1; fi\''.format(home_path)): + owner = client.execute_command("ls -ld %s | awk '{print $3}'" % home_path).stdout.strip() + global_ret = False + err_msg = ' {} is not empty, and the owner is {}'.format(home_path, owner) + stdio.error(EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=err_msg)) + continue + need_clean = True + + if need_clean: + client.execute_command("^%s/bin/ob_agentctl stop'" % home_path) + if client.execute_command('bash -c \'if [[ "$(ls -d {0} 2>/dev/null)" != "" && ! -O {0} ]]; then exit 0; else exit 1; fi\''.format(home_path)): + owner = client.execute_command("ls -ld %s | awk '{print $3}'" % home_path).stdout.strip() + global_ret = False + err_msg = ' {} is not empty, and the owner is {}'.format(home_path, owner) + stdio.error(EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=err_msg)) + continue + + if need_clean: + ret = client.execute_command('rm -fr %s' % home_path, timeout=-1) + if not ret: + global_ret = False + stdio.error(EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=ret.stderr)) + continue + else: + if client.execute_command('mkdir -p %s' % home_path): + ret = client.execute_command('ls %s' % (home_path)) + if not ret or ret.stdout.strip(): + global_ret = False + stdio.error(EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=InitDirFailedErrorMessage.NOT_EMPTY.format(path=home_path))) + continue + else: + global_ret = False + stdio.error(EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=InitDirFailedErrorMessage.CREATE_FAILED.format(path=home_path))) + continue + + if not client.execute_command("bash -c 'mkdir -p %s/{run,bin,conf,log,tmp,backup,pkg_store,task_store,position_store,site-packages}'" % home_path): + global_ret = False + stdio.error(EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=InitDirFailedErrorMessage.PATH_ONLY.format(path=home_path))) + + if global_ret: + stdio.stop_loading('succeed') + plugin_context.return_true() + else: + stdio.stop_loading('fail') \ No newline at end of file diff --git a/plugins/obagent/1.3.0/parameter.yaml b/plugins/obagent/1.3.0/parameter.yaml new file mode 100644 index 0000000..8cafaa3 --- /dev/null +++ b/plugins/obagent/1.3.0/parameter.yaml @@ -0,0 +1,243 @@ +- name: home_path + name_local: 工作目录 + require: true + essential: true + type: STRING + need_restart: true + need_redeploy: true + description_en: working directory for obagent + description_local: Obagent工作目录 +- name: log_path + require: true + type: STRING + default: log/monagent.log + need_restart: true + need_redeploy: true + description_en: log path + description_local: 日志路径 +- name: http_basic_auth_user + name_local: 用户名 + require: true + essential: true + type: STRING + default: admin + need_restart: true + description_en: username for HTTP authentication + description_local: HTTP 服务认证用户名 +- name: http_basic_auth_password + name_local: 密码 + require: true + essential: true + type: STRING + default: root + need_restart: true + need_redeploy: false + description_en: password for HTTP authentication + description_local: HTTP 服务认证密码 +- name: mgragent_http_port + name_local: 管理服务端口 + require: true + essential: true + type: INT + default: 8089 + need_restart: true + need_redeploy: false + description_en: The port of manager agent + description_local: OBAgent 管理服务端口 +- name: mgragent_log_level + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: The log level of manager agent. + description_local: ob_mgragent 日志级别 +- name: mgragent_log_max_size + require: false + type: INT + default: 30 + need_restart: true + need_redeploy: false + description_en: The total size of manager agent.Log size is measured in Megabytes. + description_local: ob_mgragent 日志文件大小(单位:mb) +- name: mgragent_log_max_days + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: Expiration time for manager agent logs. The default value is 30 days. + description_local: ob_mgragent 日志文件最大保留天数 +- name: mgragent_log_max_backups + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: The maximum number for manager agent log files. + description_local: ob_mgragent 日志文件最大备份数 +- name: mgragent_log_compress + require: false + type: BOOL + need_restart: true + need_redeploy: false + description_en: ob_mgragent log compression switch + description_local: ob_mgragent 日志压缩开关 +- name: monagent_http_port + name_local: 监控服务端口 + require: true + essential: true + type: INT + default: 8088 + need_restart: true + need_redeploy: false + description_en: The port of monitor agent. + description_local: OBAgent 监控服务端口 +- name: monagent_host_ip + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: ob_monagent host ip + description_local: ob_monagent 主机 ip +- name: monitor_password + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: monitor password for OceanBase Database + default: '' + description_local: OceanBase 数据库监控数据采集用户密码 +- name: sql_port + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: SQL port for observer + default: 2881 + min_value: 1025 + max_value: 65535 + description_local: observer的 SQL 端口 +- name: rpc_port + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: the RPC port for observer + default: 2882 + min_value: 1025 + max_value: 65535 + description_local: observer 的 RPC 端口 +- name: cluster_name + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: cluster name for OceanBase Database + default: obcluster + description_local: OceanBase Database 集群名 +- name: cluster_id + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: cluster ID for OceanBase Database + default: 1 + min_value: 1 + max_value: 4294901759 + description_local: OceanBase 集群 ID +- name: zone_name + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: zone name for your observer + default: zone1 + min_value: + max_value: + description_local: observer 所在的 zone 名字 +- name: ob_log_path + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: observer log path + description_local: observer 日志盘路径 +- name: ob_data_path + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: observer data path + description_local: observer 数据盘路径 +- name: ob_install_path + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: observer install path + description_local: observer 安装目录 +- name: observer_log_path + require: false + type: STRING + need_restart: true + need_redeploy: false + description_en: observer install path log + description_local: observer 安装目录下日志路径 +- name: monagent_log_level + require: false + type: STRING + default: info + need_restart: true + need_redeploy: false + description_en: The log level of monitor agent. + description_local: ob_monagent 日志级别 +- name: monagent_log_max_size + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: The total size of monitor agent.Log size is measured in Megabytes. + description_local: ob_monagent 日志文件大小(单位:mb) +- name: monagent_log_max_days + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: Expiration time for monitor agent logs. + description_local: ob_monagent 日志文件最大保留天数 +- name: monagent_log_max_backups + require: false + type: INT + need_restart: true + need_redeploy: false + description_en: The maximum number for monitor agent log files. + description_local: ob_monagent 日志文件最大备份数 +- name: monagent_log_compress + require: false + type: BOOL + need_restart: true + need_redeploy: false + description_en: ob_monagent log compression switch + description_local: ob_monagent 日志压缩开关 +- name: ob_monitor_status + name_local: OceanBase 指标监控采集 + require: true + essential: true + type: STRING + default: active + need_restart: true + need_redeploy: false + description_en: monitor status for OceanBase Database. Active is to enable. Inactive is to disable. + description_local: OceanBase 监控指标采集状态,active 表示开启,inactive 表示关闭 +- name: target_sync_configs + require: false + type: PARAM_LIST + need_restart: true + description_en: + description_local: '''将地址同步至指定远端目录 + target_sync_configs: + - host: 192.168.1.1 + target_dir: /data/prometheus/targets + user: user1 + port: 22 + # password: ***** + key_file: xxxxx + ''' diff --git a/plugins/obagent/1.3.0/reload.py b/plugins/obagent/1.3.0/reload.py new file mode 100644 index 0000000..886a9e7 --- /dev/null +++ b/plugins/obagent/1.3.0/reload.py @@ -0,0 +1,109 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os +import json +from copy import deepcopy +from glob import glob +from tool import YamlLoader, FileUtil + +from _errno import * + + +def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): + stdio = plugin_context.stdio + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + servers = cluster_config.servers + + for repository in plugin_context.repositories: + if repository.name == cluster_config.name: + break + repository_dir = repository.repository_dir + + yaml = YamlLoader(stdio) + config_map = { + "monitor_password": "root_password", + "sql_port": "mysql_port", + "rpc_port": "rpc_port", + "cluster_name": "appname", + "cluster_id": "cluster_id", + "zone_name": "zone", + } + global_change_conf = {} + for comp in ['oceanbase', 'oceanbase-ce']: + if comp in cluster_config.depends: + root_servers = {} + ob_config = cluster_config.get_depend_config(comp) + new_ob_config = new_cluster_config.get_depend_config(comp) + ob_config = {} if ob_config is None else ob_config + new_ob_config = {} if new_ob_config is None else new_ob_config + for key in config_map: + if ob_config.get(key) != new_ob_config.get(key): + global_change_conf[config_map[key]] = new_ob_config.get(key) + + global_ret = True + stdio.start_loading('Reload obagent') + for server in servers: + change_conf = deepcopy(global_change_conf) + client = clients[server] + api_cursor = cursor.get(server) + stdio.verbose('get %s old configuration' % (server)) + config = cluster_config.get_server_conf_with_default(server) + stdio.verbose('get %s new configuration' % (server)) + new_config = new_cluster_config.get_server_conf_with_default(server) + stdio.verbose('get %s cluster address' % (server)) + stdio.verbose('compare configuration of %s' % (server)) + with FileUtil.open(os.path.join(repository_dir, 'conf/obd_agent_mapper.yaml')) as f: + data = yaml.load(f).get('config_mapper', {}) + for key in new_config: + if key not in data: + stdio.warn('%s no in obd_agent_mapper.yaml, skip' % key) + continue + if key not in config or config[key] != new_config[key]: + item = cluster_config.get_temp_conf_item(key) + if item: + if item.need_redeploy or item.need_restart: + stdio.verbose('%s can not be reload' % key) + global_ret = False + continue + try: + item.modify_limit(config.get(key), new_config.get(key)) + except Exception as e: + stdio.verbose('%s: %s' % (server, str(e))) + global_ret = False + continue + change_conf[key] = new_config[key] + + if change_conf: + stdio.verbose('%s apply new configuration' % server) + data = [{'key': key, 'value': change_conf[key]} for key in change_conf] + if not (api_cursor and api_cursor.reload(data, stdio)): + global_ret = False + stdio.error(EC_OBAGENT_RELOAD_FAILED.format(server=server)) + + if global_ret: + stdio.stop_loading('succeed') + return plugin_context.return_true() + else: + stdio.stop_loading('fail') + return diff --git a/plugins/obagent/1.3.0/restart.py b/plugins/obagent/1.3.0/restart.py new file mode 100644 index 0000000..4ecf445 --- /dev/null +++ b/plugins/obagent/1.3.0/restart.py @@ -0,0 +1,135 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os + + +class Restart(object): + + def __init__(self, plugin_context, local_home_path, start_check_plugin, start_plugin, stop_plugin, connect_plugin, + display_plugin, repository, new_cluster_config=None, new_clients=None, deploy_name=None): + self.local_home_path = local_home_path + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + + self.components = plugin_context.components + self.clients = plugin_context.clients + self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode + self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context + self.repository = repository + self.start_check_plugin = start_check_plugin + self.start_plugin = start_plugin + self.connect_plugin = connect_plugin + self.display_plugin = display_plugin + self.stop_plugin = stop_plugin + self.new_clients = new_clients + self.new_cluster_config = new_cluster_config + self.sub_io = self.stdio.sub_io() + self.deploy_name = deploy_name + + def dir_read_check(self, client, path): + if not client.execute_command('cd %s' % path): + dirpath, name = os.path.split(path) + return self.dir_read_check(client, dirpath) and client.execute_command('sudo chmod +1 %s' % path) + return True + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) + + def restart(self): + clients = self.clients + if not self.call_plugin(self.stop_plugin, clients=clients): + self.stdio.stop_loading('stop_loading', 'fail') + return False + + if self.new_clients: + self.stdio.verbose('use new clients') + for server in self.cluster_config.servers: + new_client = self.new_clients[server] + server_config = self.cluster_config.get_server_conf(server) + home_path = server_config['home_path'] + if not new_client.execute_command('sudo chown -R %s: %s' % (new_client.config.username, home_path)): + self.stdio.stop_loading('stop_loading', 'fail') + return False + self.dir_read_check(new_client, home_path) + clients = self.new_clients + + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + + self.call_plugin(self.start_check_plugin, clients=clients, cluster_config=cluster_config) + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, repository=self.repository): + self.rollback() + self.stdio.stop_loading('stop_loading', 'fail') + return False + ret = self.call_plugin(self.connect_plugin, clients=clients, cluster_config=cluster_config) + if ret: + return self.call_plugin(self.display_plugin, clients=clients, cluster_config=cluster_config, cursor=ret.get_return('cursor')) + return False + + def rollback(self): + if self.new_clients: + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.new_clients, cluster_config=cluster_config) + for server in self.cluster_config.servers: + client = self.clients[server] + new_client = self.new_clients[server] + server_config = self.cluster_config.get_server_conf(server) + home_path = server_config['home_path'] + new_client.execute_command('sudo chown -R %s: %s' % (client.config.username, home_path)) + + +def restart(plugin_context, local_home_path, start_check_plugin, start_plugin, stop_plugin, connect_plugin, display_plugin, + new_cluster_config=None, new_clients=None, rollback=False, *args, **kwargs): + repository = kwargs.get('repository') + deploy_name = plugin_context.deploy_name + task = Restart(plugin_context=plugin_context, local_home_path=local_home_path, start_plugin=start_plugin, + stop_plugin=stop_plugin, connect_plugin=connect_plugin, + display_plugin=display_plugin, repository=repository, new_cluster_config=new_cluster_config, + new_clients=new_clients, deploy_name=deploy_name, start_check_plugin=start_check_plugin) + call = task.rollback if rollback else task.restart + if call(): + plugin_context.return_true() diff --git a/plugins/obagent/1.3.0/start.py b/plugins/obagent/1.3.0/start.py new file mode 100644 index 0000000..d11c25a --- /dev/null +++ b/plugins/obagent/1.3.0/start.py @@ -0,0 +1,337 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os +import re +import sys +import time +import random +import base64 +import tempfile +from copy import deepcopy + +from Crypto import Random +from Crypto.Cipher import AES + +from ssh import SshClient, SshConfig +from tool import YamlLoader, FileUtil + +stdio = None +OBAGNET_CONFIG_MAP = { + "monitor_password": "{ocp_agent_monitor_password}", + "sql_port": "{mysql_port}", + "rpc_port": "{rpc_port}", + "cluster_name": "{appname}", + "cluster_id": "{cluster_id}", + "zone_name": "{zone}", + "ob_log_path": "{home_path}/store", + "ob_data_path": "{home_path}/store", + "ob_install_path": "{home_path}", + "observer_log_path": "{home_path}/log", +} + +if sys.version_info.major == 2: + + def generate_key(key): + genKey = [chr(0)] * 16 + for i in range(min(16, len(key))): + genKey[i] = key[i] + i = 16 + while i < len(key): + j = 0 + while j < 16 and i < len(key): + genKey[j] = chr(ord(genKey[j]) ^ ord(key[i])) + j, i = j + 1, i + 1 + return "".join(genKey) + + + class AESCipher: + bs = AES.block_size + + def __init__(self, key): + self.key = generate_key(key) + + def encrypt(self, message): + message = self._pad(message) + iv = Random.new().read(AES.block_size) + cipher = AES.new(self.key, AES.MODE_CBC, iv) + return base64.b64encode(iv + cipher.encrypt(message)).decode('utf-8') + + def _pad(self, s): + return s + (self.bs - len(s) % self.bs) * chr(self.bs - len(s) % self.bs) + +else: + def generate_key(key): + genKey = [0] * 16 + for i in range(min(16, len(key))): + genKey[i] = key[i] + i = 16 + while i < len(key): + j = 0 + while j < 16 and i < len(key): + genKey[j] = genKey[j] ^ key[i] + j, i = j + 1, i + 1 + genKey = [chr(k) for k in genKey] + return bytes("".join(genKey), encoding="utf-8") + + + class AESCipher: + bs = AES.block_size + + def __init__(self, key): + self.key = generate_key(key) + + def encrypt(self, message): + message = self._pad(message) + iv = Random.new().read(AES.block_size) + cipher = AES.new(self.key, AES.MODE_CBC, iv) + return str(base64.b64encode(iv + cipher.encrypt(bytes(message, encoding='utf-8'))), encoding="utf-8") + + def _pad(self, s): + return s + (self.bs - len(s) % self.bs) * chr(self.bs - len(s) % self.bs) + + +def encrypt(key, data): + key = base64.b64decode(key) + cipher = AESCipher(key) + return cipher.encrypt(data) + + +def get_port_socket_inode(client, port): + port = hex(port)[2:].zfill(4).upper() + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + res = client.execute_command(cmd) + if not res or not res.stdout.strip(): + return False + stdio.verbose(res.stdout) + return res.stdout.strip().split('\n') + + +def confirm_port(client, pid, port): + socket_inodes = get_port_socket_inode(client, port) + if not socket_inodes: + return False + ret = client.execute_command("ls -l /proc/%s/fd/ |grep -E 'socket:\[(%s)\]'" % (pid, '|'.join(socket_inodes))) + if ret and ret.stdout.strip(): + return True + return False + + +def generate_aes_b64_key(): + n = random.randint(1, 3) * 8 + key = [] + c = 0 + while c < n: + key += chr(random.randint(33, 127)) + c += 1 + key = ''.join(key) + return base64.b64encode(key.encode('utf-8')) + + +def get_missing_required_parameters(parameters): + results = [] + for key in OBAGNET_CONFIG_MAP: + if parameters.get(key) is None: + results.append(key) + return results + + +def prepare_parameters(cluster_config): + env = {} + depend_info = {} + ob_servers_config = {} + depends_keys = ["ocp_agent_monitor_password", "appname", "cluster_id"] + for comp in ["oceanbase", "oceanbase-ce"]: + if comp in cluster_config.depends: + observer_globals = cluster_config.get_depend_config(comp) + for key in depends_keys: + value = observer_globals.get(key) + if value is not None: + depend_info[key] = value + ob_servers = cluster_config.get_depend_servers(comp) + for server in ob_servers: + ob_servers_config[server] = cluster_config.get_depend_config(comp, server) + + for server in cluster_config.servers: + server_config = deepcopy(cluster_config.get_server_conf_with_default(server)) + user_server_config = deepcopy(cluster_config.get_server_conf(server)) + if 'monagent_host_ip' not in user_server_config: + server_config['monagent_host_ip'] = server.ip + missed_keys = get_missing_required_parameters(user_server_config) + if missed_keys and server in ob_servers_config: + for key in depend_info: + ob_servers_config[server][key] = depend_info[key] + for key in missed_keys: + server_config[key] = OBAGNET_CONFIG_MAP[key].format(server_ip=server.ip, **ob_servers_config[server]) + env[server] = server_config + return env + + +def start(plugin_context, *args, **kwargs): + global stdio + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + options = plugin_context.options + stdio = plugin_context.stdio + deploy_name = plugin_context.deploy_name + pid_path = {} + yaml = YamlLoader(stdio) + start_env = plugin_context.get_variable('start_env') + + if not start_env: + start_env = prepare_parameters(cluster_config) + + repository_dir = None + for repository in plugin_context.repositories: + if repository.name == cluster_config.name: + repository_dir = repository.repository_dir + break + with FileUtil.open(os.path.join(repository_dir, 'conf/obd_agent_mapper.yaml')) as f: + config_mapper = yaml.load(f).get('config_mapper', {}) + stdio.start_loading('Start obagent') + + targets = [] + for server in cluster_config.servers: + client = clients[server] + server_config = start_env[server] + home_path = server_config['home_path'] + pid_path[server] = '%s/run/ob_agentd.pid' % home_path + mgragent_http_port = int(server_config['mgragent_http_port']) + targets.append('{}:{}'.format(server.ip, mgragent_http_port)) + remote_pid = client.execute_command("cat %s" % pid_path[server]).stdout.strip() + if remote_pid and client.execute_command('ls /proc/%s' % remote_pid): + continue + + home_path = server_config['home_path'] + use_parameter = True + config_flag = os.path.join(home_path, '.configured') + if getattr(options, 'without_parameter', False) and client.execute_command('ls %s' % config_flag): + use_parameter = False + + if use_parameter: + # todo: set agent secret key + mgr_conf = os.path.join(home_path, 'conf/mgragent.yaml') + mon_conf = os.path.join(home_path, 'conf/monagent.yaml') + agent_conf = os.path.join(home_path, 'conf/agentctl.yaml') + for conf in [mgr_conf, mon_conf, agent_conf]: + ret = client.execute_command('cat {}'.format(conf)) + if ret: + content = ret.stdout + content = re.sub(r"cryptoMethod:\s+aes", "cryptoMethod: plain", content) + client.write_file(content, conf) + client.execute_command('chmod 755 {}'.format(conf)) + for key in server_config: + if server_config[key] is None: + server_config[key] = '' + if isinstance(server_config[key], bool): + server_config[key] = str(server_config[key]).lower() + + cmds = [] + for key, value in server_config.items(): + if key in config_mapper: + cmds.append("%s=%s" % (config_mapper[key], value)) + cmd = 'cd %s;%s/bin/ob_agentctl config -u %s && touch %s' % (home_path, home_path, ','.join(cmds), config_flag) + res = client.execute_command(cmd) + if not res: + stdio.error('failed to set config to {} obagent.'.format(server)) + return plugin_context.return_false() + + if not client.execute_command('cd %s;%s/bin/ob_agentctl start' % (home_path, home_path)): + stdio.error('failed to start {} obagent.'.format(server)) + return plugin_context.return_false() + + stdio.stop_loading('succeed') + stdio.start_loading('obagent program health check') + time.sleep(1) + failed = [] + servers = cluster_config.servers + count = 20 + while servers and count: + count -= 1 + tmp_servers = [] + for server in servers: + client = clients[server] + server_config = start_env[server] + home_path = server_config['home_path'] + stdio.verbose('%s program health check' % server) + pid = client.execute_command("cat %s" % pid_path[server]).stdout.strip() + if pid: + mgr_pid = client.execute_command("cat %s" % os.path.join(home_path, 'run/ob_mgragent.pid')).stdout.strip() + if mgr_pid and confirm_port(client, mgr_pid, int(server_config["mgragent_http_port"])): + stdio.verbose('%s obagent[pid: %s] started', server, pid) + elif count: + tmp_servers.append(server) + else: + failed.append('failed to start %s obagent' % server) + else: + failed.append('failed to start %s obagent' % server) + servers = tmp_servers + if servers and count: + time.sleep(1) + if failed: + stdio.stop_loading('fail') + for msg in failed: + stdio.warn(msg) + plugin_context.return_false() + else: + global_config = cluster_config.get_global_conf() + target_sync_configs = global_config.get('target_sync_configs', []) + stdio.verbose('start to sync target config') + data = [{'targets': targets}] + default_ssh_config = None + for client in clients.values(): + default_ssh_config = client.config + break + for target_sync_config in target_sync_configs: + host = None + target_dir = None + try: + host = target_sync_config.get('host') + target_dir = target_sync_config.get('target_dir') + if not host or not target_dir: + continue + ssh_config_keys = ['username', 'password', 'port', 'key_file', 'timeout'] + auth_keys = ['username', 'password', 'key_file'] + for key in auth_keys: + if key in target_sync_config: + config = SshConfig(host) + break + else: + config = deepcopy(default_ssh_config) + for key in ssh_config_keys: + if key in target_sync_config: + setattr(config, key, target_sync_config[key]) + with tempfile.NamedTemporaryFile(suffix='.yaml') as f: + yaml.dump(data, f) + f.flush() + file_name = '{}.yaml'.format(deploy_name or hash(cluster_config)) + file_path = os.path.join(target_dir, file_name) + remote_client = SshClient(config) + remote_client.connect() + remote_client.put_file(f.name, file_path) + except: + stdio.warn('failed to sync target to {}:{}'.format(host, target_dir)) + stdio.exception('') + stdio.stop_loading('succeed') + plugin_context.return_true(need_bootstrap=False) + + diff --git a/plugins/obagent/1.3.0/start_check.py b/plugins/obagent/1.3.0/start_check.py new file mode 100644 index 0000000..0373a0e --- /dev/null +++ b/plugins/obagent/1.3.0/start_check.py @@ -0,0 +1,250 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os +from copy import deepcopy + +import _errno as err +from tool import YamlLoader, FileUtil + + +stdio = None +success = True + +OBAGNET_CONFIG_MAP = { + "monitor_password": "{ocp_agent_monitor_password}", + "sql_port": "{mysql_port}", + "rpc_port": "{rpc_port}", + "cluster_name": "{appname}", + "cluster_id": "{cluster_id}", + "zone_name": "{zone}", + "ob_log_path": "{home_path}/store", + "ob_data_path": "{home_path}/store", + "ob_install_path": "{home_path}", + "observer_log_path": "{home_path}/log", +} + + +def get_missing_required_parameters(parameters): + results = [] + for key in OBAGNET_CONFIG_MAP: + if parameters.get(key) is None: + results.append(key) + return results + + +def get_port_socket_inode(client, port): + port = hex(port)[2:].zfill(4).upper() + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + res = client.execute_command(cmd) + if not res or not res.stdout.strip(): + return False + stdio.verbose(res.stdout) + return res.stdout.strip().split('\n') + + +def prepare_parameters(cluster_config): + env = {} + depend_info = {} + ob_servers_config = {} + depends_keys = ["ocp_agent_monitor_password", "appname", "cluster_id"] + for comp in ["oceanbase", "oceanbase-ce"]: + if comp in cluster_config.depends: + observer_globals = cluster_config.get_depend_config(comp) + for key in depends_keys: + value = observer_globals.get(key) + if value is not None: + depend_info[key] = value + ob_servers = cluster_config.get_depend_servers(comp) + for server in ob_servers: + ob_servers_config[server] = cluster_config.get_depend_config(comp, server) + + for server in cluster_config.servers: + server_config = deepcopy(cluster_config.get_server_conf_with_default(server)) + user_server_config = deepcopy(cluster_config.get_server_conf(server)) + if 'monagent_host_ip' not in user_server_config: + server_config['monagent_host_ip'] = server.ip + missed_keys = get_missing_required_parameters(user_server_config) + if missed_keys and server in ob_servers_config: + for key in depend_info: + ob_servers_config[server][key] = depend_info[key] + for key in missed_keys: + server_config[key] = OBAGNET_CONFIG_MAP[key].format(server_ip=server.ip, **ob_servers_config[server]) + env[server] = server_config + return env + + +def start_check(plugin_context, init_check_status=False, strict_check=False, work_dir_check=False, work_dir_empty_check=True, precheck=False, *args, **kwargs): + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + + def alert(item, error, suggests=[]): + global success + if strict_check: + success = False + check_fail(item, error, suggests) + stdio.error(error) + else: + stdio.warn(error) + + def critical(item, error, suggests=[]): + global success + success = False + status = check_status.get(server).get(item) + status.status = err.CheckStatus.FAIL + status.error = error + status.suggests = suggests + stdio.error(error) + + def check_pass(item): + status = check_status.get(server).get(item).status + if status == err.CheckStatus.WAIT: + check_status.get(server).get(item).status = err.CheckStatus.PASS + + def wait_2_pass(): + status = check_status[server] + for key in status: + if status[key].status == err.CheckStatus.WAIT: + status[key].status = err.CheckStatus.PASS + + global stdio, success + success = True + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + servers_port = {} + servers_dirs = {} + servers_check_dirs = {} + check_status = {} + plugin_context.set_variable('start_check_status', check_status) + + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + 'parameter': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + + stdio.start_loading('Check before start obagent') + env = prepare_parameters(cluster_config) + for server in cluster_config.servers: + ip = server.ip + client = clients[server] + server_config = cluster_config.get_server_conf(server) + if not precheck: + remote_pid_path = "%s/run/ob_agentd.pid" % server_config['home_path'] + remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + check_pass('parameter') + + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + key = 'home_path' + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False + + if ip not in servers_port: + servers_port[ip] = {} + ports = servers_port[ip] + + stdio.verbose('%s port check' % server) + for key in ['mgragent_http_port', 'monagent_http_port']: + port = int(server_config[key]) + if port in ports: + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) + continue + ports[port] = { + 'server': server, + 'key': key + } + if get_port_socket_inode(client, port): + critical( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + check_pass('port') + plugin_context.set_variable('start_env', env) + + for server in cluster_config.servers: + wait_2_pass() + + + if success: + stdio.stop_loading('succeed') + plugin_context.return_true() + else: + stdio.stop_loading('fail') \ No newline at end of file diff --git a/plugins/obagent/1.3.0/status.py b/plugins/obagent/1.3.0/status.py new file mode 100644 index 0000000..7a3f637 --- /dev/null +++ b/plugins/obagent/1.3.0/status.py @@ -0,0 +1,40 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + + +def status(plugin_context, *args, **kwargs): + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + cluster_status = {} + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + cluster_status[server] = 0 + if 'home_path' not in server_config: + stdio.print('%s home_path is empty', server) + continue + remote_pid_path = '%s/run/ob_agentd.pid' % server_config["home_path"] + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid and client.execute_command('ls /proc/%s' % remote_pid): + cluster_status[server] = 1 + return plugin_context.return_true(cluster_status=cluster_status) diff --git a/plugins/obagent/1.3.0/stop.py b/plugins/obagent/1.3.0/stop.py new file mode 100644 index 0000000..6cbd26e --- /dev/null +++ b/plugins/obagent/1.3.0/stop.py @@ -0,0 +1,110 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import time + +from tool import OrderedDict + + +stdio = None + + +def get_port_socket_inode(client, port): + port = hex(port)[2:].zfill(4).upper() + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + res = client.execute_command(cmd) + inode = res.stdout.strip() + if not res or not inode: + return False + stdio.verbose("inode: %s" % inode) + return inode.split('\n') + + +def confirm_port(client, pid, port): + socket_inodes = get_port_socket_inode(client, port) + if not socket_inodes: + return False + ret = client.execute_command("ls -l /proc/%s/fd/ |grep -E 'socket:\[(%s)\]'" % (pid, '|'.join(socket_inodes))) + if ret and ret.stdout.strip(): + return True + return False + + +def stop(plugin_context, *args, **kwargs): + global stdio + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + + servers = {} + stdio.start_loading('Stop obagent') + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + if 'home_path' not in server_config: + stdio.verbose('%s home_path is empty', server) + continue + home_path = server_config["home_path"] + agent_processes = OrderedDict() + agent_processes['obagentd'] = {'path': '%s/run/ob_agentd.pid' % home_path, 'port': None} + agent_processes['monagent'] = {'path': '%s/run/ob_monagent.pid' % home_path, 'port': server_config['monagent_http_port']} + agent_processes['mgragent'] = {'path': '%s/run/ob_mgragent.pid' % home_path, 'port': server_config['mgragent_http_port']} + for agent in agent_processes: + pid = client.execute_command('cat %s' % agent_processes[agent]['path']).stdout.strip() + if pid: + stdio.verbose('%s %s[pid:%s] stopping ...' % (server, agent, pid)) + client.execute_command('kill -9 %s' % pid) + if server not in servers: + servers[server] = {} + servers[server][agent] = {'pid': pid, 'port': agent_processes[agent]['port'], 'path': agent_processes[agent]['path']} + else: + stdio.verbose('%s %s is not running' % (server, agent)) + + count = 10 + time.sleep(1) + while count and servers: + tmp_servers = {} + for server in servers: + agents_info = servers[server] + client = clients[server] + stdio.verbose('%s check whether the port is released' % server) + for agent in agents_info: + pid = agents_info[agent]['pid'] + if client.execute_command('ls /proc/%s' % pid) or (agents_info[agent].get('port') and confirm_port(client, pid, agents_info[agent]['port'])): + tmp_servers[server] = agents_info + break + client.execute_command('rm -f %s' % agents_info[agent]['path']) + agents_info[agent] = {} + else: + stdio.verbose('%s obagent is stopped', server) + servers = tmp_servers + count -= 1 + if count and servers: + time.sleep(3) + + if servers: + stdio.stop_loading('fail') + for server in servers: + stdio.warn('%s port not released', server) + else: + stdio.stop_loading('succeed') + plugin_context.return_true() \ No newline at end of file diff --git a/plugins/obagent/1.3.0/upgrade.py b/plugins/obagent/1.3.0/upgrade.py new file mode 100644 index 0000000..7c4d391 --- /dev/null +++ b/plugins/obagent/1.3.0/upgrade.py @@ -0,0 +1,159 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os + + +def call_plugin(plugin, plugin_context, repositories, *args, **kwargs): + namespace = plugin_context.namespace + namespaces = plugin_context.namespaces + deploy_name = plugin_context.deploy_name + components = plugin_context.components + clients = plugin_context.clients + cluster_config = plugin_context.cluster_config + cmds = plugin_context.cmds + options = plugin_context.options + stdio = plugin_context.stdio + return plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, + stdio, *args, **kwargs) + + +def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, *args, **kwargs): + + def summit_config(): + generate_global_config = generate_configs['global'] + for key in generate_global_config: + cluster_config.update_global_conf(key, generate_global_config[key], False) + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + for key in generate_server_config: + cluster_config.update_server_conf(server, key, generate_server_config[key], False) + + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + + upgrade_ctx = kwargs.get('upgrade_ctx') + local_home_path = kwargs.get('local_home_path') + upgrade_repositories = kwargs.get('upgrade_repositories') + + cur_repository = upgrade_repositories[0] + dest_repository = upgrade_repositories[-1] + repository_dir = dest_repository.repository_dir + kwargs['repository_dir'] = repository_dir + + stop_plugin = search_py_script_plugin([cur_repository], 'stop')[cur_repository] + start_plugin = search_py_script_plugin([dest_repository], 'start')[dest_repository] + connect_plugin = search_py_script_plugin([dest_repository], 'connect')[dest_repository] + display_plugin = search_py_script_plugin([dest_repository], 'display')[dest_repository] + + apply_param_plugin(cur_repository) + if not call_plugin(stop_plugin, plugin_context, repositories=[cur_repository], *args, **kwargs): + return + # clean useless config + clean_files = [ + "conf/config_properties/monagent_basic_auth.yaml", + "conf/module_config/monitor_mysql.yaml", + "conf/module_config/monagent_config.yaml", + "conf/module_config/monitor_ob_log.yaml" + ] + for server in cluster_config.servers: + client = clients[server] + home_path = cluster_config.get_server_conf(server)['home_path'] + for f in clean_files: + client.execute_command('rm -f {0}'.format(os.path.join(home_path, f))) + + # update port + generate_configs = {"global": {}} + original_global_config = cluster_config.get_original_global_conf() + port_keys = { + 'server_port': 'monagent_http_port', + 'pprof_port': 'mgragent_http_port' + } + port_warns = {} + for server in cluster_config.servers: + original_server_config = cluster_config.get_original_server_conf(server) + server_config = cluster_config.get_server_conf(server) + for port_key in port_keys: + if port_key in original_global_config or port_key in original_server_config: + port = server_config[port_key] + if server not in generate_configs: + generate_configs[server] = {} + generate_configs[server][port_keys[port_key]] = port + if port_key not in port_warns: + port_warns[port_key] = 'Configuration item {} is no longer supported, and it is converted to configuration item {}'.format(port_key, port_keys[port_key]) + if port_warns: + for msg in port_warns.values(): + stdio.warn(msg) + # merge_generate_config + merge_config = {} + generate_global_config = generate_configs['global'] + count_base = len(cluster_config.servers) - 1 + if count_base < 1: + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_global_config.update(generate_configs[server]) + generate_configs[server] = {} + else: + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + merged_server_config = {} + for key in generate_server_config: + if key in generate_global_config: + if generate_global_config[key] != generate_server_config[key]: + merged_server_config[key] = generate_server_config[key] + elif key in merge_config: + if merge_config[key]['value'] != generate_server_config[key]: + merged_server_config[key] = generate_server_config[key] + elif count_base == merge_config[key]['count']: + generate_global_config[key] = generate_server_config[key] + del merge_config[key] + else: + merge_config[key]['severs'].append(server) + merge_config[key]['count'] += 1 + else: + merge_config[key] = {'value': generate_server_config[key], 'severs': [server], 'count': 1} + generate_configs[server] = merged_server_config + + for key in merge_config: + config_st = merge_config[key] + for server in config_st['severs']: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + generate_server_config[key] = config_st['value'] + # summit_config + summit_config() + + apply_param_plugin(dest_repository) + if not call_plugin(start_plugin, plugin_context, [dest_repository], *args, **kwargs): + return + + ret = call_plugin(connect_plugin, plugin_context, [dest_repository], *args, **kwargs) + if ret and call_plugin(display_plugin, plugin_context, [dest_repository], ret.get_return('cursor'), *args, **kwargs): + upgrade_ctx['index'] = len(upgrade_repositories) + return plugin_context.return_true() diff --git a/plugins/obproxy/3.1.0/bootstrap.py b/plugins/obproxy/3.1.0/bootstrap.py index 4125841..5631878 100644 --- a/plugins/obproxy/3.1.0/bootstrap.py +++ b/plugins/obproxy/3.1.0/bootstrap.py @@ -29,14 +29,10 @@ def bootstrap(plugin_context, cursor, *args, **kwargs): server_config = cluster_config.get_server_conf(server) for key in ['observer_sys_password', 'obproxy_sys_password']: sql = 'alter proxyconfig set %s = %%s' % key - value = None - try: - value = server_config.get(key, '') - value = '' if value is None else str(value) - stdio.verbose('execute sql: %s' % (sql % value)) - cursor[server].execute(sql, [value]) - except: - stdio.exception('execute sql exception: %s' % (sql % (value))) + value = server_config.get(key, '') + value = '' if value is None else str(value) + ret = cursor[server].execute(sql, [value], exc_level="verbose") + if ret is False: stdio.error('failed to set %s for obproxy(%s)' % (key, server)) global_ret = False if global_ret: diff --git a/plugins/obproxy/3.1.0/connect.py b/plugins/obproxy/3.1.0/connect.py index 9cbb6c6..e4f2c62 100644 --- a/plugins/obproxy/3.1.0/connect.py +++ b/plugins/obproxy/3.1.0/connect.py @@ -22,43 +22,101 @@ import sys import time +from copy import copy if sys.version_info.major == 2: import MySQLdb as mysql else: import pymysql as mysql -from _errno import EC_FAIL_TO_CONNECT +from _errno import EC_FAIL_TO_CONNECT, EC_SQL_EXECUTE_FAILED +from _stdio import SafeStdio -stdio = None +class Cursor(SafeStdio): + def __init__(self, ip, port, user='root', tenant='sys', password='', stdio=None): + self.stdio = stdio + self.ip = ip + self.port = port + self._user = user + self.tenant = tenant + self.password = password + self.cursor = None + self.db = None + self._connect() + self._raise_exception = False + self._raise_cursor = None + + @property + def user(self): + if "@" in self._user: + return self._user + if self.tenant: + return "{}@{}".format(self._user, self.tenant) + else: + return self._user + + @property + def raise_cursor(self): + if self._raise_cursor: + return self._raise_cursor + raise_cursor = copy(self) + raise_cursor._raise_exception = True + self._raise_cursor = raise_cursor + return raise_cursor -def _connect(ip, port, user, password=''): - stdio.verbose('connect %s -P%s -u%s -p%s' % (ip, port, user, password)) if sys.version_info.major == 2: - db = mysql.connect(host=ip, user=user, port=int(port), passwd=str(password)) - cursor = db.cursor(cursorclass=mysql.cursors.DictCursor) + def _connect(self): + self.stdio.verbose('connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + self.db = mysql.connect(host=self.ip, user=self.user, port=int(self.port), passwd=str(self.password)) + self.cursor = self.db.cursor(cursorclass=mysql.cursors.DictCursor) else: - db = mysql.connect(host=ip, user=user, port=int(port), password=str(password), cursorclass=mysql.cursors.DictCursor) - cursor = db.cursor() - return db, cursor - - -def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - # stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - - -def connect(plugin_context, target_server=None, sys_root=True, *args, **kwargs): - global stdio + def _connect(self): + self.stdio.verbose('connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + self.db = mysql.connect(host=self.ip, user=self.user, port=int(self.port), password=str(self.password), + cursorclass=mysql.cursors.DictCursor) + self.cursor = self.db.cursor() + + def new_cursor(self, tenant='sys', user='root', password=''): + try: + return Cursor(ip=self.ip, port=self.port, user=user, tenant=tenant, password=password, stdio=self.stdio) + except: + self.stdio.exception('') + self.stdio.verbose('fail to connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + return None + + def execute(self, sql, args=None, execute_func=None, raise_exception=False, exc_level='error', stdio=None): + try: + stdio.verbose('execute sql: %s. args: %s' % (sql, args)) + self.cursor.execute(sql, args) + if not execute_func: + return self.cursor + return getattr(self.cursor, execute_func)() + except Exception as e: + getattr(stdio, exc_level)(EC_SQL_EXECUTE_FAILED.format(sql=sql)) + if raise_exception is None: + raise_exception = self._raise_exception + if raise_exception: + stdio.exception('') + raise e + return False + + def fetchone(self, sql, args=None, raise_exception=False, exc_level='error', stdio=None): + return self.execute(sql, args=args, execute_func='fetchone', raise_exception=raise_exception, exc_level=exc_level, stdio=stdio) + + def fetchall(self, sql, args=None, raise_exception=False, exc_level='error', stdio=None): + return self.execute(sql, args=args, execute_func='fetchall', raise_exception=raise_exception, exc_level=exc_level, stdio=stdio) + + def close(self): + if self.cursor: + self.cursor.close() + self.cursor = None + if self.db: + self.db.close() + self.db = None + + +def connect(plugin_context, target_server=None, connect_proxysys=True, *args, **kwargs): count = 10 cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio @@ -72,7 +130,7 @@ def connect(plugin_context, target_server=None, sys_root=True, *args, **kwargs): user = kwargs.get('user') password = kwargs.get('password') if not user: - if sys_root: + if connect_proxysys: user = 'root@proxysys' else: user = 'root' @@ -101,19 +159,17 @@ def connect(plugin_context, target_server=None, sys_root=True, *args, **kwargs): for server in servers: try: server_config = cluster_config.get_server_conf(server) - if sys_root: + if connect_proxysys: pwd_key = 'obproxy_sys_password' else: pwd_key = 'observer_root_password' r_password = password if password else server_config.get(pwd_key) if r_password is None: r_password = '' - db, cursor = _connect(server.ip, server_config['listen_port'], user, r_password if count % 2 else '') + cursor = Cursor(ip=server.ip, port=server_config['listen_port'], user=user, tenant='', password=r_password if count % 2 else '', stdio=stdio) if user in ['root', 'root@sys']: - stdio.verbose('execute sql: select * from information_schema.TABLES limit 1') - cursor.execute('select * from information_schema.TABLES limit 1') - stdio.verbose("result: {}".format(cursor.fetchone())) - dbs[server] = db + stdio.verbose("result: {}".format(cursor.fetchone('select * from information_schema.TABLES limit 1', raise_exception=True))) + dbs[server] = cursor.db cursors[server] = cursor except: tmp_servers.append(server) diff --git a/plugins/obproxy/3.1.0/display.py b/plugins/obproxy/3.1.0/display.py index 2310ca7..2e20633 100644 --- a/plugins/obproxy/3.1.0/display.py +++ b/plugins/obproxy/3.1.0/display.py @@ -20,6 +20,10 @@ from __future__ import absolute_import, division, print_function +def passwd_format(passwd): + return "'{}'".format(passwd.replace("'", "'\"'\"'")) + + def display(plugin_context, cursor, *args, **kwargs): stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config @@ -32,37 +36,46 @@ def display(plugin_context, cursor, *args, **kwargs): 'listen_port': '-', 'prometheus_listen_port': '-' } - try: - cursor[server].execute('show proxyconfig like "%port"') - for item in cursor[server].fetchall(): + res = cursor[server].fetchall('show proxyconfig like "%port"', exc_level='verbose') + if res: + for item in res: if item['name'] in data: data[item['name']] = item['value'] data['status'] = 'active' - except: - stdio.exception('') - pass + else: + continue result.append(data) - stdio.print_list(result, ['ip', 'port', 'prometheus_port', 'status'], + stdio.print_list(result, ['ip', 'port', 'prometheus_port', 'status'], lambda x: [x['ip'], x['listen_port'], x['prometheus_listen_port'], x['status']], title='obproxy') - server = servers[0] with_observer = False server_config = cluster_config.get_server_conf(server) cmd = '' + info_dict = { + "type": "db", + "ip": server.ip, + "port": server_config['listen_port'] + } for comp in ['oceanbase', 'oceanbase-ce']: if comp in cluster_config.depends: ob_config = cluster_config.get_depend_config(comp) if not ob_config: continue + user = 'root' password = ob_config.get('root_password', '') with_observer = True - cmd = 'obclient -h%s -P%s -uroot %s-Doceanbase -A' % (server.ip, server_config['listen_port'], '-p%s ' % password if password else '') + info_dict['user'] = user + info_dict['password'] = password + cmd = 'obclient -h%s -P%s -u%s %s-Doceanbase -A' % (server.ip, server_config['listen_port'], user, '-p%s ' % passwd_format(password) if password else '') break if not with_observer: + user = 'root@proxysys' password = server_config.get('obproxy_sys_password', '') - cmd = 'obclient -h%s -P%s -uroot@proxysys %s-Doceanbase -A' % (server.ip, server_config['listen_port'], '-p%s ' % password if password else '') + info_dict['user'] = user + info_dict['password'] = password + cmd = 'obclient -h%s -P%s -u%s %s-Doceanbase -A' % (server.ip, server_config['listen_port'], user, '-p%s ' % passwd_format(password) if password else '') stdio.print(cmd) - - plugin_context.return_true() + info_dict['cmd'] = cmd + plugin_context.return_true(info=info_dict) diff --git a/plugins/obproxy/3.1.0/generate_config.py b/plugins/obproxy/3.1.0/generate_config.py index 08def59..1705979 100644 --- a/plugins/obproxy/3.1.0/generate_config.py +++ b/plugins/obproxy/3.1.0/generate_config.py @@ -21,65 +21,39 @@ from __future__ import absolute_import, division, print_function -def generate_config(plugin_context, deploy_config, auto_depend=False, *args, **kwargs): +def generate_config(plugin_context, generate_config_mini=False, auto_depend=False, return_generate_keys=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=['skip_proxy_sys_private_check', 'enable_strict_kernel_release', 'enable_cluster_checkout', 'proxy_mem_limited']) + cluster_config = plugin_context.cluster_config - clients = plugin_context.clients stdio = plugin_context.stdio - success = True + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) stdio.start_loading('Generate obproxy configuration') - for server in cluster_config.servers: - server_config = cluster_config.get_server_conf(server) - if not server_config.get('home_path'): - stdio.error("obproxy %s: missing configuration 'home_path' in configuration file" % server) - success = False - continue - cluster_config.update_server_conf(server, 'enable_cluster_checkout', False) - if not success: - stdio.stop_loading('fail') - return - global_config = cluster_config.get_original_global_conf() if 'skip_proxy_sys_private_check' not in global_config: + generate_configs['global']['skip_proxy_sys_private_check'] = True cluster_config.update_global_conf('skip_proxy_sys_private_check', True, False) + if 'enable_strict_kernel_release' not in global_config: + generate_configs['global']['enable_strict_kernel_release'] = False cluster_config.update_global_conf('enable_strict_kernel_release', False, False) + + if 'enable_cluster_checkout' not in global_config: + generate_configs['global']['enable_cluster_checkout'] = False + cluster_config.update_global_conf('enable_cluster_checkout', False, False) - if getattr(plugin_context.options, 'mini', False): + if generate_config_mini: if 'proxy_mem_limited' not in global_config: + generate_configs['global']['proxy_mem_limited'] = '500M' cluster_config.update_global_conf('proxy_mem_limited', '500M', False) - ob_comps = ['oceanbase', 'oceanbase-ce'] - ob_cluster_config = None - for comp in ob_comps: - if comp in cluster_config.depends: - stdio.stop_loading('succeed') - return plugin_context.return_true() - if comp in deploy_config.components: - ob_cluster_config = deploy_config.components[comp] - if auto_depend: for depend in ['oceanbase', 'oceanbase-ce']: if cluster_config.add_depend_component(depend): stdio.stop_loading('succeed') return plugin_context.return_true() - if ob_cluster_config: - root_servers = {} - cluster_name = ob_cluster_config.get_global_conf().get('appname') - for server in ob_cluster_config.servers: - config = ob_cluster_config.get_server_conf_with_default(server) - zone = config['zone'] - cluster_name = cluster_name if cluster_name else config.get('appname') - if zone not in root_servers: - root_servers[zone] = '%s:%s' % (server.ip, config['mysql_port']) - rs_list = ';'.join([root_servers[zone] for zone in root_servers]) - - cluster_name = cluster_name if cluster_name else 'obcluster' - if not global_config.get('rs_list'): - cluster_config.update_global_conf('rs_list', rs_list, False) - if not global_config.get('cluster_name'): - cluster_config.update_global_conf('cluster_name', cluster_name, False) - stdio.stop_loading('succeed') return plugin_context.return_true() \ No newline at end of file diff --git a/plugins/obproxy/3.1.0/init.py b/plugins/obproxy/3.1.0/init.py index 1ea7ee9..ae019a9 100644 --- a/plugins/obproxy/3.1.0/init.py +++ b/plugins/obproxy/3.1.0/init.py @@ -22,7 +22,7 @@ from _errno import EC_FAIL_TO_INIT_PATH, InitDirFailedErrorMessage -def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def init(plugin_context, *args, **kwargs): cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio @@ -35,8 +35,6 @@ def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): server_config = cluster_config.get_server_conf(server) client = clients[server] home_path = server_config['home_path'] - remote_home_path = client.execute_command('echo ${OBD_HOME:-"$HOME"}/.obd').stdout.strip() - remote_repository_dir = repository_dir.replace(local_home_path, remote_home_path) stdio.verbose('%s init cluster work home', server) need_clean = force if clean and not force: diff --git a/plugins/obproxy/3.1.0/parameter.yaml b/plugins/obproxy/3.1.0/parameter.yaml index 0f40a1b..37c7576 100644 --- a/plugins/obproxy/3.1.0/parameter.yaml +++ b/plugins/obproxy/3.1.0/parameter.yaml @@ -1,11 +1,15 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING - need_restart: true + need_redeploy: true description_en: the directory for the work data file description_local: ObProxy工作目录 - name: listen_port + name_local: 服务端口 require: true + essential: true type: INT default: 2883 min_value: 1025 @@ -14,7 +18,9 @@ description_en: port number for mysql connection description_local: SQL服务协议端口号 - name: prometheus_listen_port + name_local: Exporter 端口 require: true + essential: true type: INT default: 2884 min_value: 1025 @@ -39,6 +45,15 @@ need_restart: true description_en: root server list(format ip:sql_port) description_local: observer列表(格式 ip:sql_port) +- name: proxy_mem_limited + name_local: 最大运行内存 + essential: true + type: CAPACITY + default: 2G + min_value: 100MB + max_value: 100GB + description_en: The upper limit of ODP runtime memory. If the ODP exceeds the upper limit, it will exit automatically. Please enter an capacity, such as 2G + description_local: ODP 运行时内存上限。超过上限 ODP 即自动退出。请输入带容量带单位的整数,如2G - name: refresh_json_config type: BOOL default: false @@ -423,6 +438,8 @@ need_restart: true description_en: beyond trust sdk retry times - name: obproxy_sys_password + name_local: 密码 + essential: true type: STRING default: '' need_restart: false diff --git a/plugins/obproxy/3.1.0/reload.py b/plugins/obproxy/3.1.0/reload.py index 291a55a..1a2acc8 100644 --- a/plugins/obproxy/3.1.0/reload.py +++ b/plugins/obproxy/3.1.0/reload.py @@ -32,7 +32,8 @@ def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): config_map = { 'observer_sys_password': 'proxyro_password', - 'cluster_name': 'appname' + 'cluster_name': 'appname', + 'observer_root_password': 'root_password' } for comp in ['oceanbase', 'oceanbase-ce']: if comp in cluster_config.depends: @@ -54,7 +55,10 @@ def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): stdio.verbose('get %s cluster address' % (server)) cluster_server[server] = '%s:%s' % (server.ip, config['listen_port']) stdio.verbose('compare configuration of %s' % (server)) + reload_unused = ['observer_root_password'] for key in new_config: + if key in reload_unused: + continue if key not in config or config[key] != new_config[key]: item = cluster_config.get_temp_conf_item(key) if item: @@ -85,15 +89,12 @@ def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): for server in servers: if key not in change_conf[server]: continue - try: - sql = 'alter proxyconfig set %s = %%s' % key - value = change_conf[server][key] if change_conf[server].get(key) is not None else '' - stdio.verbose('execute sql: %s' % (sql % value)) - cursor[server].execute(sql, [value]) - success_conf[key].append(server) - except: + sql = 'alter proxyconfig set %s = %%s' % key + value = change_conf[server][key] if change_conf[server].get(key) is not None else '' + if cursor[server].execute(sql, [value]) is False: global_ret = False - stdio.exception('execute sql exception: %s' % (sql % value)) + continue + success_conf[key].append(server) for key in success_conf: if global_change_conf[key] == servers_num == len(success_conf): cluster_config.update_global_conf(key, value, False) diff --git a/plugins/obproxy/3.1.0/restart.py b/plugins/obproxy/3.1.0/restart.py index 0a646e6..c550fad 100644 --- a/plugins/obproxy/3.1.0/restart.py +++ b/plugins/obproxy/3.1.0/restart.py @@ -28,11 +28,22 @@ class Restart(object): def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, bootstrap_plugin=None): self.local_home_path = local_home_path - self.plugin_context = plugin_context + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + self.components = plugin_context.components self.clients = plugin_context.clients self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context self.repository = repository self.start_plugin = start_plugin self.reload_plugin = reload_plugin @@ -45,20 +56,29 @@ def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, self.sub_io = self.stdio.sub_io() self.dbs = None self.cursors = None - - # def close(self): - # if self.dbs: - # for server in self.cursors: - # self.cursors[server].close() - # for db in self.dbs: - # self.dbs[server].close() - # self.cursors = None - # self.dbs = None + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) def connect(self, cluster_config): - self.stdio.verbose('Call %s for %s' % (self.connect_plugin, self.repository)) self.sub_io.start_loading('Connect to obproxy') - ret = self.connect_plugin(self.components, self.clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + ret = self.call_plugin(self.connect_plugin, cluster_config=cluster_config) if not ret: self.sub_io.stop_loading('fail') return False @@ -79,18 +99,15 @@ def restart(self): if self.new_cluster_config: if not self.connect(self.cluster_config): return False - self.stdio.verbose('Call %s for %s' % (self.reload_plugin, self.repository)) - self.reload_plugin(self.components, self.clients, self.cluster_config, [], {}, self.sub_io, cursor=self.cursors, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir) + self.call_plugin(self.reload_plugin, clients=clients, cursor=self.cursors, new_cluster_config=self.new_cluster_config) - self.stdio.verbose('Call %s for %s' % (self.stop_plugin, self.repository)) - if not self.stop_plugin(self.components, clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io): + if not self.call_plugin(self.stop_plugin, clients=clients): self.stdio.stop_loading('stop_loading', 'fail') return False if self.new_clients: self.stdio.verbose('use new clients') for server in self.cluster_config.servers: - client = clients[server] new_client = self.new_clients[server] server_config = self.cluster_config.get_server_conf(server) home_path = server_config['home_path'] @@ -101,25 +118,22 @@ def restart(self): clients = self.new_clients cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config - self.stdio.verbose('Call %s for %s' % (self.start_plugin, self.repository)) need_bootstrap = self.bootstrap_plugin is not None - if not self.start_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, local_home_path=self.local_home_path, repository_dir=self.repository.repository_dir, need_bootstrap=need_bootstrap): + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, repository=self.repository, need_bootstrap=need_bootstrap): self.rollback() self.stdio.stop_loading('stop_loading', 'fail') return False if self.connect(cluster_config): if self.bootstrap_plugin: - self.stdio.verbose('Call %s for %s' % (self.bootstrap_plugin, self.repository)) - self.bootstrap_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, cursor=self.cursors) - self.stdio.verbose('Call %s for %s' % (self.display_plugin, self.repository)) - ret = self.display_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, cursor=self.cursors) - return ret + self.call_plugin(self.bootstrap_plugin, cursor=self.cursors) + return self.call_plugin(self.display_plugin, cursor=self.cursors) return False def rollback(self): if self.new_clients: - self.stop_plugin(self.components, self.new_clients, self.new_cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.new_clients, cluster_config=cluster_config) for server in self.cluster_config.servers: client = self.clients[server] new_client = self.new_clients[server] @@ -128,7 +142,8 @@ def rollback(self): new_client.execute_command('sudo chown -R %s: %s' % (client.config.username, home_path)) -def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, *args, **kwargs): +def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, *args, **kwargs): + repository = kwargs.get('repository') task = Restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config, new_clients, bootstrap_plugin=bootstrap_plugin) call = task.rollback if rollback else task.restart if call(): diff --git a/plugins/obproxy/3.1.0/start.py b/plugins/obproxy/3.1.0/start.py index 8135266..a3bc5c7 100644 --- a/plugins/obproxy/3.1.0/start.py +++ b/plugins/obproxy/3.1.0/start.py @@ -24,12 +24,14 @@ import time from copy import deepcopy +from _errno import EC_CONFLICT_PORT + stdio = None def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -106,7 +108,7 @@ def __exit__(self, *args, **kwargs): self.client.del_env(env_key) -def start(plugin_context, local_home_path, repository_dir, need_bootstrap=False, *args, **kwargs): +def start(plugin_context, need_bootstrap=False, *args, **kwargs): global stdio cluster_config = plugin_context.cluster_config clients = plugin_context.clients @@ -214,7 +216,7 @@ def start(plugin_context, local_home_path, repository_dir, need_bootstrap=False, if confirm_port(client, remote_pid, port): continue stdio.stop_loading('fail') - stdio.error('%s:%s port is already used' % (server.ip, port)) + stdio.error(EC_CONFLICT_PORT.format(server=server.ip, port=port)) return plugin_context.return_false() stdio.verbose('starting %s obproxy', server) diff --git a/plugins/obproxy/3.1.0/start_check.py b/plugins/obproxy/3.1.0/start_check.py index 68b2189..bca4c0c 100644 --- a/plugins/obproxy/3.1.0/start_check.py +++ b/plugins/obproxy/3.1.0/start_check.py @@ -20,7 +20,8 @@ from __future__ import absolute_import, division, print_function -from _errno import EC_CONFIG_CONFLICT_PORT +import os +import _errno as err stdio = None @@ -29,7 +30,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -37,43 +38,115 @@ def get_port_socket_inode(client, port): return res.stdout.strip().split('\n') -def start_check(plugin_context, strict_check=False, *args, **kwargs): - def alert(*arg, **kwargs): +def start_check(plugin_context, init_check_status=False, strict_check=False, work_dir_check=False, work_dir_empty_check=True, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def alert(item, error, suggests=[]): global success if strict_check: success = False - stdio.error(*arg, **kwargs) + check_fail(item, error, suggests) + stdio.error(error) else: - stdio.warn(*arg, **kwargs) - def error(*arg, **kwargs): - global success - if plugin_context.dev_mode: - stdio.warn(*arg, **kwargs) - else: - success = False - stdio.error(*arg, **kwargs) - def critical(*arg, **kwargs): + stdio.warn(error) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) - global stdio + check_fail(item, error, suggests) + stdio.error(error) + + global stdio, success + success = True cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio servers_port = {} + check_status = {} + servers_dirs = {} + servers_check_dirs = {} + + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + stdio.start_loading('Check before start obproxy') for server in cluster_config.servers: ip = server.ip client = clients[server] server_config = cluster_config.get_server_conf(server) port = int(server_config["listen_port"]) - prometheus_port = int(server_config["prometheus_listen_port"]) - remote_pid_path = "%s/run/obproxy-%s-%s.pid" % (server_config['home_path'], server.ip, server_config["listen_port"]) - remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s/fd' % remote_pid): - continue + if not precheck: + remote_pid_path = "%s/run/obproxy-%s-%s.pid" % (server_config['home_path'], server.ip, server_config["listen_port"]) + remote_pid = client.execute_command("cat %s" % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s/fd' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + key = 'home_path' + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False + if ip not in servers_port: servers_port[ip] = {} ports = servers_port[ip] @@ -83,15 +156,26 @@ def critical(*arg, **kwargs): port = int(server_config[key]) alert_f = alert if key == 'prometheus_listen_port' else critical if port in ports: - alert_f(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key'])) + alert_f( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) continue ports[port] = { 'server': server, 'key': key } if get_port_socket_inode(client, port): - alert_f('%s:%s port is already used' % (ip, port)) - + alert_f( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + + for server in cluster_config.servers: + wait_2_pass() + if success: stdio.stop_loading('succeed') plugin_context.return_true() diff --git a/plugins/obproxy/3.1.0/stop.py b/plugins/obproxy/3.1.0/stop.py index 0a748c6..169960d 100644 --- a/plugins/obproxy/3.1.0/stop.py +++ b/plugins/obproxy/3.1.0/stop.py @@ -28,7 +28,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) inode = res.stdout.strip() if not res or not inode: diff --git a/plugins/obproxy/3.1.0/upgrade.py b/plugins/obproxy/3.1.0/upgrade.py index af4ea3e..eec38ae 100644 --- a/plugins/obproxy/3.1.0/upgrade.py +++ b/plugins/obproxy/3.1.0/upgrade.py @@ -22,11 +22,18 @@ def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, *args, **kwargs): + namespace = plugin_context.namespace + namespaces = plugin_context.namespaces + deploy_name = plugin_context.deploy_name + repositories = plugin_context.repositories + plugin_name = plugin_context.plugin_name + components = plugin_context.components clients = plugin_context.clients cluster_config = plugin_context.cluster_config - cmd = plugin_context.cmd + cmds = plugin_context.cmds options = plugin_context.options + dev_mode = plugin_context.dev_mode stdio = plugin_context.stdio upgrade_ctx = kwargs.get('upgrade_ctx') @@ -45,15 +52,15 @@ def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, *args, bootstrap_plugin = search_py_script_plugin([dest_repository], 'bootstrap')[dest_repository] apply_param_plugin(cur_repository) - if not stop_plugin(components, clients, cluster_config, cmd, options, stdio, *args, **kwargs): + if not stop_plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, stdio, *args, **kwargs): return apply_param_plugin(dest_repository) - if not start_plugin(components, clients, cluster_config, cmd, options, stdio, need_bootstrap=True, *args, **kwargs): + if not start_plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, stdio, need_bootstrap=True, *args, **kwargs): return - ret = connect_plugin(components, clients, cluster_config, cmd, options, stdio, *args, **kwargs) + ret = connect_plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, stdio, *args, **kwargs) if ret: - if bootstrap_plugin(components, clients, cluster_config, cmd, options, stdio, ret.get_return('cursor'), *args, **kwargs) and display_plugin(components, clients, cluster_config, cmd, options, stdio, ret.get_return('cursor'), *args, **kwargs): + if bootstrap_plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, stdio, ret.get_return('cursor'), *args, **kwargs) and display_plugin(namespace, namespaces, deploy_name, repositories, components, clients, cluster_config, cmds, options, stdio, ret.get_return('cursor'), *args, **kwargs): upgrade_ctx['index'] = len(upgrade_repositories) return plugin_context.return_true() diff --git a/plugins/oceanbase/3.1.0/bootstrap.py b/plugins/oceanbase/3.1.0/bootstrap.py index 2f30640..1070ac4 100644 --- a/plugins/oceanbase/3.1.0/bootstrap.py +++ b/plugins/oceanbase/3.1.0/bootstrap.py @@ -20,7 +20,6 @@ from __future__ import absolute_import, division, print_function -import time from _deploy import InnerConfigItem @@ -38,7 +37,6 @@ def bootstrap(plugin_context, cursor, *args, **kwargs): if componet_name in plugin_context.components: has_obproxy = True break - inner_keys = inner_config.keys() for server in cluster_config.servers: server_config = cluster_config.get_server_conf(server) zone = server_config['zone'] @@ -59,49 +57,39 @@ def bootstrap(plugin_context, cursor, *args, **kwargs): continue zone_config[key] = server_config[key] try: + raise_cursor = cursor.raise_cursor sql = 'set session ob_query_timeout=1000000000' - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + raise_cursor.execute(sql) sql = 'alter system bootstrap %s' % (','.join(bootstrap)) stdio.start_loading('Cluster bootstrap') - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + raise_cursor.execute(sql, exc_level='verbose') for zone in floor_servers: for addr in floor_servers[zone]: sql = 'alter system add server "%s" zone "%s"' % (addr, zone) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + raise_cursor.execute(sql) global_conf = cluster_config.get_global_conf() if has_obproxy or 'proxyro_password' in global_conf: value = global_conf['proxyro_password'] if global_conf.get('proxyro_password') is not None else '' sql = 'create user "proxyro" IDENTIFIED BY %s' - stdio.verbose(sql) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) sql = 'grant select on oceanbase.* to proxyro IDENTIFIED BY %s' - stdio.verbose(sql) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) + if global_conf.get('root_password') is not None: - sql = 'alter user "root" IDENTIFIED BY "%s"' % global_conf.get('root_password') - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + sql = 'alter user "root" IDENTIFIED BY %s' + raise_cursor.execute(sql, [global_conf.get('root_password')]) for zone in zones_config: zone_config = zones_config[zone] for key in zone_config: sql = 'alter system modify zone %s set %s = %%s' % (zone, inner_config[key]) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql, [zone_config[key]]) + raise_cursor.execute(sql, [zone_config[key]]) stdio.stop_loading('succeed') - plugin_context.return_true() except: - stdio.exception('') - try: - cursor.execute('select * from oceanbase.__all_rootservice_event_history where module = "bootstrap" and event = "bootstrap_succeed"') - event = cursor.fetchall() - if not event: - raise Exception('Not found bootstrap_succeed event') - stdio.stop_loading('succeed') - plugin_context.return_true() - except: + event = cursor.fetchall('select * from oceanbase.__all_rootservice_event_history where module = "bootstrap" and event = "bootstrap_succeed"') + if not event: stdio.stop_loading('fail') - stdio.exception('') - plugin_context.return_false() + return plugin_context.return_false() + stdio.stop_loading('succeed') + return plugin_context.return_true() + + return plugin_context.return_true() diff --git a/plugins/oceanbase/3.1.0/connect.py b/plugins/oceanbase/3.1.0/connect.py index 289dc26..893b414 100644 --- a/plugins/oceanbase/3.1.0/connect.py +++ b/plugins/oceanbase/3.1.0/connect.py @@ -22,31 +22,102 @@ import sys import time +from copy import copy if sys.version_info.major == 2: import MySQLdb as mysql else: import pymysql as mysql -from _errno import EC_FAIL_TO_CONNECT +from _errno import EC_FAIL_TO_CONNECT, EC_SQL_EXECUTE_FAILED +from _stdio import SafeStdio -stdio = None +class Cursor(SafeStdio): + def __init__(self, ip, port, user='root', tenant='sys', password='', stdio=None): + self.stdio = stdio + self.ip = ip + self.port = port + self._user = user + self.tenant = tenant + self.password = password + self.cursor = None + self.db = None + self._connect() + self._raise_exception = False + self._raise_cursor = None + + @property + def user(self): + if "@" in self._user: + return self._user + if self.tenant: + return "{}@{}".format(self._user, self.tenant) + else: + return self._user + + @property + def raise_cursor(self): + if self._raise_cursor: + return self._raise_cursor + raise_cursor = copy(self) + raise_cursor._raise_exception = True + self._raise_cursor = raise_cursor + return raise_cursor -def _connect(ip, port, password=''): - user = 'root' - stdio.verbose('connect %s -P%s -u%s -p%s' % (ip, port, user, password)) if sys.version_info.major == 2: - db = mysql.connect(host=ip, user=user, port=int(port), passwd=str(password)) - cursor = db.cursor(cursorclass=mysql.cursors.DictCursor) + def _connect(self): + self.stdio.verbose('connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + self.db = mysql.connect(host=self.ip, user=self.user, port=int(self.port), passwd=str(self.password)) + self.cursor = self.db.cursor(cursorclass=mysql.cursors.DictCursor) else: - db = mysql.connect(host=ip, user=user, port=int(port), password=str(password), cursorclass=mysql.cursors.DictCursor) - cursor = db.cursor() - return db, cursor + def _connect(self): + self.stdio.verbose('connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + self.db = mysql.connect(host=self.ip, user=self.user, port=int(self.port), password=str(self.password), + cursorclass=mysql.cursors.DictCursor) + self.cursor = self.db.cursor() + + def new_cursor(self, tenant='sys', user='root', password=''): + try: + return Cursor(ip=self.ip, port=self.port, user=user, tenant=tenant, password=password, stdio=self.stdio) + except: + self.stdio.exception('') + self.stdio.verbose('fail to connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + return None + + def execute(self, sql, args=None, execute_func=None, raise_exception=None, exc_level='error', stdio=None): + + try: + stdio.verbose('execute sql: %s. args: %s' % (sql, args)) + self.cursor.execute(sql, args) + if not execute_func: + return self.cursor + return getattr(self.cursor, execute_func)() + except Exception as e: + getattr(stdio, exc_level)(EC_SQL_EXECUTE_FAILED.format(sql=sql)) + if raise_exception is None: + raise_exception = self._raise_exception + if raise_exception: + stdio.exception('') + raise e + return False + + def fetchone(self, sql, args=None, raise_exception=None, exc_level='error', stdio=None): + return self.execute(sql, args=args, execute_func='fetchone', raise_exception=raise_exception, exc_level=exc_level, stdio=stdio) + + def fetchall(self, sql, args=None, raise_exception=None, exc_level='error', stdio=None): + return self.execute(sql, args=args, execute_func='fetchall', raise_exception=raise_exception, exc_level=exc_level, stdio=stdio) + + def close(self): + if self.cursor: + self.cursor.close() + self.cursor = None + if self.db: + self.db.close() + self.db = None def connect(plugin_context, target_server=None, *args, **kwargs): - global stdio - count = 10 + count = 100 cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio if target_server: @@ -62,13 +133,14 @@ def connect(plugin_context, target_server=None, *args, **kwargs): try: server_config = cluster_config.get_server_conf(server) password = server_config.get('root_password', '') if count % 2 else '' - db, cursor = _connect(server.ip, server_config['mysql_port'], password if password is not None else '') + cursor = Cursor(ip=server.ip, port=server_config['mysql_port'], tenant='', password=password if password is not None else '', stdio=stdio) stdio.stop_loading('succeed') - return plugin_context.return_true(connect=db, cursor=cursor, server=server) + return plugin_context.return_true(connect=cursor.db, cursor=cursor, server=server) except: - stdio.exception('') + if count == 0: + stdio.exception('') time.sleep(3) - + stdio.stop_loading('fail') stdio.error(EC_FAIL_TO_CONNECT.format(component=cluster_config.name)) plugin_context.return_false() diff --git a/plugins/oceanbase/3.1.0/create_tenant.py b/plugins/oceanbase/3.1.0/create_tenant.py index f3e16d4..e828234 100644 --- a/plugins/oceanbase/3.1.0/create_tenant.py +++ b/plugins/oceanbase/3.1.0/create_tenant.py @@ -26,6 +26,8 @@ from _errno import EC_OBSERVER_CAN_NOT_MIGRATE_IN +tenant_cursor = None + def parse_size(size): _bytes = 0 @@ -55,8 +57,22 @@ def format_size(size, precision=1): return format % (size, units[idx]) -def create_tenant(plugin_context, cursor, *args, **kwargs): +def exec_sql_in_tenant(sql, cursor, tenant, mode, retries=10): + global tenant_cursor + if not tenant_cursor: + user = 'SYS' if mode == 'oracle' else 'root' + tenant_cursor = cursor.new_cursor(tenant=tenant, user=user) + if not tenant_cursor and retries: + retries -= 1 + time.sleep(2) + return exec_sql_in_tenant(sql, cursor, tenant, mode, retries) + return tenant_cursor.execute(sql) + + +def create_tenant(plugin_context, cursor, create_tenant_options=None, *args, **kwargs): def get_option(key, default=''): + if key in kwargs: + return kwargs[key] value = getattr(options, key, default) if not value: value = default @@ -74,13 +90,12 @@ def get_parsed_option(key, default=''): def error(*arg, **kwargs): stdio.error(*arg, **kwargs) stdio.stop_loading('fail') - def exception(*arg, **kwargs): - stdio.exception(*arg, **kwargs) - stdio.stop_loading('fail') cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio - options = plugin_context.options + options = create_tenant_options if create_tenant_options else plugin_context.options + create_if_not_exists = get_option('create_if_not_exists', False) + tenant_exists = False mode = get_option('mode', 'mysql').lower() if not mode in ['mysql', 'oracle']: @@ -90,225 +105,227 @@ def exception(*arg, **kwargs): name = get_option('tenant_name', 'test') unit_name = '%s_unit' % name pool_name = '%s_pool' % name - - stdio.start_loading('Create tenant %s' % name) - sql = "select tenant_name from oceanbase.gv$tenant where tenant_name = %s" - try: - stdio.verbose('execute sql: %s' % (sql % name)) - cursor.execute(sql, [name]) - if cursor.fetchone(): + sql = "select tenant_name from oceanbase.gv$tenant where tenant_name = '%s'" % name + res = cursor.fetchone(sql) + if res: + if create_if_not_exists: + tenant_exists = True + else: error('Tenant %s already exists' % name) return - except: - exception('execute sql exception: %s' % (sql % name)) + elif res is False: return - - zone_list = get_option('zone_list', set()) - zone_obs_num = {} - sql = "select zone, count(*) num from oceanbase.__all_server where status = 'active' group by zone" - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - res = cursor.fetchall() + if not tenant_exists: + stdio.start_loading('Create tenant %s' % name) + zone_list = get_option('zone_list', set()) + zone_obs_num = {} + sql = "select zone, count(*) num from oceanbase.__all_server where status = 'active' group by zone" + res = cursor.fetchall(sql) + if res is False: + stdio.stop_loading('fail') + return for row in res: zone_obs_num[str(row['zone'])] = row['num'] - except: - exception('execute sql exception: %s' % sql) - return - if not zone_list: - zone_list = zone_obs_num.keys() - if isinstance(zone_list, str): - zones = zone_list.replace(';', ',').split(',') - else: - zones = zone_list - zone_list = "('%s')" % "','".join(zones) - - min_unit_num = min(zone_obs_num.items(),key=lambda x: x[1])[1] - unit_num = get_option('unit_num', min_unit_num) - if unit_num > min_unit_num: - return error('resource pool unit num is bigger than zone server count') - - sql = "select count(*) num from oceanbase.__all_server where status = 'active' and start_service_time > 0" - try: + if not zone_list: + zone_list = zone_obs_num.keys() + if isinstance(zone_list, str): + zones = zone_list.replace(';', ',').split(',') + else: + zones = zone_list + zone_list = "('%s')" % "','".join(zones) + + min_unit_num = min(zone_obs_num.items(), key=lambda x: x[1])[1] + unit_num = get_option('unit_num', min_unit_num) + if unit_num > min_unit_num: + return error('resource pool unit num is bigger than zone server count') + + sql = "select count(*) num from oceanbase.__all_server where status = 'active' and start_service_time > 0" count = 30 - while count: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - num = cursor.fetchone()['num'] - if num >= unit_num: - break - count -= 1 - time.sleep(1) - if count == 0: - stdio.error(EC_OBSERVER_CAN_NOT_MIGRATE_IN) + try: + while count: + num = cursor.fetchone(sql, raise_exception=True)['num'] + if num >= unit_num: + break + count -= 1 + time.sleep(1) + if count == 0: + stdio.error(EC_OBSERVER_CAN_NOT_MIGRATE_IN) + return + except: + stdio.stop_loading('fail') return - except: - exception('execute sql exception: %s' % sql) - return - - cpu_total = 0 - mem_total = 0 - disk_total = 0 - sql = "SELECT min(cpu_total) cpu_total, min(mem_total) mem_total, min(disk_total) disk_total FROM oceanbase.__all_virtual_server_stat where zone in %s" - try: - sql = sql % zone_list - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) - return - resource = cursor.fetchone() - cpu_total = resource['cpu_total'] - mem_total = resource['mem_total'] - disk_total = resource['disk_total'] - - sql = 'select * from oceanbase.__all_resource_pool order by name' - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) - return - - units_id = {} - res = cursor.fetchall() - for row in res: - if str(row['name']) == unit_name: - unit_name += '1' - if row['tenant_id'] < 1: - continue - for zone in str(row['zone_list']).replace(';', ',').split(','): - if zone in zones: - unit_config_id = row['unit_config_id'] - units_id[unit_config_id] = units_id.get(unit_config_id, 0) + 1 - break - - sql = 'select * from oceanbase.__all_unit_config order by name' - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) - return - - res = cursor.fetchall() - for row in res: - if str(row['name']) == unit_name: - unit_name += '1' - if row['unit_config_id'] in units_id: - cpu_total -= row['max_cpu'] * units_id[row['unit_config_id']] - mem_total -= row['max_memory'] * units_id[row['unit_config_id']] - # disk_total -= row['max_disk_size'] - - MIN_CPU = 2 - MIN_MEMORY = 1073741824 - MIN_DISK_SIZE = 536870912 - MIN_IOPS = 128 - MIN_SESSION_NUM = 64 - if cpu_total < MIN_CPU: - return error('%s: resource not enough: cpu count less than %s' % (zone_list, MIN_CPU)) - if mem_total < MIN_MEMORY: - return error('%s: resource not enough: memory less than %s' % (zone_list, format_size(MIN_MEMORY))) - if disk_total < MIN_DISK_SIZE: - return error('%s: resource not enough: disk space less than %s' % (zone_list, format_size(MIN_DISK_SIZE))) - - try: - max_memory = get_parsed_option('max_memory', mem_total) - max_disk_size = get_parsed_option('max_disk_size', disk_total) - min_memory = get_parsed_option('min_memory', max_memory) - except Exception as e: - error(e) - return - max_cpu = get_option('max_cpu', cpu_total) - max_iops = get_option('max_iops', MIN_IOPS) - max_session_num = get_option('max_session_num', MIN_SESSION_NUM) - min_cpu = get_option('min_cpu', max_cpu) - min_iops = get_option('min_iops', max_iops) - - if cpu_total < max_cpu: - return error('resource not enough: cpu (Avail: %s, Need: %s)' % (cpu_total, max_cpu)) - if mem_total < max_memory: - return error('resource not enough: memory (Avail: %s, Need: %s)' % (format_size(mem_total), format_size(max_memory))) - if disk_total < max_disk_size: - return error('resource not enough: disk space (Avail: %s, Need: %s)' % (format_size(disk_total), format_size(max_disk_size))) - - if max_iops < MIN_IOPS: - return error('max_iops must greater than %d' % MIN_IOPS) - if max_session_num < MIN_SESSION_NUM: - return error('max_session_num must greater than %d' % MIN_SESSION_NUM) - - if max_cpu < min_cpu: - return error('min_cpu must less then max_cpu') - if max_memory < min_memory: - return error('min_memory must less then max_memory') - if max_iops < min_iops: - return error('min_iops must less then max_iops') - - - zone_num = len(zones) - charset = get_option('charset', '') - collate = get_option('collate', '') - replica_num = get_option('replica_num', zone_num) - logonly_replica_num = get_option('logonly_replica_num', 0) - tablegroup = get_option('tablegroup', '') - primary_zone = get_option('primary_zone', 'RANDOM') - locality = get_option('locality', '') - variables = get_option('variables', '') - - if replica_num == 0: - replica_num = zone_num - elif replica_num > zone_num: - return error('replica_num cannot be greater than zone num (%s)' % zone_num) - if not primary_zone: - primary_zone = 'RANDOM' - if logonly_replica_num > replica_num: - return error('logonly_replica_num cannot be greater than replica_num (%s)' % replica_num) - - # create resource unit - sql = 'create resource unit %s max_cpu %.1f, max_memory %d, max_iops %d, max_disk_size %d, max_session_num %d, min_cpu %.1f, min_memory %d, min_iops %d' - try: + cpu_total = 0 + mem_total = 0 + disk_total = 0 + sql = "SELECT min(cpu_total) cpu_total, min(mem_total) mem_total, min(disk_total) disk_total FROM oceanbase.__all_virtual_server_stat where zone in %s" % zone_list + resource = cursor.fetchone(sql) + if resource is False: + stdio.stop_loading('fail') + return + cpu_total = resource['cpu_total'] + mem_total = resource['mem_total'] + disk_total = resource['disk_total'] + + sql = 'select * from oceanbase.__all_resource_pool order by name' + + units_id = {} + res = cursor.fetchall(sql) + if res is False: + stdio.stop_loading('fail') + return + for row in res: + if str(row['name']) == unit_name: + unit_name += '1' + if row['tenant_id'] < 1: + continue + for zone in str(row['zone_list']).replace(';', ',').split(','): + if zone in zones: + unit_config_id = row['unit_config_id'] + units_id[unit_config_id] = units_id.get(unit_config_id, 0) + 1 + break + + sql = 'select * from oceanbase.__all_unit_config order by name' + res = cursor.fetchall(sql) + if res is False: + stdio.stop_loading('fail') + return + for row in res: + if str(row['name']) == unit_name: + unit_name += '1' + if row['unit_config_id'] in units_id: + cpu_total -= row['max_cpu'] * units_id[row['unit_config_id']] + mem_total -= row['max_memory'] * units_id[row['unit_config_id']] + # disk_total -= row['max_disk_size'] + + MIN_CPU = 2 + MIN_MEMORY = 1073741824 + MIN_DISK_SIZE = 536870912 + MIN_IOPS = 128 + MIN_SESSION_NUM = 64 + if cpu_total < MIN_CPU: + return error('%s: resource not enough: cpu count less than %s' % (zone_list, MIN_CPU)) + if mem_total < MIN_MEMORY: + return error('%s: resource not enough: memory less than %s' % (zone_list, format_size(MIN_MEMORY))) + if disk_total < MIN_DISK_SIZE: + return error('%s: resource not enough: disk space less than %s' % (zone_list, format_size(MIN_DISK_SIZE))) + + try: + max_memory = get_parsed_option('max_memory', mem_total) + max_disk_size = get_parsed_option('max_disk_size', disk_total) + min_memory = get_parsed_option('min_memory', max_memory) + except Exception as e: + error(e) + return + + max_cpu = get_option('max_cpu', cpu_total) + max_iops = get_option('max_iops', MIN_IOPS) + max_session_num = get_option('max_session_num', MIN_SESSION_NUM) + min_cpu = get_option('min_cpu', max_cpu) + min_iops = get_option('min_iops', max_iops) + + if cpu_total < max_cpu: + return error('resource not enough: cpu (Avail: %s, Need: %s)' % (cpu_total, max_cpu)) + if mem_total < max_memory: + return error('resource not enough: memory (Avail: %s, Need: %s)' % (format_size(mem_total), format_size(max_memory))) + if disk_total < max_disk_size: + return error('resource not enough: disk space (Avail: %s, Need: %s)' % (format_size(disk_total), format_size(max_disk_size))) + + if max_iops < MIN_IOPS: + return error('max_iops must greater than %d' % MIN_IOPS) + if max_session_num < MIN_SESSION_NUM: + return error('max_session_num must greater than %d' % MIN_SESSION_NUM) + + if max_cpu < min_cpu: + return error('min_cpu must less then max_cpu') + if max_memory < min_memory: + return error('min_memory must less then max_memory') + if max_iops < min_iops: + return error('min_iops must less then max_iops') + + + zone_num = len(zones) + charset = get_option('charset', '') + collate = get_option('collate', '') + replica_num = get_option('replica_num', zone_num) + logonly_replica_num = get_option('logonly_replica_num', 0) + tablegroup = get_option('tablegroup', '') + primary_zone = get_option('primary_zone', 'RANDOM') + locality = get_option('locality', '') + variables = get_option('variables', '') + + if replica_num == 0: + replica_num = zone_num + elif replica_num > zone_num: + return error('replica_num cannot be greater than zone num (%s)' % zone_num) + if not primary_zone: + primary_zone = 'RANDOM' + if logonly_replica_num > replica_num: + return error('logonly_replica_num cannot be greater than replica_num (%s)' % replica_num) + + # create resource unit + sql = 'create resource unit %s max_cpu %.1f, max_memory %d, max_iops %d, max_disk_size %d, max_session_num %d, min_cpu %.1f, min_memory %d, min_iops %d' sql = sql % (unit_name, max_cpu, max_memory, max_iops, max_disk_size, max_session_num, min_cpu, min_memory, min_iops) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('faild to crate unit, execute sql exception: %s' % sql) - return + res = cursor.execute(sql) + if res is False: + stdio.stop_loading('fail') + return - # create resource pool - sql = "create resource pool %s unit='%s', unit_num=%d, zone_list=%s" % (pool_name, unit_name, unit_num, zone_list) - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('failed to create pool, execute sql exception: %s' % sql) - return + # create resource pool + sql = "create resource pool %s unit='%s', unit_num=%d, zone_list=%s" % (pool_name, unit_name, unit_num, zone_list) + res = cursor.execute(sql) + if res is False: + stdio.stop_loading('fail') + return + + # create tenant + sql = "create tenant %s replica_num=%d,zone_list=%s,primary_zone='%s',resource_pool_list=('%s')" + sql = sql % (name, replica_num, zone_list, primary_zone, pool_name) + if charset: + sql += ", charset = '%s'" % charset + if collate: + sql += ", collate = '%s'" % collate + if logonly_replica_num: + sql += ", logonly_replica_num = %d" % logonly_replica_num + if tablegroup: + sql += ", default tablegroup ='%s'" % tablegroup + if locality: + sql += ", locality = '%s'" % locality + + set_mode = "ob_compatibility_mode = '%s'" % mode + if variables: + sql += "set %s, %s" % (variables, set_mode) + else: + sql += "set %s" % set_mode + res = cursor.execute(sql) + if res is False: + stdio.stop_loading('fail') + return - # create tenant - sql = "create tenant %s replica_num=%d,zone_list=%s,primary_zone='%s',resource_pool_list=('%s')" - sql = sql % (name, replica_num, zone_list, primary_zone, pool_name) - if charset: - sql += ", charset = '%s'" % charset - if collate: - sql += ", collate = '%s'" % collate - if logonly_replica_num: - sql += ", logonly_replica_num = %d" % logonly_replica_num - if tablegroup: - sql += ", default tablegroup ='%s'" % tablegroup - if locality: - sql += ", locality = '%s'" % locality - - set_mode = "ob_compatibility_mode = '%s'" % mode - if variables: - sql += "set %s, %s" % (variables, set_mode) - else: - sql += "set %s" % set_mode - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('faild to crate tenant, execute sql exception: %s' % sql) - return - stdio.stop_loading('succeed') + + database = get_option('database') + if database: + sql = 'create database {}'.format(database) + if not exec_sql_in_tenant(sql=sql, cursor=cursor, tenant=name, mode=mode) and not create_if_not_exists: + stdio.error('failed to create database {}'.format(database)) + return + + db_username = get_option('db_username') + db_password = get_option('db_password', '') + if db_username: + if mode == "mysql": + sql = """create user if not exists '{username}' IDENTIFIED BY '{password}'; + grant all on *.* to '{username}' WITH GRANT OPTION;""".format( + username=db_username, password=db_password) + else: + # todo: fix oracle user create + sql = """create {username} IDENTIFIED BY {password}; +grant all on *.* to {username} WITH GRANT OPTION; +grant dba to {username}; +grant all privileges to {username};""" + if not exec_sql_in_tenant(sql=sql, cursor=cursor, tenant=name, mode=mode): + stdio.error('failed to create user {}'.format(db_username)) + return + return plugin_context.return_true() \ No newline at end of file diff --git a/plugins/oceanbase/3.1.0/display.py b/plugins/oceanbase/3.1.0/display.py index a2a12db..e04bcfe 100644 --- a/plugins/oceanbase/3.1.0/display.py +++ b/plugins/oceanbase/3.1.0/display.py @@ -23,6 +23,10 @@ import time +def passwd_format(passwd): + return "'{}'".format(passwd.replace("'", "'\"'\"'")) + + def display(plugin_context, cursor, *args, **kwargs): stdio = plugin_context.stdio stdio.start_loading('Wait for observer init') @@ -30,21 +34,28 @@ def display(plugin_context, cursor, *args, **kwargs): try: while True: try: - cursor.execute('select * from oceanbase.__all_server') - servers = cursor.fetchall() + servers = cursor.fetchall('select * from oceanbase.__all_server', raise_exception=True, exc_level='verbose') if servers: - stdio.print_list(servers, ['ip', 'version', 'port', 'zone', 'status'], + stdio.print_list(servers, ['ip', 'version', 'port', 'zone', 'status'], lambda x: [x['svr_ip'], x['build_version'].split('_')[0], x['inner_port'], x['zone'], x['status']], title='observer') + user = 'root' password = cluster_config.get_global_conf().get('root_password', '') - cmd = 'obclient -h%s -P%s -uroot %s-Doceanbase -A' % (servers[0]['svr_ip'], servers[0]['inner_port'], '-p%s ' % password if password else '') + cmd = 'obclient -h%s -P%s -u%s %s-Doceanbase -A' % (servers[0]['svr_ip'], servers[0]['inner_port'], user, '-p%s ' % passwd_format(password) if password else '') stdio.print(cmd) stdio.stop_loading('succeed') - return plugin_context.return_true() + info_dict = { + "type": "db", + "ip": servers[0]['svr_ip'], + "port": servers[0]['inner_port'], + "user": user, + "password": password, + "cmd": cmd + } + return plugin_context.return_true(info=info_dict) except Exception as e: if e.args[0] != 1146: raise e time.sleep(3) except: stdio.stop_loading('fail', 'observer need bootstarp') - stdio.exception('') plugin_context.return_false() diff --git a/plugins/oceanbase/3.1.0/drop_tenant.py b/plugins/oceanbase/3.1.0/drop_tenant.py index 7ff28e7..fdf36d2 100644 --- a/plugins/oceanbase/3.1.0/drop_tenant.py +++ b/plugins/oceanbase/3.1.0/drop_tenant.py @@ -25,9 +25,6 @@ def drop_tenant(plugin_context, cursor, *args, **kwargs): def error(*arg, **kwargs): stdio.error(*arg, **kwargs) stdio.stop_loading('fail') - def exception(*arg, **kwargs): - stdio.exception(*arg, **kwargs) - stdio.stop_loading('fail') cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio @@ -43,49 +40,42 @@ def exception(*arg, **kwargs): stdio.start_loading('Drop tenant %s' % tenant_name) - tenant = None - sql = "select * from oceanbase.gv$tenant where tenant_name = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant = cursor.fetchone() - if not tenant: - error('No such Tenant %s' % tenant_name) - return - except: - exception('execute sql exception: %s' % (sql % tenant_name)) + sql = "select * from oceanbase.gv$tenant where tenant_name = '%s'" % tenant_name + tenant = cursor.fetchone(sql) + if tenant is False: + return + if not tenant: + error('No such Tenant %s' % tenant_name) return - pool = None sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant['tenant_id'] - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - pool = cursor.fetchone() - sql = "drop tenant %s FORCE" % tenant_name - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - if not pool: - return - sql = "drop resource pool %s" % pool['name'] - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) + pool = cursor.fetchone(sql) + if pool is False: + error() + return + sql = "drop tenant %s FORCE" % tenant_name + res = cursor.execute(sql) + if res is False: + error() + return + if not pool: + error() + return + sql = "drop resource pool %s" % pool['name'] + res = cursor.execute(sql) + if res is False: + error() return sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - unit = cursor.fetchone() - if not unit: - return - sql = "drop resource unit %s" % unit['name'] - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) + unit = cursor.fetchone(sql) + if not unit: + error() + return + sql = "drop resource unit %s" % unit['name'] + res = cursor.execute(sql) + if res is False: + error() return stdio.stop_loading('succeed') diff --git a/plugins/oceanbase/3.1.0/generate_config.py b/plugins/oceanbase/3.1.0/generate_config.py index 5e453f7..3f350db 100644 --- a/plugins/oceanbase/3.1.0/generate_config.py +++ b/plugins/oceanbase/3.1.0/generate_config.py @@ -66,70 +66,79 @@ def get_system_memory(memory_limit): return format_size(system_memory, 0) -def generate_config(plugin_context, deploy_config, *args, **kwargs): +def generate_config(plugin_context, generate_config_mini=False, generate_check=True, return_generate_keys=False, generate_consistent_config=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=[ + 'memory_limit', 'datafile_size', 'clog_disk_utilization_threshold', 'clog_disk_usage_limit_percentage', + 'syslog_level', 'enable_syslog_recycle', 'enable_syslog_wf', 'max_syslog_file_count', 'cluster_id', + 'devname', 'system_memory', 'cpu_count', + ]) + + def update_server_conf(server, key, value): + if server not in generate_configs: + generate_configs[server] = {} + generate_configs[server][key] = value + def update_global_conf(key, value): + generate_configs['global'][key] = value + + def summit_config(): + generate_global_config = generate_configs['global'] + for key in generate_global_config: + cluster_config.update_global_conf(key, generate_global_config[key], False) + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + for key in generate_server_config: + cluster_config.update_server_conf(server, key, generate_server_config[key], False) + cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio success = True + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) stdio.start_loading('Generate observer configuration') - if not cluster_config.get_global_conf().get('appname'): - default_appname = 'obcluster' - for componet_name in ['obproxy', 'obproxy-ce']: - if componet_name in deploy_config.components: - obproxy_cluster_config = deploy_config.components[componet_name] - cluster_name = obproxy_cluster_config.get_global_conf().get('cluster_name') - if not cluster_name: - for server in obproxy_cluster_config.servers: - server_config = obproxy_cluster_config.get_server_conf(server) - if server_config.get('cluster_name'): - default_appname = server_config['cluster_name'] - break - break - cluster_config.update_global_conf('appname', default_appname, False) - MIN_MEMORY = 8 << 30 MIN_CPU_COUNT = 16 START_NEED_MEMORY = 3 << 30 clog_disk_utilization_threshold_max = 95 clog_disk_usage_limit_percentage_max = 98 global_config = cluster_config.get_original_global_conf() - - if getattr(plugin_context.options, 'mini', False): - if not global_config.get('memory_limit_percentage') and not global_config.get('memory_limit'): - cluster_config.update_global_conf('memory_limit', format_size(MIN_MEMORY, 0), False) - if not global_config.get('datafile_size') and not global_config.get('datafile_disk_percentage'): - cluster_config.update_global_conf('datafile_size', '20G', False) - if not global_config.get('clog_disk_utilization_threshold'): - cluster_config.update_global_conf('clog_disk_utilization_threshold', clog_disk_utilization_threshold_max, False) - if not global_config.get('clog_disk_usage_limit_percentage'): - cluster_config.update_global_conf('clog_disk_usage_limit_percentage', clog_disk_usage_limit_percentage_max, False) max_syslog_file_count_default = 4 if global_config.get('syslog_level') is None: - cluster_config.update_global_conf('syslog_level', 'INFO', False) + update_global_conf('syslog_level', 'INFO') if global_config.get('enable_syslog_recycle') is None: - cluster_config.update_global_conf('enable_syslog_recycle', True, False) + update_global_conf('enable_syslog_recycle', True) if global_config.get('enable_syslog_wf') is None: - cluster_config.update_global_conf('enable_syslog_wf', True, False) + update_global_conf('enable_syslog_wf', False) if global_config.get('max_syslog_file_count') is None: - cluster_config.update_global_conf('max_syslog_file_count', max_syslog_file_count_default, False) + update_global_conf('max_syslog_file_count', max_syslog_file_count_default) if global_config.get('cluster_id') is None: - cluster_config.update_global_conf('cluster_id', 1, False) - + update_global_conf('cluster_id', 1) + + if generate_config_mini: + if not global_config.get('memory_limit_percentage') and not global_config.get('memory_limit'): + update_global_conf('memory_limit', format_size(MIN_MEMORY, 0)) + if not global_config.get('datafile_size') and not global_config.get('datafile_disk_percentage'): + update_global_conf('datafile_size', '20G') + if not global_config.get('clog_disk_utilization_threshold'): + update_global_conf('clog_disk_utilization_threshold', clog_disk_utilization_threshold_max) + if not global_config.get('clog_disk_usage_limit_percentage'): + update_global_conf('clog_disk_usage_limit_percentage', clog_disk_usage_limit_percentage_max) + summit_config() + for server in cluster_config.servers: ip = server.ip client = clients[server] server_config = cluster_config.get_server_conf_with_default(server) - user_server_config = cluster_config.get_original_server_conf(server) - if not server_config.get('home_path'): - stdio.error("observer %s: missing configuration 'home_path' in configuration file" % server) - success = False - continue + user_server_config = cluster_config.get_original_server_conf_with_global(server) if user_server_config.get('devname') is None: if client.is_localhost(): - cluster_config.update_server_conf(server, 'devname', 'lo') + update_server_conf(server, 'devname', 'lo') else: devinfo = client.execute_command('cat /proc/net/dev').stdout interfaces = re.findall('\n\s+(\w+):', devinfo) @@ -137,7 +146,7 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): if interface == 'lo': continue if client.execute_command('ping -W 1 -c 1 -I %s %s' % (interface, ip)): - cluster_config.update_server_conf(server, 'devname', interface) + update_server_conf(server, 'devname', interface) break dirs = {"home_path": server_config['home_path']} @@ -174,35 +183,32 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): key = memory_key_map[k] server_memory_stats[key] = parse_size(str(v)) - if server_memory_stats['available'] < START_NEED_MEMORY: - stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(START_NEED_MEMORY))) - success = False - continue - - if server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached'] < MIN_MEMORY: - stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(MIN_MEMORY))) - success = False - continue + if generate_check: + if server_memory_stats['available'] < START_NEED_MEMORY: + stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(START_NEED_MEMORY))) + success = False + continue + + if server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached'] < MIN_MEMORY: + stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(MIN_MEMORY))) + success = False + continue memory_limit = max(MIN_MEMORY, server_memory_stats['available'] * 0.9) server_config['memory_limit'] = format_size(memory_limit, 0) - cluster_config.update_server_conf(server, 'memory_limit', server_config['memory_limit'], False) + update_server_conf(server, 'memory_limit', server_config['memory_limit']) + auto_set_memory = True else: stdio.error("%s: fail to get memory info.\nPlease configure 'memory_limit' manually in configuration file") success = False continue else: - try: - memory_limit = parse_size(server_config.get('memory_limit')) - auto_set_memory = True - except: - stdio.error('memory_limit must be an integer') - return + memory_limit = parse_size(server_config.get('memory_limit')) auto_set_system_memory = False if not user_server_config.get('system_memory'): auto_set_system_memory = True - cluster_config.update_server_conf(server, 'system_memory', get_system_memory(memory_limit), False) + update_server_conf(server, 'system_memory', get_system_memory(memory_limit)) # cpu if not server_config.get('cpu_count'): @@ -212,9 +218,9 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): server_config['cpu_count'] = max(MIN_CPU_COUNT, int(cpu_num - 2)) else: server_config['cpu_count'] = MIN_CPU_COUNT - cluster_config.update_server_conf(server, 'cpu_count', server_config['cpu_count'], False) + update_server_conf(server, 'cpu_count', server_config['cpu_count']) elif server_config['cpu_count'] < MIN_CPU_COUNT: - cluster_config.update_server_conf(server, 'cpu_count', MIN_CPU_COUNT, False) + update_server_conf(server, 'cpu_count', MIN_CPU_COUNT) stdio.warn('(%s): automatically adjust the cpu_count %s' % (server, MIN_CPU_COUNT)) # disk @@ -264,7 +270,7 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): if user_server_config.get('enable_syslog_recycle') is False: log_size = real_disk_total * 0.1 else: - log_size = (256 << 20) * user_server_config.get('max_syslog_file_count', max_syslog_file_count_default) * 4 + log_size = (256 << 20) * int(user_server_config.get('max_syslog_file_count', max_syslog_file_count_default)) * 4 else: log_size = 0 clog_padding_size = int(real_disk_total * (1 - clog_disk_utilization_threshold_max / 100.0 * 0.8)) @@ -288,17 +294,17 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): if min_need <= disk_free: memory_limit = (disk_free - padding_size) / 7 server_config['memory_limit'] = format_size(memory_limit, 0) - cluster_config.update_server_conf(server, 'memory_limit', server_config['memory_limit'], False) + update_server_conf(server, 'memory_limit', server_config['memory_limit']) memory_limit = parse_size(server_config['memory_limit']) clog_disk_size = memory_limit * 4 clog_size = int(round(clog_disk_size * 0.64)) if auto_set_system_memory: - cluster_config.update_server_conf(server, 'system_memory', get_system_memory(memory_limit), False) + update_server_conf(server, 'system_memory', get_system_memory(memory_limit)) disk_flag = True else: disk_flag = True - if not disk_flag: + if generate_check and not disk_flag: stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s). Use `redo_dir` to set other disk for clog' % (ip, kp, format_size(disk_free), format_size(min_need))) success = False continue @@ -309,12 +315,81 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): clog_disk_utilization_threshold = min(clog_disk_utilization_threshold, clog_disk_utilization_threshold_max) clog_disk_usage_limit_percentage = min(int(clog_disk_utilization_threshold / 80.0 * 95), clog_disk_usage_limit_percentage_max) - cluster_config.update_server_conf(server, 'datafile_size', datafile_size_format, False) - cluster_config.update_server_conf(server, 'clog_disk_utilization_threshold', clog_disk_utilization_threshold, False) - cluster_config.update_server_conf(server, 'clog_disk_usage_limit_percentage', clog_disk_usage_limit_percentage, False) + update_server_conf(server, 'datafile_size', datafile_size_format) + update_server_conf(server, 'clog_disk_utilization_threshold', clog_disk_utilization_threshold) + update_server_conf(server, 'clog_disk_usage_limit_percentage', clog_disk_usage_limit_percentage) else: datafile_size = max(5 << 30, data_dir_disk['avail'] * 0.8, 0) - cluster_config.update_server_conf(server, 'datafile_size', format_size(datafile_size, 0), False) + update_server_conf(server, 'datafile_size', format_size(datafile_size, 0)) + + if generate_consistent_config: + generate_global_config = generate_configs['global'] + server_num = len(cluster_config.servers) + MIN_KEY = ['memory_limit', 'datafile_size', 'system_memory', 'cpu_count'] + MAX_KEY = ['clog_disk_utilization_threshold', 'clog_disk_usage_limit_percentage'] + CAPACITY_KEY = ['memory_limit', 'datafile_size', 'system_memory'] + keys = MIN_KEY + MAX_KEY + for key in keys: + servers = [] + values = [] + is_capacity_key = key in CAPACITY_KEY + for server in cluster_config.servers: + if key in generate_configs.get(server, {}): + value = generate_configs[server][key] + servers.append(server) + values.append(parse_size(value) if is_capacity_key else value) + if values: + if len(values) != server_num and key in generate_global_config: + continue + comp = min if key in MIN_KEY else max + value = comp(values) + generate_global_config[key] = format_size(value, 0) if is_capacity_key else value + for server in servers: + del generate_configs[server][key] + + # merge_generate_config + merge_config = {} + generate_global_config = generate_configs['global'] + count_base = len(cluster_config.servers) - 1 + if count_base < 1: + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_global_config.update(generate_configs[server]) + generate_configs[server] = {} + else: + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + merged_server_config = {} + for key in generate_server_config: + if key in generate_global_config: + if generate_global_config[key] != generate_server_config[key]: + merged_server_config[key] = generate_server_config[key] + elif key in merge_config: + if merge_config[key]['value'] != generate_server_config[key]: + merged_server_config[key] = generate_server_config[key] + elif count_base == merge_config[key]['count']: + generate_global_config[key] = generate_server_config[key] + del merge_config[key] + else: + merge_config[key]['severs'].append(server) + merge_config[key]['count'] += 1 + else: + merge_config[key] = {'value': generate_server_config[key], 'severs': [server], 'count': 1} + generate_configs[server] = merged_server_config + + for key in merge_config: + config_st = merge_config[key] + for server in config_st['severs']: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + generate_server_config[key] = config_st['value'] + + # summit_config + summit_config() if success: stdio.stop_loading('succeed') diff --git a/plugins/oceanbase/3.1.0/init.py b/plugins/oceanbase/3.1.0/init.py index e89aea7..f7e1742 100644 --- a/plugins/oceanbase/3.1.0/init.py +++ b/plugins/oceanbase/3.1.0/init.py @@ -60,7 +60,7 @@ def init_dir(server, client, key, path, link_path=None): return False -def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def init(plugin_context, *args, **kwargs): global stdio, force cluster_config = plugin_context.cluster_config clients = plugin_context.clients @@ -78,8 +78,6 @@ def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): server_config = cluster_config.get_server_conf(server) client = clients[server] home_path = server_config['home_path'] - remote_home_path = client.execute_command('echo ${OBD_HOME:-"$HOME"}/.obd').stdout.strip() - remote_repository_dir = repository_dir.replace(local_home_path, remote_home_path) if not server_config.get('data_dir'): server_config['data_dir'] = '%s/store' % home_path diff --git a/plugins/oceanbase/3.1.0/list_tenant.py b/plugins/oceanbase/3.1.0/list_tenant.py new file mode 100644 index 0000000..a86f7cc --- /dev/null +++ b/plugins/oceanbase/3.1.0/list_tenant.py @@ -0,0 +1,85 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import re + + +def parse_size(size): + _bytes = 0 + if isinstance(size, str): + size = size.strip() + if not isinstance(size, str) or size.isdigit(): + _bytes = int(size) + else: + units = {"B": 1, "K": 1 << 10, "M": 1 << 20, "G": 1 << 30, "T": 1 << 40} + match = re.match(r'^([1-9][0-9]*)\s*([B,K,M,G,T])$', size.upper()) + _bytes = int(match.group(1)) * units[match.group(2)] + return _bytes + + +def format_size(size, precision=1): + units = ['B', 'K', 'M', 'G', 'T', 'P'] + idx = 0 + if precision: + div = 1024.0 + format = '%.' + str(precision) + 'f%s' + else: + div = 1024 + format = '%d%s' + while idx < 5 and size >= 1024: + size /= 1024.0 + idx += 1 + return format % (size, units[idx]) + + +def list_tenant(plugin_context, cursor, *args, **kwargs): + + cluster_config = plugin_context.cluster_config + stdio = plugin_context.stdio + + stdio.start_loading('Select tenant') + tenant_infos = [] + sql = "select * from oceanbase.gv$tenant;" + tenants = cursor.fetchall(sql) + if tenants is False: + stdio.stop_loading('fail') + return + for tenant in tenants: + unit_name = '%s_unit' % tenant['tenant_name'] if tenant['tenant_name'] != 'sys' else 'sys_unit_config' + sql = "select * from oceanbase.__all_unit_config where name = '%s'" + res = cursor.fetchone(sql % unit_name) + if res is False: + stdio.stop_loading('fail') + return + tenant_infos.append(dict(tenant, **res)) + if tenant_infos: + stdio.print_list(tenant_infos, ['tenant_name', 'zone_list', 'primary_zone', 'max_cpu', 'min_cpu', 'max_memory', + 'min_memory', 'max_iops', 'min_iops', 'max_disk_size', 'max_session_num'], + lambda x: [x['tenant_name'], x['zone_list'], x['primary_zone'], x['max_cpu'], x['min_cpu'], + format_size(x['max_memory']), format_size(x['min_memory']), x['max_iops'], + x['min_iops'], format_size(x['max_disk_size']), x['max_session_num']], + title='tenant') + stdio.stop_loading('succeed') + return plugin_context.return_true() + + stdio.stop_loading('fail') + plugin_context.return_false() \ No newline at end of file diff --git a/plugins/oceanbase/3.1.0/major_freeze.py b/plugins/oceanbase/3.1.0/major_freeze.py index f61c76b..b788fed 100644 --- a/plugins/oceanbase/3.1.0/major_freeze.py +++ b/plugins/oceanbase/3.1.0/major_freeze.py @@ -25,34 +25,32 @@ def major_freeze(plugin_context, cursor, *args, **kwargs): - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - stdio = plugin_context.stdio - merge_version = execute(cursor, "select value from oceanbase.__all_zone where name='frozen_version'")['value'] + merge_version = cursor.fetchone("select value from oceanbase.__all_zone where name='frozen_version'") + if merge_version is False: + return + merge_version = merge_version['value'] stdio.start_loading('Merge') - execute(cursor, 'alter system major freeze') + if cursor.fetchone('alter system major freeze') is False: + return sql = "select value from oceanbase.__all_zone where name='frozen_version' and value != %s" % merge_version while True: - if execute(cursor, sql): + res = cursor.fetchone(sql) + if res is False: + return + if res: break time.sleep(1) while True: - if not execute(cursor, """select * from oceanbase.__all_zone + res = cursor.fetchone("""select * from oceanbase.__all_zone where name='last_merged_version' and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) - and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') - """): + and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') + """) + if res is False: + return + if not res: break time.sleep(5) stdio.stop_loading('succeed') diff --git a/plugins/oceanbase/3.1.0/ocp_check.py b/plugins/oceanbase/3.1.0/ocp_check.py index 0523778..d0af70b 100644 --- a/plugins/oceanbase/3.1.0/ocp_check.py +++ b/plugins/oceanbase/3.1.0/ocp_check.py @@ -58,8 +58,7 @@ def ocp_check(plugin_context, ocp_version, cursor, new_cluster_config=None, new_ stdio.error('%s Multiple OBservers exist.' % server) try: - cursor.execute("select * from oceanbase.__all_user where user_name = 'root' and passwd = ''") - if cursor.fetchone() and not cluster_config.get_global_conf().get("root_password"): + if cursor.fetchone("select * from oceanbase.__all_user where user_name = 'root' and passwd = ''", raise_exception=True) and not cluster_config.get_global_conf().get("root_password"): pwd_not_empty = False stdio.error('The password of root@sys is empty. Run the edit-config command to modify the root_password value of %s.' % cluster_config.name) except: @@ -68,8 +67,7 @@ def ocp_check(plugin_context, ocp_version, cursor, new_cluster_config=None, new_ zones = {} try: - cursor.execute("select zone from oceanbase.__all_zone where name = 'idc' and info = ''") - ret = cursor.fetchall() + ret = cursor.fetchall("select zone from oceanbase.__all_zone where name = 'idc' and info = ''", raise_exception=True) if ret: for row in ret: zones[str(row['zone'])] = 1 diff --git a/plugins/oceanbase/3.1.0/parameter.yaml b/plugins/oceanbase/3.1.0/parameter.yaml index b10bffa..1f55c1d 100644 --- a/plugins/oceanbase/3.1.0/parameter.yaml +++ b/plugins/oceanbase/3.1.0/parameter.yaml @@ -1,5 +1,7 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING min_value: NULL max_value: NULL @@ -7,7 +9,9 @@ description_en: the directory for the work data file description_local: OceanBase工作目录 - name: cluster_id + name_local: 集群ID require: true + essential: true type: INT default: 1 min_value: 1 @@ -17,14 +21,18 @@ description_en: ID of the cluster description_local: 本OceanBase集群ID - name: data_dir + name_local: 数据目录 type: STRING + essential: true min_value: NULL max_value: NULL need_redeploy: true description_en: the directory for the data file description_local: 存储sstable等数据的目录 - name: redo_dir + name_local: 日志目录 type: STRING + essential: true min_value: NULL max_value: NULL need_redeploy: true @@ -52,14 +60,18 @@ description_en: the directory for the ilog file description_local: 存储ilog数据的目录 - name: devname + name_local: 网卡名 type: STRING + essential: true min_value: NULL max_value: NULL need_restart: true description_en: name of network adapter description_local: 服务进程绑定的网卡设备名 - name: rpc_port + name_local: 内部通信端口 require: true + essential: true type: INT default: 2882 min_value: 1025 @@ -69,7 +81,9 @@ description_en: the port number for RPC protocol. description_local: 集群内部通信的端口号 - name: mysql_port + name_local: 服务端口 require: true + essential: true type: INT default: 2881 min_value: 1025 @@ -289,7 +303,9 @@ description_en: the time interval between the schedules of the task that checks whether the partition load balancing task has timed-out. description_local: 检查负载均衡等后台任务是否超时的时间间隔 - name: datafile_size + name_local: 数据文件大小 require: false + essential: true type: CAPACITY default: 0 min_value: 0M @@ -297,8 +313,8 @@ modify_limit: decrease section: SSTABLE need_restart: false - description_en: size of the data file. - description_local: 数据文件大小。一般不要设置。 + description_en: size of the data file. Please enter an capacity, such as 20G + description_local: 数据文件大小。请输入带容量带单位的整数,如20G - name: clog_cache_priority require: false type: INT @@ -780,6 +796,7 @@ description_local: 本地存储配置文件的多个目录,为了冗余存储多份配置文件 - name: enable_syslog_recycle require: false + essential: true type: BOOL default: false min_value: NULL @@ -788,6 +805,17 @@ need_restart: false description_en: specifies whether log file recycling is turned on description_local: 是否自动回收系统日志 +- name: max_syslog_file_count + require: false + essential: true + type: INT + default: 0 + min_value: 0 + max_value: NULL + section: OBSERVER + need_restart: false + description_en: specifies the maximum number of the log files that can co-exist before the log file recycling kicks in. Each log file can occupy at most 256MB disk space. When this value is set to 0, no log file will be removed. + description_local: 系统日志自动回收复用时,最多保留多少个。值0表示不自动清理。 - name: meta_table_read_write_mode require: false type: INT @@ -1007,7 +1035,6 @@ section: OBSERVER need_restart: false description_en: specify what kind of verification should be done when merging micro block. 0, no verification will be done; 1, verify encoding algorithm, encoded micro block will be read to ensure data is correct; 2, verify encoding and compression algorithm, besides encoding verification, compressed block will be decompressed to ensure data is correct; 3, verify encoding, compression algorithm and lost write protect - description_local: 控制合并时宏块的校验级别 - name: backup_net_limit require: false @@ -1279,15 +1306,6 @@ need_restart: false description_en: size of clog files that a replica lag behind leader to trigger rebuild description_local: 备副本的事务日志和主副本差距超过该阈值时,触发副本重建 -- name: system_memory - type: CAPACITY - default: 30G - min_value: 0M - max_value: NULL - section: OBSERVER - need_restart: false - description_en: the memory reserved for internal use which cannot be allocated to any outer-tenant, and should be determined to guarantee every server functions normally. - description_local: 系统预留内存大小,不能分配给普通租户使用 - name: cpu_quota_concurrency require: false type: DOUBLE @@ -1389,7 +1407,9 @@ description_en: the time interval between the schedules of the partition load-balancing task. description_local: 负载均衡等后台任务线程空闲时的唤醒间隔时间 - name: memory_limit + name_local: 最大运行内存 require: false + essential: true type: CAPACITY default: 0 min_value: NULL @@ -1397,8 +1417,19 @@ modify_limit: decrease section: OBSERVER need_restart: false - description_en: the size of the memory reserved for internal use(for testing purpose) - description_local: 可用总内存大小。用于调试,不要设置。 + description_en: the size of the memory reserved for internal use(for testing purpose). Please enter an capacity, such as 8G + description_local: 可用总内存大小。请输入带容量带单位的整数,如8G +- name: system_memory + name_local: 集群系统内存 + essential: true + type: CAPACITY + default: 30G + min_value: 0M + max_value: NULL + section: OBSERVER + need_restart: false + description_en: the memory reserved for internal use which cannot be allocated to any outer-tenant, and should be determined to guarantee every server functions normally. Please enter an capacity, such as 2G + description_local: 系统预留内存大小,不能分配给普通租户使用。请输入带容量带单位的整数,如2G - name: __min_full_resource_pool_memory require: true type: INT @@ -1639,7 +1670,9 @@ description_en: disable write to memstore when observer memstore free memory(plus memory hold by blockcache) lower than this limit, description_local: 当全局剩余内存小于这个百分比时,暂停普通租户写入(sys租户不受影响) - name: cpu_count + name_local: 系统CPU总数 require: false + essential: true type: INT default: 0 min_value: 0 @@ -1658,16 +1691,6 @@ need_restart: false description_en: control if auto delete expired backup description_local: 自动删除过期的备份 -- name: max_syslog_file_count - require: false - type: INT - default: 0 - min_value: 0 - max_value: NULL - section: OBSERVER - need_restart: false - description_en: specifies the maximum number of the log files that can co-exist before the log file recycling kicks in. Each log file can occupy at most 256MB disk space. When this value is set to 0, no log file will be removed. - description_local: 系统日志自动回收复用时,最多保留多少个。值0表示不自动清理。 - name: appname require: false type: STRING diff --git a/plugins/oceanbase/3.1.0/reload.py b/plugins/oceanbase/3.1.0/reload.py index d0021af..385ed9f 100644 --- a/plugins/oceanbase/3.1.0/reload.py +++ b/plugins/oceanbase/3.1.0/reload.py @@ -79,18 +79,13 @@ def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): for zone in zones_config: zone_config = zones_config[zone] for key in zone_config: - msg = '' - try: - msg = sql = 'alter system modify zone %s set %s = %%s' % (zone, inner_config[key]) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql, [zone_config[key]]) - stdio.verbose('%s ok' % sql) - except: - global_ret = False - stdio.exception('execute sql exception: %s' % msg) + sql = 'alter system modify zone %s set %s = %%s' % (zone, inner_config[key]) + if cursor.execute(sql, [zone_config[key]]) is False: + return + stdio.verbose('%s ok' % sql) + raise_cursor = cursor.raise_cursor for key in global_change_conf: - msg = '' try: if key in ['proxyro_password', 'root_password']: if global_change_conf[key]['count'] != servers_num: @@ -99,17 +94,14 @@ def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): value = change_conf[server][key] if change_conf[server].get(key) is not None else '' user = key.split('_')[0] msg = sql = 'CREATE USER IF NOT EXISTS %s IDENTIFIED BY %%s' % (user) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) msg = sql = 'alter user "%s" IDENTIFIED BY %%s' % (user) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) continue if global_change_conf[key]['count'] == servers_num: - sql = 'alter system set %s = %%s' % key + msg = sql = 'alter system set %s = %%s' % key value = change_conf[server][key] - stdio.verbose('execute sql: %s' % msg) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) cluster_config.update_global_conf(key, value, False) continue for server in servers: @@ -117,16 +109,17 @@ def reload(plugin_context, cursor, new_cluster_config, *args, **kwargs): continue value = change_conf[server][key] msg = sql = 'alter system set %s = %%s server=%%s' % key - stdio.verbose('execute sql: %s' % msg) - cursor.execute(sql, [value, cluster_server[server]]) + raise_cursor.execute(sql, [value, cluster_server[server]]) cluster_config.update_server_conf(server, key, value, False) except: global_ret = False - stdio.exception('execute sql exception: %s' % msg) - cursor.execute('alter system reload server') - cursor.execute('alter system reload zone') - cursor.execute('alter system reload unit') + try: + raise_cursor.execute('alter system reload server') + raise_cursor.execute('alter system reload zone') + raise_cursor.execute('alter system reload unit') + except: + global_ret = False if global_ret: stdio.stop_load('succeed') diff --git a/plugins/oceanbase/3.1.0/restart.py b/plugins/oceanbase/3.1.0/restart.py index 22a5775..a36f71f 100644 --- a/plugins/oceanbase/3.1.0/restart.py +++ b/plugins/oceanbase/3.1.0/restart.py @@ -28,11 +28,22 @@ class Restart(object): def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None): self.local_home_path = local_home_path - self.plugin_context = plugin_context + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + self.components = plugin_context.components self.clients = plugin_context.clients self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context self.repository = repository self.start_plugin = start_plugin self.reload_plugin = reload_plugin @@ -47,24 +58,41 @@ def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, self.cursor = None for server in self.cluster_config.servers: self.now_clients[server] = self.clients[server] + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) def close(self): if self.db: self.cursor.close() - self.db.close() self.cursor = None - self.db = None def connect(self): if self.cursor is None or self.execute_sql('select version()', error=False) is False: - self.stdio.verbose('Call %s for %s' % (self.connect_plugin, self.repository)) self.sub_io.start_loading('Connect to observer') - ret = self.connect_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + ret = self.call_plugin(self.connect_plugin) if not ret: self.sub_io.stop_loading('fail') return False self.sub_io.stop_loading('succeed') - self.close() + if self.cursor: + self.close() self.cursor = ret.get_return('cursor') self.db = ret.get_return('connect') while self.execute_sql('use oceanbase', error=False) is False: @@ -73,18 +101,13 @@ def connect(self): return True def execute_sql(self, query, args=None, one=True, error=True): - msg = query % tuple(args) if args is not None else query - self.stdio.verbose("query: %s. args: %s" % (query, args)) - try: - self.stdio.verbose('execute sql: %s' % msg) - self.cursor.execute(query, args) - result = self.cursor.fetchone() if one else self.cursor.fetchall() - result and self.stdio.verbose(result) - return result - except: - msg = 'execute sql exception: %s' % msg if error else '' - self.stdio.exception(msg) - return False + exc_level = 'error' if error is True else 'verbose' + if one: + result = self.cursor.fetchone(query, args, exc_level=exc_level) + else: + result = self.cursor.fetchall(query, args, exc_level=exc_level) + result and self.stdio.verbose(result) + return result def broken_sql(self, sql, sleep_time=3): while True: @@ -135,12 +158,12 @@ def stop_zone(self, zone): def rollback(self): if self.new_clients: self.stdio.start_loading('Rollback') - self.stop_plugin(self.components, self.now_clients, self.new_cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.now_clients, cluster_config=cluster_config) for server in self.cluster_config.servers: client = self.clients[server] new_client = self.now_clients[server] server_config = self.cluster_config.get_server_conf(server) - home_path = server_config['home_path'] chown_cmd = 'sudo chown -R %s:' % client.config.username for key in ['home_path', 'data_dir', 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir']: if key in server_config: @@ -150,17 +173,16 @@ def rollback(self): def dir_read_check(self, client, path): if not client.execute_command('cd %s' % path): - dirpath, name = os.path.split(path) + dirpath, _ = os.path.split(path) return self.dir_read_check(client, dirpath) and client.execute_command('sudo chmod +1 %s' % path) return True def _restart(self): clients = self.clients - self.stdio.verbose('Call %s for %s' % (self.stop_plugin, self.repository)) - if not self.stop_plugin(self.components, clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io): + if not self.call_plugin(self.stop_plugin, clients=clients): self.stdio.stop_loading('stop_loading', 'fail') return False - + if self.new_clients: self.stdio.verbose('use new clients') for server in self.cluster_config.servers: @@ -178,10 +200,10 @@ def _restart(self): clients = self.new_clients cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config - self.stdio.verbose('Call %s for %s' % (self.start_plugin, self.repository)) - if not self.start_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, local_home_path=self.local_home_path, repository_dir=self.repository.repository_dir): + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, repository=self.repository): self.stdio.stop_loading('stop_loading', 'fail') return False + self.close() return True def rolling(self, zones_servers): @@ -205,24 +227,24 @@ def rolling(self, zones_servers): where svr_ip = %s and svr_port = %s and refreshed_schema_version > 1 ) as b on a.tenant_id = b.tenant_id where b.tenant_id is null''' - if self.execute_sql(sql, args=(server.ip, config['rpc_port'])).get('cnt'): + if self.execute_sql(sql, args=(server.ip, config['rpc_port']), error=False).get('cnt'): break else: break time.sleep(3) - while self.execute_sql("select * from oceanbase.__all_virtual_clog_stat where table_id = 1099511627777 and status != 'ACTIVE'"): + while self.execute_sql("select * from oceanbase.__all_virtual_clog_stat where table_id = 1099511627777 and status != 'ACTIVE'", error=False): time.sleep(3) - + self.stop_zone(zone) if not self._restart(): return False pre_zone = zone - + if not self.start_zone(pre_zone): self.stdio.stop_loading('stop_loading', 'fail') return False - + self.cluster_config.servers = all_servers if self.new_cluster_config: self.new_cluster_config.servers = all_servers @@ -245,7 +267,7 @@ def restart(self): if self.connect(): self.stdio.start_loading('Server check') servers = self.execute_sql("select * from oceanbase.__all_server", one=False, error=False) - if len(self.cluster_config.servers) == len(servers): + if isinstance(servers, list) and len(self.cluster_config.servers) == len(servers): for server in servers: if server['status'] != 'active' or server['stop_time'] > 0 or server['start_service_time'] == 0: break @@ -264,13 +286,11 @@ def restart(self): ret = self.rolling(zones_servers) else: ret = self.un_rolling() - + if ret and self.connect(): - self.display_plugin(self.components, self.new_clients if self.new_clients else self.clients, self.new_cluster_config if self.new_cluster_config else self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, cursor=self.cursor) + self.call_plugin(self.display_plugin, clients=self.now_clients, cluster_config=self.new_cluster_config if self.new_cluster_config else self.cluster_config, cursor=self.cursor) if self.new_cluster_config: - self.stdio.verbose('Call %s for %s' % (self.reload_plugin, self.repository)) - self.reload_plugin(self.components, self.clients, self.cluster_config, [], {}, self.sub_io, - cursor=self.cursor, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir) + self.call_plugin(self.reload_plugin, clients=self.now_clients, cursor=self.cursor, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir) except Exception as e: self.stdio.exception('Run Exception: %s' % e) finally: @@ -282,7 +302,8 @@ def restart(self): return ret -def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, rollback=False, *args, **kwargs): +def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, new_cluster_config=None, new_clients=None, rollback=False, *args, **kwargs): + repository = kwargs.get('repository') task = Restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config, new_clients) call = task.rollback if rollback else task.restart if call(): diff --git a/plugins/oceanbase/3.1.0/start.py b/plugins/oceanbase/3.1.0/start.py index 62d3441..c401db3 100644 --- a/plugins/oceanbase/3.1.0/start.py +++ b/plugins/oceanbase/3.1.0/start.py @@ -20,13 +20,12 @@ from __future__ import absolute_import, division, print_function -import os import json import time import requests from copy import deepcopy -from _errno import EC_OBSERVER_FAIL_TO_START +from _errno import EC_OBSERVER_FAIL_TO_START, EC_OBSERVER_FAIL_TO_START_WITH_ERR, EC_OBSERVER_FAILED_TO_REGISTER, EC_OBSERVER_FAILED_TO_REGISTER_WITH_DETAILS from collections import OrderedDict @@ -81,7 +80,7 @@ def __exit__(self, *args, **kwargs): self.client.del_env(env_key) -def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def start(plugin_context, *args, **kwargs): cluster_config = plugin_context.cluster_config options = plugin_context.options clients = plugin_context.clients @@ -101,10 +100,10 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): try: cfg_url = init_config_server(obconfig_url, appname, cluster_id, getattr(options, 'force_delete', False), stdio) if not cfg_url: - stdio.error('failed to register cluster. %s may have been registered in %s.' % (appname, obconfig_url)) + stdio.error(EC_OBSERVER_FAILED_TO_REGISTER_WITH_DETAILS.format(appname, obconfig_url)) return except: - stdio.exception('failed to register cluster') + stdio.exception(EC_OBSERVER_FAILED_TO_REGISTER.format()) return stdio.start_loading('Start observer') @@ -156,7 +155,8 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): }) not_cmd_opt = [ 'home_path', 'obconfig_url', 'root_password', 'proxyro_password', - 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir', '$_zone_idc' + 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir', '$_zone_idc', + 'ocp_meta_tenant', 'ocp_meta_username', 'ocp_meta_password', 'ocp_meta_db', 'ocp_agent_monitor_password' ] get_value = lambda key: "'%s'" % server_config[key] if isinstance(server_config[key], str) else server_config[key] opt_str = [] @@ -189,7 +189,7 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): ret = client.execute_command(clusters_cmd[server]) if not ret: stdio.stop_loading('fail') - stdio.error(EC_OBSERVER_FAIL_TO_START.format(server=server) + ': ' + ret.stderr) + stdio.error(EC_OBSERVER_FAIL_TO_START_WITH_ERR.format(server=server, stderr=ret.stderr)) return stdio.stop_loading('succeed') diff --git a/plugins/oceanbase/3.1.0/start_check.py b/plugins/oceanbase/3.1.0/start_check.py index fe757f8..c5e9f47 100644 --- a/plugins/oceanbase/3.1.0/start_check.py +++ b/plugins/oceanbase/3.1.0/start_check.py @@ -24,11 +24,7 @@ import re import time -from _errno import ( - EC_OBSERVER_NOT_ENOUGH_DISK_4_CLOG, EC_CONFIG_CONFLICT_PORT, - EC_OBSERVER_NOT_ENOUGH_MEMORY, EC_ULIMIT_CHECK, WC_ULIMIT_CHECK, - EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE, EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED -) +import _errno as err stdio = None @@ -37,7 +33,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -75,58 +71,197 @@ def time_delta(client): return time_srv - time_st -def _start_check(plugin_context, strict_check=False, *args, **kwargs): - def alert(*arg, **kwargs): +def get_disk_info_by_path(path, client, stdio): + disk_info = {} + ret = client.execute_command('df --block-size=1024 {}'.format(path)) + if ret: + for total, used, avail, puse, path in re.findall(r'(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', ret.stdout): + disk_info[path] = {'total': int(total) << 10, 'avail': int(avail) << 10, 'need': 0, 'threshold': 2} + stdio.verbose('get disk info for path {}, total: {} avail: {}'.format(path, disk_info[path]['total'], disk_info[path]['avail'])) + return disk_info + + +def get_disk_info(all_paths, client, stdio): + overview_ret = True + disk_info = get_disk_info_by_path('', client, stdio) + if not disk_info: + overview_ret = False + disk_info = get_disk_info_by_path('/', client, stdio) + if not disk_info: + disk_info['/'] = {'total': 0, 'avail': 0, 'need': 0, 'threshold': 2} + all_path_success = {} + for path in all_paths: + all_path_success[path] = False + cur_path = path + while cur_path not in disk_info: + disk_info_for_current_path = get_disk_info_by_path(cur_path, client, stdio) + if disk_info_for_current_path: + disk_info.update(disk_info_for_current_path) + all_path_success[path] = True + break + else: + cur_path = os.path.dirname(cur_path) + if overview_ret or all(all_path_success.values()): + return disk_info + + +def start_check(plugin_context, init_check_status=False, strict_check=False, work_dir_check=False, work_dir_empty_check=True, generate_configs={}, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def alert(item, error, suggests=[]): global success if strict_check: success = False - stdio.error(*arg, **kwargs) + check_fail(item, error, suggests) + stdio.error(error) else: - stdio.warn(*arg, **kwargs) - def error(*arg, **kwargs): + stdio.warn(error) + def error(item, _error, suggests=[]): global success if plugin_context.dev_mode: - stdio.warn(*arg, **kwargs) + stdio.warn(_error) else: success = False - stdio.error(*arg, **kwargs) - def critical(*arg, **kwargs): + check_fail(item, _error, suggests) + stdio.error(_error) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) + check_fail(item, error, suggests) + stdio.error(error) - global stdio + global stdio, success + success = True + check_status = {} cluster_config = plugin_context.cluster_config + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + 'mem': err.CheckStatus(), + 'disk': err.CheckStatus(), + 'ulimit': err.CheckStatus(), + 'aio': err.CheckStatus(), + 'net': err.CheckStatus(), + 'ntp': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + clients = plugin_context.clients stdio = plugin_context.stdio + stdio.start_loading('Check before start observer') servers_clients = {} servers_port = {} servers_memory = {} servers_disk = {} servers_clog_mount = {} - servers_net_inferface = {} - server_num = len(cluster_config.servers) + servers_net_inferface = {} + servers_dirs = {} + servers_check_dirs = {} START_NEED_MEMORY = 3 << 30 - + global_generate_config = generate_configs.get('global', {}) stdio.start_loading('Check before start observer') for server in cluster_config.servers: ip = server.ip client = clients[server] + server_generate_config = generate_configs.get(server, {}) servers_clients[ip] = client server_config = cluster_config.get_server_conf_with_default(server) home_path = server_config['home_path'] - remote_pid_path = '%s/run/observer.pid' % home_path - remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s' % remote_pid): - continue + if not precheck: + remote_pid_path = '%s/run/observer.pid' % home_path + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + original_server_conf = cluster_config.get_server_conf(server) + + if not server_config.get('data_dir'): + server_config['data_dir'] = '%s/store' % home_path + if not server_config.get('redo_dir'): + server_config['redo_dir'] = server_config['data_dir'] + if not server_config.get('clog_dir'): + server_config['clog_dir'] = '%s/clog' % server_config['redo_dir'] + if not server_config.get('ilog_dir'): + server_config['ilog_dir'] = '%s/ilog' % server_config['redo_dir'] + if not server_config.get('slog_dir'): + server_config['slog_dir'] = '%s/slog' % server_config['redo_dir'] + if server_config['redo_dir'] == server_config['data_dir']: + keys = ['home_path', 'data_dir', 'clog_dir', 'ilog_dir', 'slog_dir'] + else: + keys = ['home_path', 'data_dir', 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir'] + + for key in keys: + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + if key not in original_server_conf: + continue + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False if ip not in servers_port: servers_disk[ip] = {} servers_port[ip] = {} servers_clog_mount[ip] = {} servers_net_inferface[ip] = {} - servers_memory[ip] = {'num': 0, 'percentage': 0, 'server_num': 0} + servers_memory[ip] = {'num': 0, 'percentage': 0, 'servers': {}} memory = servers_memory[ip] ports = servers_port[ip] disk = servers_disk[ip] @@ -136,40 +271,44 @@ def critical(*arg, **kwargs): for key in ['mysql_port', 'rpc_port']: port = int(server_config[key]) if port in ports: - critical(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key'])) + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) continue ports[port] = { 'server': server, 'key': key } if get_port_socket_inode(client, port): - critical('%s:%s port is already used' % (ip, port)) + critical( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) - memory['server_num'] += 1 + memory_limit = 0 + percentage = 0 if 'memory_limit' in server_config: - try: - memory['num'] += parse_size(server_config['memory_limit']) - except: - error('memory_limit must be an integer') - return + memory_limit = parse_size(server_config['memory_limit']) + memory['num'] += memory_limit elif 'memory_limit_percentage' in server_config: - try: - memory['percentage'] += int(parse_size(server_config['memory_limit_percentage'])) - except: - error('memory_limit_percentage must be an integer') - return + percentage = int(parse_size(server_config['memory_limit_percentage'])) + memory['percentage'] += percentage else: - memory['percentage'] += 80 + percentage = 80 + memory['percentage'] += percentage + memory['servers'][server] = { + 'num': memory_limit, + 'percentage': percentage, + 'system_memory': parse_size(server_config.get('system_memory', 0)) + } + data_path = server_config['data_dir'] if server_config.get('data_dir') else os.path.join(server_config['home_path'], 'store') redo_dir = server_config['redo_dir'] if server_config.get('redo_dir') else data_path clog_dir = server_config['clog_dir'] if server_config.get('clog_dir') else os.path.join(redo_dir, 'clog') if not client.execute_command('ls %s/sstable/block_file' % data_path): - if data_path in disk: - critical('Same Path: %s in %s and %s' % (data_path, server, disk[data_path]['server'])) - continue - if clog_dir in clog_mount: - critical('Same Path: %s in %s and %s' % (clog_dir, server, clog_mount[clog_dir]['server'])) - continue disk[data_path] = { 'need': 90, 'server': server @@ -186,38 +325,50 @@ def critical(*arg, **kwargs): devname = server_config.get('devname') if devname: if not client.execute_command("grep -e '^ *%s:' /proc/net/dev" % devname): - critical('%s No such net interface: %s' % (server, devname)) + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + suggest.auto_fix = 'devname' not in global_generate_config and 'devname' not in server_generate_config + critical('net', err.EC_NO_SUCH_NET_DEVICE.format(server=server, devname=devname), suggests=[suggest]) if devname not in inferfaces: inferfaces[devname] = [] inferfaces[devname].append(ip) for ip in servers_disk: client = servers_clients[ip] + ip_servers = servers_memory[ip]['servers'].keys() + server_num = len(ip_servers) + ret = client.execute_command('cat /proc/sys/fs/aio-max-nr /proc/sys/fs/aio-nr') if not ret: - alert('(%s) failed to get fs.aio-max-nr and fs.aio-nr' % ip) + for server in ip_servers: + alert('aio', err.EC_FAILED_TO_GET_AIO_NR.format(ip=ip), [err.SUG_CONNECT_EXCEPT.format()]) else: try: max_nr, nr = ret.stdout.strip().split('\n') max_nr, nr = int(max_nr), int(nr) need = server_num * 20000 + RECD_AIO = 1048576 if need > max_nr - nr: - critical('(%s) Insufficient AIO remaining (Avail: %s, Need: %s), The recommended value of fs.aio-max-nr is 1048576' % (ip, max_nr - nr, need)) - elif int(max_nr) < 1048576: - alert('(%s) The recommended value of fs.aio-max-nr is 1048576 (Current value: %s)' % (ip, max_nr)) + for server in ip_servers: + critical('aio', err.EC_AIO_NOT_ENOUGH.format(ip=ip, avail=max_nr - nr, need=need), [err.SUG_SYSCTL.format(var='fs.aio-max-nr', value=max(RECD_AIO, need), ip=ip)]) + elif int(max_nr) < RECD_AIO: + for server in ip_servers: + alert('aio', err.WC_AIO_NOT_ENOUGH.format(ip=ip, current=max_nr), [err.SUG_SYSCTL.format(var='fs.aio-max-nr', value=RECD_AIO, ip=ip)]) except: - alert('(%s) failed to get fs.aio-max-nr and fs.aio-nr' % ip) + for server in ip_servers: + alert('aio', err.EC_FAILED_TO_GET_AIO_NR.format(ip=ip), [err.SUG_UNSUPPORT_OS.format()]) stdio.exception('') ret = client.execute_command('ulimit -a') ulimits_min = { 'open files': { 'need': lambda x: 20000 * x, - 'recd': lambda x: 655350 + 'recd': lambda x: 655350, + 'name': 'nofile' }, 'max user processes': { 'need': lambda x: 4096, - 'recd': lambda x: 4096 * x + 'recd': lambda x: 4096 * x, + 'name': 'nproc' }, } ulimits = {} @@ -229,16 +380,19 @@ def critical(*arg, **kwargs): if value == 'unlimited': continue if not value or not (value.strip().isdigit()): - alert('(%s) failed to get %s' % (ip, key)) + for server in ip_servers: + alert('ulimit', '(%s) failed to get %s' % (ip, key), [err.SUG_UNSUPPORT_OS.format()]) else: value = int(value) need = ulimits_min[key]['need'](server_num) if need > value: - critical(EC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value)) + for server in ip_servers: + critical('ulimit', err.EC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value), [err.SUG_ULIMIT.format(name=ulimits_min[key]['name'], value=need, ip=ip)]) else: need = ulimits_min[key]['recd'](server_num) if need > value: - alert(WC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value)) + for server in ip_servers: + alert('ulimit', err.WC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value), [err.SUG_ULIMIT.format(name=ulimits_min[key]['name'], value=need, ip=ip)]) # memory ret = client.execute_command('cat /proc/meminfo') @@ -257,42 +411,45 @@ def critical(*arg, **kwargs): if k in memory_key_map: key = memory_key_map[k] server_memory_stats[key] = parse_size(str(v)) - - min_start_need = servers_memory[ip]['server_num'] * START_NEED_MEMORY - total_use = servers_memory[ip]['percentage'] * server_memory_stats['total'] / 100 + servers_memory[ip]['num'] + + server_memory_stat = servers_memory[ip] + min_start_need = server_num * START_NEED_MEMORY + total_use = server_memory_stat['percentage'] * server_memory_stats['total'] / 100 + server_memory_stat['num'] if min_start_need > server_memory_stats['available']: - error(EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(min_start_need))) + for server in ip_servers: + error('mem', err.EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(min_start_need)), [err.SUG_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip)]) elif total_use > server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached']: - error(EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(total_use))) + for server in ip_servers: + server_generate_config = generate_configs.get(server, {}) + suggest = err.SUG_OBSERVER_REDUCE_MEM.format() + suggest.auto_fix = True + for key in ['memory_limit', 'memory_limit_percentage']: + if key in global_generate_config or key in server_generate_config: + suggest.auto_fix = False + break + error('mem', err.EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(total_use)), [suggest]) elif total_use > server_memory_stats['free']: - alert(EC_OBSERVER_NOT_ENOUGH_MEMORY.format(ip=ip, free=format_size(server_memory_stats['free']), need=format_size(total_use))) + for server in ip_servers: + alert('mem', err.EC_OBSERVER_NOT_ENOUGH_MEMORY.format(ip=ip, free=format_size(server_memory_stats['free']), need=format_size(total_use)), [err.SUG_OBSERVER_REDUCE_MEM.format()]) + else: + server_memory_config = server_memory_stat['servers'] + for server in server_memory_config: + if server_memory_config[server]['system_memory']: + memory_limit = server_memory_config[server]['num'] + if not memory_limit: + memory_limit = server_memory_config[server]['percentage'] * server_memory_stats['total'] + + factor = 0.7 + suggest = err.SUG_OBSERVER_SYS_MEM_TOO_LARGE.format(factor=factor) + suggest.auto_fix = 'system_memory' not in global_generate_config and 'system_memory' not in generate_configs.get(server, {}) + if memory_limit < server_memory_config[server]['system_memory']: + critical('mem', err.EC_OBSERVER_SYS_MEM_TOO_LARGE.format(server=server), [suggest]) + elif memory_limit * factor < server_memory_config[server]['system_memory']: + alert('mem', err.WC_OBSERVER_SYS_MEM_TOO_LARGE.format(server=server, factor=factor), [suggest]) + # disk - disk = {'/': 0} - ret = client.execute_command('df --block-size=1024') - if ret: - for total, used, avail, puse, path in re.findall('(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', ret.stdout): - disk[path] = { - 'total': int(total) << 10, - 'avail': int(avail) << 10, - 'need': 0, - 'threshold': 2 - } all_path = set(list(servers_disk[ip].keys()) + list(servers_clog_mount[ip].keys())) - for include_dir in all_path: - while include_dir not in disk: - ret = client.execute_command('df --block-size=1024 %s' % include_dir) - if ret: - for total, used, avail, puse, path in re.findall('(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', - ret.stdout): - disk[path] = { - 'total': int(total) << 10, - 'avail': int(avail) << 10, - 'need': 0, - 'threshold': 2 - } - break - else: - include_dir = os.path.dirname(include_dir) + disk = get_disk_info(all_paths=all_path, client=client, stdio=stdio) stdio.verbose('disk: {}'.format(disk)) for path in servers_disk[ip]: kp = '/' @@ -304,11 +461,7 @@ def critical(*arg, **kwargs): if isinstance(need, int): disk[kp]['need'] += disk[kp]['total'] * need / 100 else: - try: - disk[kp]['need'] += parse_size(need) - except: - critical('datafile_size must be an integer') - return + disk[kp]['need'] += parse_size(need) for path in servers_clog_mount[ip]: kp = '/' @@ -323,15 +476,34 @@ def critical(*arg, **kwargs): avail = disk[p]['avail'] need = disk[p]['need'] threshold = disk[p]['threshold'] + suggests = [] if need > 0 and threshold < 2: - alert('(%s) clog and data use the same disk (%s)' % (ip, p)) + suggests.append(err.SUG_OBSERVER_SAME_DISK.format()) + for server in ip_servers: + alert('disk', err.WC_OBSERVER_SAME_DISK.format(ip=ip, disk=p), suggests) if need > avail: - critical('(%s) %s not enough disk space. (Avail: %s, Need: %s)' % (ip, p, format_size(avail), format_size(need))) + for server in ip_servers: + server_generate_config = generate_configs.get(server, {}) + suggest = err.SUG_OBSERVER_NOT_ENOUGH_DISK.format() + suggest.auto_fix = True + for key in ['datafile_size', 'datafile_disk_percentage']: + if key in global_generate_config or key in server_generate_config: + suggest.auto_fix = False + break + critical('disk', err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=p, avail=format_size(avail), need=format_size(need)), [suggest] + suggests) elif 1.0 * (total - avail + need) / total > disk[p]['threshold']: # msg = '(%s) %s not enough disk space for clog. Use `redo_dir` to set other disk for clog' % (ip, p) # msg += ', or reduce the value of `datafile_size`' if need > 0 else '.' # critical(msg) - critical(EC_OBSERVER_NOT_ENOUGH_DISK_4_CLOG.format(ip=ip, path=p)) + for server in ip_servers: + server_generate_config = generate_configs.get(server, {}) + suggest = err.SUG_OBSERVER_NOT_ENOUGH_DISK_4_CLOG.format() + suggest.auto_fix = True + for key in ['clog_disk_utilization_threshold', 'clog_disk_usage_limit_percentage']: + if key in global_generate_config or key in server_generate_config: + suggest.auto_fix = False + break + critical('disk', err.EC_OBSERVER_NOT_ENOUGH_DISK_4_CLOG.format(ip=ip, path=p), [suggest] + suggests) if success: for ip in servers_net_inferface: @@ -345,7 +517,9 @@ def critical(*arg, **kwargs): interfaces = ['lo'] if len(interfaces) > 1: servers = ','.join(str(server) for server in servers_net_inferface[ip][None]) - critical('%s has more than one network inferface. Please set `devname` for (%s)' % (ip, servers)) + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + for server in ip_servers: + critical('net', err.EC_OBSERVER_MULTI_NET_DEVICE.format(ip=ip, server=servers), [suggest]) else: servers_net_inferface[ip][interfaces[0]] = servers_net_inferface[ip][None] del servers_net_inferface[ip][None] @@ -354,13 +528,19 @@ def critical(*arg, **kwargs): client = servers_clients[ip] for devname in servers_net_inferface[ip]: if client.is_localhost() and devname != 'lo' or (not client.is_localhost() and devname == 'lo'): - critical('%s %s fail to ping %s. Please check configuration `devname`' % (server, devname, ip)) - continue + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + suggest.auto_fix = client.is_localhost() and 'devname' not in global_generate_config and 'devname' not in server_generate_config + for server in ip_servers: + critical('net', err.EC_OBSERVER_PING_FAILED.format(ip1=ip, devname=devname, ip2=ip), [suggest]) + continue for _ip in servers_clients: if ip == _ip: continue if not client.execute_command('ping -W 1 -c 1 -I %s %s' % (devname, _ip)): - critical('%s %s fail to ping %s. Please check configuration `devname`' % (server, devname, _ip)) + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + suggest.auto_fix = 'devname' not in global_generate_config and 'devname' not in server_generate_config + for server in ip_servers: + critical('net', err.EC_OBSERVER_PING_FAILED.format(ip1=ip, devname=devname, ip2=_ip), [suggest]) break if success: @@ -371,11 +551,11 @@ def critical(*arg, **kwargs): stdio.verbose('%s time delta %s' % (ip, delta)) times.append(delta) if times and max(times) - min(times) > 200: - critical('Cluster NTP is out of sync') + critical('ntp', err.EC_OBSERVER_TIME_OUT_OF_SYNC, [err.SUG_OBSERVER_TIME_OUT_OF_SYNC.format()]) + for server in cluster_config.servers: + wait_2_pass() -def start_check(plugin_context, strict_check=False, *args, **kwargs): - _start_check(plugin_context, strict_check) if success: stdio.stop_loading('succeed') plugin_context.return_true() diff --git a/plugins/oceanbase/3.1.0/stop.py b/plugins/oceanbase/3.1.0/stop.py index 740aa1c..ec70b9b 100644 --- a/plugins/oceanbase/3.1.0/stop.py +++ b/plugins/oceanbase/3.1.0/stop.py @@ -38,7 +38,7 @@ def config_url(ocp_config_server, appname, cid): def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return [] diff --git a/plugins/oceanbase/3.1.0/upgrade.py b/plugins/oceanbase/3.1.0/upgrade.py index 911b60c..03a6132 100644 --- a/plugins/oceanbase/3.1.0/upgrade.py +++ b/plugins/oceanbase/3.1.0/upgrade.py @@ -205,11 +205,18 @@ def _clear_plugin(self): self._stop_plugin = None self._display_plugin = None + def call_plugin(self, plugin, *args, **kwargs): + return plugin(self.plugin_context.namespace, self.plugin_context.namespaces, self.plugin_context.deploy_name, + self.plugin_context.repositories, self.plugin_context.components, self.plugin_context.clients, + self.plugin_context.cluster_config, self.plugin_context.cmds, self.plugin_context.options, + self.plugin_context.stdio, *args, **kwargs) + def run(self): total = len(self.route) self.apply_param_plugin(self.repositories[self.route_index - 1]) while self.route_index < total: - self.start_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.stdio, local_home_path=None, repository_dir=None) + self.call_plugin(self.start_plugin, local_home_path=None, repository_dir=None) + self.close() if not self.connect(): return False self.stdio.verbose('upgrade %s to %s' % (self.repositories[self.route_index], self.repositories[self.next_stage])) @@ -241,17 +248,17 @@ def _dump(self): def close(self): if self.db: self.cursor.close() - self.db.close() self.cursor = None self.db = None self.exector = None def connect(self): if self.cursor is None or self.execute_sql('select version()', error=False) is False: - ret = self.connect_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.stdio) + ret = self.call_plugin(self.connect_plugin) if not ret: return False - self.close() + if self.cursor: + self.close() self.cursor = ret.get_return('cursor') self.db = ret.get_return('connect') while self.execute_sql('use oceanbase', error=False) is False: @@ -266,18 +273,13 @@ def connect(self): return True def execute_sql(self, query, args=None, one=True, error=True): - msg = query % tuple(args) if args is not None else query - self.stdio.verbose("query: %s. args: %s" % (query, args)) - try: - self.stdio.verbose('execute sql: %s' % msg) - self.cursor.execute(query, args) - result = self.cursor.fetchone() if one else self.cursor.fetchall() - result and self.stdio.verbose(result) - return result - except: - msg = 'execute sql exception: %s' % msg if error else '' - self.stdio.exception(msg) - return False + exc_level = 'error' if error else 'verbose' + if one: + result = self.cursor.fetchone(query, args, exc_level=exc_level) + else: + result = self.cursor.fetchall(query, args, exc_level=exc_level) + result and self.stdio.verbose(result) + return result @property def next_stage(self): @@ -418,15 +420,15 @@ def un_rolling_upgrade(self): repository_dir = repository.repository_dir self.install_repository_to_servers(self.components, self.cluster_config, repository, self.clients, self.unuse_lib_repository) - - if not self.stop_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.stdio): + if not self.call_plugin(self.stop_plugin): self.stdio.stop_loading('stop_loading', 'fail') return False self.apply_param_plugin(repository) - if not self.start_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.stdio, local_home_path=self.local_home_path, repository_dir=repository_dir): + if not self.call_plugin(self.start_plugin, local_home_path=self.local_home_path, repository_dir=repository_dir): self.stdio.stop_loading('stop_loading', 'fail') return False + self.close() self.wait() self.stdio.stop_loading('succeed') return True @@ -469,14 +471,15 @@ def rolling_upgrade(self, zones_servers): if pre_zone: self.apply_param_plugin(self.repositories[self.route_index - 1]) - if not self.stop_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.stdio): + if not self.call_plugin(self.stop_plugin): self.stdio.stop_loading('stop_loading', 'fail') return False self.apply_param_plugin(repository) - if not self.start_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.stdio, local_home_path=self.local_home_path, repository_dir=repository_dir): + if not self.call_plugin(self.start_plugin, local_home_path=self.local_home_path, repository_dir=repository_dir): self.stdio.stop_loading('stop_loading', 'fail') return False + self.close() pre_zone = zone if not self.start_zone(pre_zone): @@ -517,17 +520,11 @@ def exec_upgrade_post_checker(self): def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, install_repository_to_servers, unuse_lib_repository, *args, **kwargs): - components = plugin_context.components - clients = plugin_context.clients - cluster_config = plugin_context.cluster_config - cmd = plugin_context.cmd - options = plugin_context.options - stdio = plugin_context.stdio upgrade_ctx = kwargs.get('upgrade_ctx') local_home_path = kwargs.get('local_home_path') upgrade_repositories = kwargs.get('upgrade_repositories') - exector_path = getattr(options, 'executer_path', '/usr/obd/lib/executer') + exector_path = getattr(plugin_context.options, 'executer_path', '/usr/obd/lib/executer') upgrader = Upgrader( plugin_context=plugin_context, @@ -541,5 +538,6 @@ def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, install unuse_lib_repository=unuse_lib_repository) if upgrader.run(): if upgrader.route_index >= len(upgrader.route): - upgrader.display_plugin(components, clients, cluster_config, cmd, options, stdio, upgrader.cursor, *args, **kwargs) + upgrader.call_plugin(upgrader.display_plugin, upgrader.cursor, *args, **kwargs) plugin_context.return_true() + diff --git a/plugins/oceanbase/3.1.0/upgrade_check.py b/plugins/oceanbase/3.1.0/upgrade_check.py index 42ca225..4fdb3c5 100644 --- a/plugins/oceanbase/3.1.0/upgrade_check.py +++ b/plugins/oceanbase/3.1.0/upgrade_check.py @@ -23,21 +23,9 @@ import os -def upgrade_check(plugin_context, current_repository, repositories, route, cursor, *args, **kwargs): - def execute_sql(query, args=None, one=True, error=True): - msg = query % tuple(args) if args is not None else query - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - stdio.verbose('execute sql: %s' % msg) - cursor.execute(query, args) - result = cursor.fetchone() if one else cursor.fetchall() - result and stdio.verbose(result) - return result - except: - msg = 'execute sql exception: %s' % msg if error else '' - stdio.exception(msg) - return False +def upgrade_check(plugin_context, current_repository, route, cursor, *args, **kwargs): + repositories = plugin_context.repositories options = plugin_context.options stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config @@ -53,7 +41,10 @@ def execute_sql(query, args=None, one=True, error=True): zones.add(zone) if len(zones) > 2: - tenants = execute_sql('select * from oceanbase.gv$tenant', one=False) + tenants = cursor.fetchall('select * from oceanbase.gv$tenant') + if tenants is False: + return + tenants and stdio.verbose(tenants) for tenant in tenants: zone_list = tenant.get('zone_list', '').split(';') if len(zone_list) < 3: diff --git a/plugins/oceanbase/3.1.0/upgrade_file_check.py b/plugins/oceanbase/3.1.0/upgrade_file_check.py index 1b909da..bebcd0c 100644 --- a/plugins/oceanbase/3.1.0/upgrade_file_check.py +++ b/plugins/oceanbase/3.1.0/upgrade_file_check.py @@ -23,7 +23,9 @@ import os -def upgrade_file_check(plugin_context, current_repository, repositories, route, *args, **kwargs): +def upgrade_file_check(plugin_context, route, *args, **kwargs): + current_repository = kwargs.get('repository') + repositories = plugin_context.repositories options = plugin_context.options stdio = plugin_context.stdio diff --git a/plugins/oceanbase/4.0.0.0/bootstrap.py b/plugins/oceanbase/4.0.0.0/bootstrap.py index 6ebfa14..111cd5b 100644 --- a/plugins/oceanbase/4.0.0.0/bootstrap.py +++ b/plugins/oceanbase/4.0.0.0/bootstrap.py @@ -20,13 +20,9 @@ from __future__ import absolute_import, division, print_function -import sys import time - -if sys.version_info.major == 2: - from MySQLdb import DatabaseError -else: - from pymysql.err import DatabaseError +from copy import deepcopy +from optparse import Values from _deploy import InnerConfigItem @@ -43,24 +39,16 @@ def bootstrap(plugin_context, cursor, *args, **kwargs): def is_bootstrap(): sql = "select column_value from oceanbase.__all_core_table where table_name = '__all_global_stat' and column_name = 'baseline_schema_version'" - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - return int(cursor.fetchone().get("column_value")) > 0 - - try: - if is_bootstrap(): - return plugin_context.return_true() - except DatabaseError as e: - stdio.verbose('%s:%s' % e.args) - except: - stdio.exception("") + ret = cursor.fetchone(sql, raise_exception=False, exc_level='verbose') + if ret is False: + return False + return int(ret.get("column_value")) > 0 has_obproxy = False for componet_name in ['obproxy', 'obproxy-ce']: if componet_name in plugin_context.components: has_obproxy = True break - inner_keys = inner_config.keys() for server in cluster_config.servers: server_config = cluster_config.get_server_conf(server) zone = server_config['zone'] @@ -81,46 +69,75 @@ def is_bootstrap(): continue zone_config[key] = server_config[key] try: + raise_cursor = cursor.raise_cursor + sql = 'set session ob_query_timeout=1000000000' + stdio.verbose('execute sql: %s' % sql) + raise_cursor.execute(sql) sql = 'alter system bootstrap %s' % (','.join(bootstrap)) stdio.start_loading('Cluster bootstrap') - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + raise_cursor.execute(sql, exc_level='verbose') for zone in floor_servers: for addr in floor_servers[zone]: sql = 'alter system add server "%s" zone "%s"' % (addr, zone) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + raise_cursor.execute(sql) global_conf = cluster_config.get_global_conf() if has_obproxy or 'proxyro_password' in global_conf: value = global_conf['proxyro_password'] if global_conf.get('proxyro_password') is not None else '' sql = 'create user "proxyro" IDENTIFIED BY %s' - stdio.verbose(sql) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) sql = 'grant select on oceanbase.* to proxyro IDENTIFIED BY %s' + raise_cursor.execute(sql, [value]) + + has_obagent = "obagent" in plugin_context.components + if has_obagent or 'ocp_agent_monitor_password' in global_conf: + value = global_conf['ocp_agent_monitor_password'] if global_conf.get('ocp_agent_monitor_password') is not None else '' + sql = 'create user "ocp_monitor" IDENTIFIED BY %s' + stdio.verbose(sql) + raise_cursor.execute(sql, [value]) + sql = 'grant select on oceanbase.* to ocp_monitor IDENTIFIED BY %s' stdio.verbose(sql) - cursor.execute(sql, [value]) + raise_cursor.execute(sql, [value]) + if global_conf.get('root_password') is not None: - sql = 'alter user "root" IDENTIFIED BY "%s"' % global_conf.get('root_password') - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) + sql = 'alter user "root" IDENTIFIED BY %s' + raise_cursor.execute(sql, [global_conf.get('root_password')]) for zone in zones_config: zone_config = zones_config[zone] for key in zone_config: sql = 'alter system modify zone %s set %s = %%s' % (zone, inner_config[key]) - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql, [zone_config[key]]) + raise_cursor.execute(sql, [zone_config[key]]) stdio.stop_loading('succeed') - plugin_context.return_true() except: - stdio.exception('') - try: - sql = 'set session ob_query_timeout=1000000000' - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - if is_bootstrap(): - stdio.stop_loading('succeed') - return plugin_context.return_true() - except: - stdio.exception('') - stdio.stop_loading('fail') - return plugin_context.return_false() + if not is_bootstrap(): + stdio.stop_loading('fail') + return plugin_context.return_false() + stdio.stop_loading('succeed') + return plugin_context.return_true() + + has_ocp = 'ocp-express' in plugin_context.components + if any([key in global_conf for key in ["ocp_meta_tenant", "ocp_meta_db", "ocp_meta_username", "ocp_meta_password"]]): + has_ocp = True + if has_ocp: + global_conf_with_default = deepcopy(cluster_config.get_global_conf_with_default()) + ocp_meta_tenant_prefix = 'ocp_meta_tenant_' + for key in global_conf_with_default: + if key.startswith(ocp_meta_tenant_prefix): + global_conf_with_default['ocp_meta_tenant'][key.replace(ocp_meta_tenant_prefix, '', 1)] = global_conf_with_default[key] + tenant_info = global_conf_with_default["ocp_meta_tenant"] + tenant_info["variables"] = "ob_tcp_invited_nodes='%'" + tenant_info["create_if_not_exists"] = True + tenant_info["database"] = global_conf_with_default["ocp_meta_db"] + tenant_info["db_username"] = global_conf_with_default["ocp_meta_username"] + tenant_info["db_password"] = global_conf_with_default.get("ocp_meta_password", "") + tenant_options = Values(tenant_info) + plugin_context.set_variable("create_tenant_options", tenant_options) + # wait for server online + all_server_online = False + while not all_server_online: + servers = cursor.fetchall('select * from oceanbase.__all_server', raise_exception=False, exc_level='verbose') + if servers and all([s.get('status') for s in servers]): + all_server_online = True + else: + time.sleep(1) + + return plugin_context.return_true() diff --git a/plugins/oceanbase/4.0.0.0/create_tenant.py b/plugins/oceanbase/4.0.0.0/create_tenant.py index 6315ccd..09cf244 100644 --- a/plugins/oceanbase/4.0.0.0/create_tenant.py +++ b/plugins/oceanbase/4.0.0.0/create_tenant.py @@ -26,6 +26,8 @@ from _errno import EC_OBSERVER_CAN_NOT_MIGRATE_IN +tenant_cursor = None + def parse_size(size): _bytes = 0 @@ -55,7 +57,19 @@ def format_size(size, precision=1): return format % (size, units[idx]) -def create_tenant(plugin_context, cursor, *args, **kwargs): +def exec_sql_in_tenant(sql, cursor, tenant, mode, retries=10): + global tenant_cursor + if not tenant_cursor: + user = 'SYS' if mode == 'oracle' else 'root' + tenant_cursor = cursor.new_cursor(tenant=tenant, user=user) + if not tenant_cursor and retries: + retries -= 1 + time.sleep(2) + return exec_sql_in_tenant(sql, cursor, tenant, mode, retries) + return tenant_cursor.execute(sql) + + +def create_tenant(plugin_context, cursor, create_tenant_options=None, *args, **kwargs): def get_option(key, default=''): value = getattr(options, key, default) if not value: @@ -76,14 +90,12 @@ def get_parsed_option(key, default=''): def error(*arg, **kwargs): stdio.error(*arg, **kwargs) stdio.stop_loading('fail') - def exception(*arg, **kwargs): - stdio.exception(*arg, **kwargs) - stdio.stop_loading('fail') cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio - options = plugin_context.options - + options = create_tenant_options if create_tenant_options else plugin_context.options + create_if_not_exists = get_option('create_if_not_exists', False) + mode = get_option('mode', 'mysql').lower() if not mode in ['mysql', 'oracle']: error('No such tenant mode: %s.\n--mode must be `mysql` or `oracle`' % mode) @@ -98,64 +110,60 @@ def exception(*arg, **kwargs): name = get_option('tenant_name', 'test') unit_name = '%s_unit' % name sql = 'select * from oceanbase.DBA_OB_UNIT_CONFIGS order by name' - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) + res = cursor.fetchall(sql) + if res is False: return - - res = cursor.fetchall() for row in res: if str(row['NAME']) == unit_name: unit_name += '1' pool_name = '%s_pool' % name - - stdio.start_loading('Create tenant %s' % name) + sql = "select * from oceanbase.DBA_OB_TENANTS where TENANT_NAME = %s" - try: - stdio.verbose('execute sql: %s' % (sql % name)) - cursor.execute(sql, [name]) - if cursor.fetchone(): + tenant_exists = False + res = cursor.fetchone(sql, [name]) + if res: + if create_if_not_exists: + tenant_exists = True + else: error('Tenant %s already exists' % name) return - except: - exception('execute sql exception: %s' % (sql % name)) + elif res is False: return + if not tenant_exists: + stdio.start_loading('Create tenant %s' % name) + zone_list = get_option('zone_list', set()) + zone_obs_num = {} + sql = "select zone, count(*) num from oceanbase.__all_server where status = 'active' group by zone" + res = cursor.fetchall(sql) + if res is False: + error() + return - zone_list = get_option('zone_list', set()) - zone_obs_num = {} - sql = "select zone, count(*) num from oceanbase.__all_server where status = 'active' group by zone" - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - res = cursor.fetchall() for row in res: zone_obs_num[str(row['zone'])] = row['num'] - except: - exception('execute sql exception: %s' % sql) - return - if not zone_list: - zone_list = zone_obs_num.keys() - if isinstance(zone_list, str): - zones = zone_list.replace(';', ',').split(',') - else: - zones = zone_list - zone_list = "('%s')" % "','".join(zones) - - min_unit_num = min(zone_obs_num.items(),key=lambda x: x[1])[1] - unit_num = get_option('unit_num', min_unit_num) - if unit_num > min_unit_num: - return error('resource pool unit num is bigger than zone server count') - - sql = "select count(*) num from oceanbase.__all_server where status = 'active' and start_service_time > 0" - try: + + if not zone_list: + zone_list = zone_obs_num.keys() + if isinstance(zone_list, str): + zones = zone_list.replace(';', ',').split(',') + else: + zones = zone_list + zone_list = "('%s')" % "','".join(zones) + + min_unit_num = min(zone_obs_num.items(), key=lambda x: x[1])[1] + unit_num = get_option('unit_num', min_unit_num) + if unit_num > min_unit_num: + return error('resource pool unit num is bigger than zone server count') + + sql = "select count(*) num from oceanbase.__all_server where status = 'active' and start_service_time > 0" count = 30 while count: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - num = cursor.fetchone()['num'] + num = cursor.fetchone(sql) + if num is False: + error() + return + num = num['num'] if num >= unit_num: break count -= 1 @@ -163,146 +171,156 @@ def exception(*arg, **kwargs): if count == 0: stdio.error(EC_OBSERVER_CAN_NOT_MIGRATE_IN) return - except: - exception('execute sql exception: %s' % sql) - return - sql = "SELECT * FROM oceanbase.GV$OB_SERVERS where zone in %s" - try: - sql = sql % zone_list - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) - return - servers_stats = cursor.fetchall() - cpu_available = servers_stats[0]['CPU_CAPACITY_MAX'] - servers_stats[0]['CPU_ASSIGNED_MAX'] - mem_available = servers_stats[0]['MEM_CAPACITY'] - servers_stats[0]['MEM_ASSIGNED'] - disk_available = servers_stats[0]['DATA_DISK_CAPACITY'] - servers_stats[0]['DATA_DISK_IN_USE'] - log_disk_available = servers_stats[0]['LOG_DISK_CAPACITY'] - servers_stats[0]['LOG_DISK_ASSIGNED'] - for servers_stat in servers_stats[1:]: - cpu_available = min(servers_stat['CPU_CAPACITY_MAX'] - servers_stat['CPU_ASSIGNED_MAX'], cpu_available) - mem_available = min(servers_stat['MEM_CAPACITY'] - servers_stat['MEM_ASSIGNED'], mem_available) - disk_available = min(servers_stat['DATA_DISK_CAPACITY'] - servers_stat['DATA_DISK_IN_USE'], disk_available) - log_disk_available = min(servers_stat['LOG_DISK_CAPACITY'] - servers_stat['LOG_DISK_ASSIGNED'], log_disk_available) - - MIN_CPU = 1 - MIN_MEMORY = 1073741824 - MIN_LOG_DISK_SIZE = 2147483648 - MIN_IOPS = 1024 - - if cpu_available < MIN_CPU: - return error('%s: resource not enough: cpu count less than %s' % (zone_list, MIN_CPU)) - if mem_available < MIN_MEMORY: - return error('%s: resource not enough: memory less than %s' % (zone_list, format_size(MIN_MEMORY))) - if log_disk_available < MIN_LOG_DISK_SIZE: - return error('%s: resource not enough: log disk size less than %s' % (zone_list, format_size(MIN_MEMORY))) - - # cpu options - max_cpu = get_option('max_cpu', cpu_available) - min_cpu = get_option('min_cpu', max_cpu) - if cpu_available < max_cpu: - return error('resource not enough: cpu (Avail: %s, Need: %s)' % (cpu_available, max_cpu)) - if max_cpu < min_cpu: - return error('min_cpu must less then max_cpu') - - # memory options - memory_size = get_parsed_option('memory_size', None) - log_disk_size = get_parsed_option('log_disk_size', None) - - if memory_size is None: - memory_size = mem_available - if log_disk_size is None: - log_disk_size = log_disk_available - - if mem_available < memory_size: - return error('resource not enough: memory (Avail: %s, Need: %s)' % (format_size(mem_available), format_size(memory_size))) - - # log disk size options - if log_disk_size is not None and log_disk_available < log_disk_size: - return error('resource not enough: log disk space (Avail: %s, Need: %s)' % (format_size(disk_available), format_size(log_disk_size))) - - # iops options - max_iops = get_option('max_iops', None) - min_iops = get_option('min_iops', None) - iops_weight = get_option('iops_weight', None) - if max_iops is not None and max_iops < MIN_IOPS: - return error('max_iops must greater than %d' % MIN_IOPS) - if max_iops is not None and min_iops is not None and max_iops < min_iops: - return error('min_iops must less then max_iops') - - zone_num = len(zones) - charset = get_option('charset', '') - collate = get_option('collate', '') - replica_num = get_option('replica_num', zone_num) - logonly_replica_num = get_option('logonly_replica_num', 0) - tablegroup = get_option('tablegroup', '') - primary_zone = get_option('primary_zone', 'RANDOM') - locality = get_option('locality', '') - variables = get_option('variables', '') - - if replica_num == 0: - replica_num = zone_num - elif replica_num > zone_num: - return error('replica_num cannot be greater than zone num (%s)' % zone_num) - if not primary_zone: - primary_zone = 'RANDOM' - if logonly_replica_num > replica_num: - return error('logonly_replica_num cannot be greater than replica_num (%s)' % replica_num) - - # create resource unit - sql = "create resource unit %s max_cpu %.1f, memory_size %d" % (unit_name, max_cpu, memory_size) - if min_cpu is not None: - sql += ', min_cpu %.1f' % min_cpu - if max_iops is not None: - sql += ', max_iops %d' % max_iops - if min_iops is not None: - sql += ', min_iops %d' % min_iops - if iops_weight is not None: - sql += ', iops_weight %d' % iops_weight - if log_disk_size is not None: - sql += ', log_disk_size %d' % log_disk_size - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('faild to crate unit, execute sql exception: %s' % sql) - return + sql = "SELECT * FROM oceanbase.GV$OB_SERVERS where zone in %s" % zone_list + servers_stats = cursor.fetchall(sql) + if servers_stats is False: + error() + return + cpu_available = servers_stats[0]['CPU_CAPACITY_MAX'] - servers_stats[0]['CPU_ASSIGNED_MAX'] + mem_available = servers_stats[0]['MEM_CAPACITY'] - servers_stats[0]['MEM_ASSIGNED'] + disk_available = servers_stats[0]['DATA_DISK_CAPACITY'] - servers_stats[0]['DATA_DISK_IN_USE'] + log_disk_available = servers_stats[0]['LOG_DISK_CAPACITY'] - servers_stats[0]['LOG_DISK_ASSIGNED'] + for servers_stat in servers_stats[1:]: + cpu_available = min(servers_stat['CPU_CAPACITY_MAX'] - servers_stat['CPU_ASSIGNED_MAX'], cpu_available) + mem_available = min(servers_stat['MEM_CAPACITY'] - servers_stat['MEM_ASSIGNED'], mem_available) + disk_available = min(servers_stat['DATA_DISK_CAPACITY'] - servers_stat['DATA_DISK_IN_USE'], disk_available) + log_disk_available = min(servers_stat['LOG_DISK_CAPACITY'] - servers_stat['LOG_DISK_ASSIGNED'], log_disk_available) + + MIN_CPU = 1 + MIN_MEMORY = 1073741824 + MIN_LOG_DISK_SIZE = 2147483648 + MIN_IOPS = 1024 + + if cpu_available < MIN_CPU: + return error('%s: resource not enough: cpu count less than %s' % (zone_list, MIN_CPU)) + if mem_available < MIN_MEMORY: + return error('%s: resource not enough: memory less than %s' % (zone_list, format_size(MIN_MEMORY))) + if log_disk_available < MIN_LOG_DISK_SIZE: + return error('%s: resource not enough: log disk size less than %s' % (zone_list, format_size(MIN_MEMORY))) + + # cpu options + max_cpu = get_option('max_cpu', cpu_available) + min_cpu = get_option('min_cpu', max_cpu) + if cpu_available < max_cpu: + return error('resource not enough: cpu (Avail: %s, Need: %s)' % (cpu_available, max_cpu)) + if max_cpu < min_cpu: + return error('min_cpu must less then max_cpu') + + # memory options + memory_size = get_parsed_option('memory_size', None) + log_disk_size = get_parsed_option('log_disk_size', None) + + if memory_size is None: + memory_size = mem_available + if log_disk_size is None: + log_disk_size = log_disk_available + + if mem_available < memory_size: + return error('resource not enough: memory (Avail: %s, Need: %s)' % (format_size(mem_available), format_size(memory_size))) + + # log disk size options + if log_disk_size is not None and log_disk_available < log_disk_size: + return error('resource not enough: log disk space (Avail: %s, Need: %s)' % (format_size(disk_available), format_size(log_disk_size))) + + # iops options + max_iops = get_option('max_iops', None) + min_iops = get_option('min_iops', None) + iops_weight = get_option('iops_weight', None) + if max_iops is not None and max_iops < MIN_IOPS: + return error('max_iops must greater than %d' % MIN_IOPS) + if max_iops is not None and min_iops is not None and max_iops < min_iops: + return error('min_iops must less then max_iops') + + zone_num = len(zones) + charset = get_option('charset', '') + collate = get_option('collate', '') + replica_num = get_option('replica_num', zone_num) + logonly_replica_num = get_option('logonly_replica_num', 0) + tablegroup = get_option('tablegroup', '') + primary_zone = get_option('primary_zone', 'RANDOM') + locality = get_option('locality', '') + variables = get_option('variables', '') + + if replica_num == 0: + replica_num = zone_num + elif replica_num > zone_num: + return error('replica_num cannot be greater than zone num (%s)' % zone_num) + if not primary_zone: + primary_zone = 'RANDOM' + if logonly_replica_num > replica_num: + return error('logonly_replica_num cannot be greater than replica_num (%s)' % replica_num) + + # create resource unit + sql = "create resource unit %s max_cpu %.1f, memory_size %d" % (unit_name, max_cpu, memory_size) + if min_cpu is not None: + sql += ', min_cpu %.1f' % min_cpu + if max_iops is not None: + sql += ', max_iops %d' % max_iops + if min_iops is not None: + sql += ', min_iops %d' % min_iops + if iops_weight is not None: + sql += ', iops_weight %d' % iops_weight + if log_disk_size is not None: + sql += ', log_disk_size %d' % log_disk_size + + res = cursor.execute(sql) + if res is False: + error() + return - # create resource pool - sql = "create resource pool %s unit='%s', unit_num=%d, zone_list=%s" % (pool_name, unit_name, unit_num, zone_list) - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('failed to create pool, execute sql exception: %s' % sql) - return + # create resource pool + sql = "create resource pool %s unit='%s', unit_num=%d, zone_list=%s" % (pool_name, unit_name, unit_num, zone_list) + res = cursor.execute(sql) + if res is False: + error() + return - # create tenant - sql = "create tenant %s replica_num=%d,zone_list=%s,primary_zone='%s',resource_pool_list=('%s')" - sql = sql % (name, replica_num, zone_list, primary_zone, pool_name) - if charset: - sql += ", charset = '%s'" % charset - if collate: - sql += ", collate = '%s'" % collate - if logonly_replica_num: - sql += ", logonly_replica_num = %d" % logonly_replica_num - if tablegroup: - sql += ", default tablegroup ='%s'" % tablegroup - if locality: - sql += ", locality = '%s'" % locality - - set_mode = "ob_compatibility_mode = '%s'" % mode - if variables: - sql += "set %s, %s" % (variables, set_mode) - else: - sql += "set %s" % set_mode - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('faild to crate tenant, execute sql exception: %s' % sql) - return - + # create tenant + sql = "create tenant %s replica_num=%d,zone_list=%s,primary_zone='%s',resource_pool_list=('%s')" + sql = sql % (name, replica_num, zone_list, primary_zone, pool_name) + if charset: + sql += ", charset = '%s'" % charset + if collate: + sql += ", collate = '%s'" % collate + if logonly_replica_num: + sql += ", logonly_replica_num = %d" % logonly_replica_num + if tablegroup: + sql += ", default tablegroup ='%s'" % tablegroup + if locality: + sql += ", locality = '%s'" % locality + + set_mode = "ob_compatibility_mode = '%s'" % mode + if variables: + sql += "set %s, %s" % (variables, set_mode) + else: + sql += "set %s" % set_mode + res = cursor.execute(sql) + if res is False: + error() + return stdio.stop_loading('succeed') + database = get_option('database') + if database: + sql = 'create database {}'.format(database) + if not exec_sql_in_tenant(sql=sql, cursor=cursor, tenant=name, mode=mode) and not create_if_not_exists: + stdio.error('failed to create database {}'.format(database)) + return + + db_username = get_option('db_username') + db_password = get_option('db_password', '') + if db_username: + if mode == "mysql": + sql = """create user if not exists '{username}' IDENTIFIED BY '{password}'; + grant all on *.* to '{username}' WITH GRANT OPTION;""".format( + username=db_username, password=db_password) + else: + # todo: fix oracle user create + sql = """create {username} IDENTIFIED BY {password}; +grant all on *.* to {username} WITH GRANT OPTION; +grant dba to {username}; +grant all privileges to {username};""" + if not exec_sql_in_tenant(sql=sql, cursor=cursor, tenant=name, mode=mode): + stdio.error('failed to create user {}'.format(db_username)) + return return plugin_context.return_true() \ No newline at end of file diff --git a/plugins/oceanbase/4.0.0.0/drop_tenant.py b/plugins/oceanbase/4.0.0.0/drop_tenant.py index 2ae16b1..2d4261e 100644 --- a/plugins/oceanbase/4.0.0.0/drop_tenant.py +++ b/plugins/oceanbase/4.0.0.0/drop_tenant.py @@ -25,9 +25,6 @@ def drop_tenant(plugin_context, cursor, *args, **kwargs): def error(*arg, **kwargs): stdio.error(*arg, **kwargs) stdio.stop_loading('fail') - def exception(*arg, **kwargs): - stdio.exception(*arg, **kwargs) - stdio.stop_loading('fail') cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio @@ -45,47 +42,40 @@ def exception(*arg, **kwargs): tenant = None sql = "select * from oceanbase.DBA_OB_TENANTS where tenant_name = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant = cursor.fetchone() - if not tenant: - error('No such Tenant %s' % tenant_name) - return - except: - exception('execute sql exception: %s' % (sql % tenant_name)) + tenant = cursor.fetchone(sql, [tenant_name]) + if tenant is False: + return + if not tenant: + error('No such Tenant %s' % tenant_name) return pool = None sql = "select * from oceanbase.DBA_OB_RESOURCE_POOLS where tenant_id = %d" % tenant['TENANT_ID'] - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - pool = cursor.fetchone() - sql = "drop tenant %s FORCE" % tenant_name - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - if not pool: - return - sql = "drop resource pool %s" % pool['NAME'] - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) + pool = cursor.fetchone(sql) + if pool is False: + return + sql = "drop tenant %s FORCE" % tenant_name + res = cursor.execute(sql) + if res is False: + error() + return + if not pool: + error() + return + sql = "drop resource pool %s" % pool['NAME'] + res = cursor.execute(sql) + if res is False: + error() return sql = "select * from oceanbase.DBA_OB_UNIT_CONFIGS where unit_config_id = %d" % pool['UNIT_CONFIG_ID'] - try: - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - unit = cursor.fetchone() - if not unit: - return - sql = "drop resource unit %s" % unit['NAME'] - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - except: - exception('execute sql exception: %s' % sql) + unit = cursor.fetchone(sql) + if not unit: + return + sql = "drop resource unit %s" % unit['NAME'] + res = cursor.execute(sql) + if res is False: + error() return stdio.stop_loading('succeed') diff --git a/plugins/oceanbase/4.0.0.0/generate_config.py b/plugins/oceanbase/4.0.0.0/generate_config.py index 03271cc..8b4d13c 100644 --- a/plugins/oceanbase/4.0.0.0/generate_config.py +++ b/plugins/oceanbase/4.0.0.0/generate_config.py @@ -22,8 +22,10 @@ import re, os +from math import sqrt -from _errno import EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE, EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED +from _errno import EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE, EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED, EC_OBSERVER_GET_MEMINFO_FAIL +import _errno as err def parse_size(size): @@ -55,79 +57,90 @@ def format_size(size, precision=1): return format % (size, units[idx]) -def get_system_memory(memory_limit): - if memory_limit < (8 << 30): +def get_system_memory(memory_limit, min_pool_memory, generate_config_mini): + if generate_config_mini and memory_limit <= 6 << 30: system_memory = 1 << 30 - elif memory_limit <= (64 << 30): - system_memory = memory_limit * 0.5 - elif memory_limit <= (150 << 30): - system_memory = memory_limit * 0.4 + elif memory_limit <= 8 << 30: + system_memory = 2 << 30 + elif memory_limit <= 16 << 30: + system_memory = 3 << 30 + elif memory_limit <= 32 << 30: + system_memory = 5 << 30 + elif memory_limit <= 48 << 30: + system_memory = 7 << 30 + elif memory_limit <= 64 << 30: + system_memory = 10 << 30 else: - system_memory = memory_limit * 0.3 - return max(1 << 30, system_memory) + memory_limit_gb = memory_limit >> 30 + system_memory = int(3 * (sqrt(memory_limit_gb) - 3)) << 30 + return max(system_memory, min_pool_memory) + + +def generate_config(plugin_context, generate_config_mini=False, generate_check=True, return_generate_keys=False, generate_consistent_config=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=[ + 'memory_limit', 'datafile_size', 'log_disk_size', 'devname', 'system_memory', 'cpu_count', 'production_mode', + 'syslog_level', 'enable_syslog_recycle', 'enable_syslog_wf', 'max_syslog_file_count', 'cluster_id', 'ocp_meta_tenant_log_disk_size' + ]) + + def update_server_conf(server, key, value): + if server not in generate_configs: + generate_configs[server] = {} + generate_configs[server][key] = value + def update_global_conf(key, value): + generate_configs['global'][key] = value + def summit_config(): + generate_global_config = generate_configs['global'] + for key in generate_global_config: + cluster_config.update_global_conf(key, generate_global_config[key], False) + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + for key in generate_server_config: + cluster_config.update_server_conf(server, key, generate_server_config[key], False) - -def generate_config(plugin_context, deploy_config, *args, **kwargs): cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio success = True + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) stdio.start_loading('Generate observer configuration') global_config = cluster_config.get_global_conf() - if not global_config.get('appname'): - default_appname = 'obcluster' - for componet_name in ['obproxy', 'obproxy-ce']: - if componet_name in deploy_config.components: - obproxy_cluster_config = deploy_config.components[componet_name] - cluster_name = obproxy_cluster_config.get_global_conf().get('cluster_name') - if not cluster_name: - for server in obproxy_cluster_config.servers: - server_config = obproxy_cluster_config.get_server_conf(server) - if server_config.get('cluster_name'): - default_appname = server_config['cluster_name'] - break - break - cluster_config.update_global_conf('appname', default_appname, False) - max_syslog_file_count_default = 4 - if global_config.get('syslog_level') is None: - cluster_config.update_global_conf('syslog_level', 'INFO', False) if global_config.get('enable_syslog_recycle') is None: - cluster_config.update_global_conf('enable_syslog_recycle', True, False) + update_global_conf('enable_syslog_recycle', True) if global_config.get('enable_syslog_wf') is None: - cluster_config.update_global_conf('enable_syslog_wf', True, False) + update_global_conf('enable_syslog_wf', False) if global_config.get('max_syslog_file_count') is None: - cluster_config.update_global_conf('max_syslog_file_count', max_syslog_file_count_default, False) + update_global_conf('max_syslog_file_count', max_syslog_file_count_default) if global_config.get('cluster_id') is None: - cluster_config.update_global_conf('cluster_id', 1, False) + update_global_conf('cluster_id', 1) MIN_MEMORY = 6 << 30 PRO_MEMORY_MIN = 16 << 30 SLOG_SIZE = 10 << 30 MIN_CPU_COUNT = 16 START_NEED_MEMORY = 3 << 30 - if getattr(plugin_context.options, 'mini', False): - if not global_config.get('memory_limit_percentage') and not global_config.get('memory_limit'): - cluster_config.update_global_conf('memory_limit', format_size(MIN_MEMORY, 0), False) - if not global_config.get('datafile_size') and not global_config.get('datafile_disk_percentage'): - cluster_config.update_global_conf('datafile_size', '20G', False) - if not global_config.get('log_disk_size') and not global_config.get('log_disk_percentage'): - cluster_config.update_global_conf('log_disk_size', '24G', False) + MINI_MEMORY_SIZE = MIN_MEMORY + MINI_DATA_FILE_SIZE = 20 << 30 + MINI_LOG_DISK_SIZE = 15 << 30 + + has_ocp = 'ocp-express' in [repo.name for repo in plugin_context.repositories] + ip_server_memory_info = {} + servers_info = {} for server in cluster_config.servers: ip = server.ip client = clients[server] server_config = cluster_config.get_server_conf_with_default(server) - user_server_config = cluster_config.get_original_server_conf(server) - if not server_config.get('home_path'): - stdio.error("observer %s: missing configuration 'home_path' in configuration file" % server) - success = False - continue + user_server_config = cluster_config.get_original_server_conf_with_global(server) if user_server_config.get('devname') is None: if client.is_localhost(): - cluster_config.update_server_conf(server, 'devname', 'lo') + update_server_conf(server, 'devname', 'lo') else: devinfo = client.execute_command('cat /proc/net/dev').stdout interfaces = re.findall('\n\s+(\w+):', devinfo) @@ -135,7 +148,7 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): if interface == 'lo': continue if client.execute_command('ping -W 1 -c 1 -I %s %s' % (interface, ip)): - cluster_config.update_server_conf(server, 'devname', interface) + update_server_conf(server, 'devname', interface) break dirs = {"home_path": server_config['home_path']} @@ -146,67 +159,80 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): # memory auto_set_memory = False auto_set_system_memory = False + auto_set_min_pool_memory = False system_memory = 0 if user_server_config.get('system_memory'): system_memory = parse_size(user_server_config.get('system_memory')) + if generate_config_mini and '__min_full_resource_pool_memory' not in user_server_config: + auto_set_min_pool_memory = True + min_pool_memory = server_config['__min_full_resource_pool_memory'] min_memory = max(system_memory, MIN_MEMORY) - if user_server_config.get('memory_limit_percentage'): + if ip not in ip_server_memory_info: ret = client.execute_command('cat /proc/meminfo') if ret: - total_memory = 0 + ip_server_memory_info[ip] = server_memory_stats = {} + memory_key_map = { + 'MemTotal': 'total', + 'MemFree': 'free', + 'MemAvailable': 'available', + 'Buffers': 'buffers', + 'Cached': 'cached' + } + for key in memory_key_map: + server_memory_stats[memory_key_map[key]] = 0 for k, v in re.findall('(\w+)\s*:\s*(\d+\s*\w+)', ret.stdout): - if k == 'MemTotal': - total_memory = parse_size(str(v)) - memory_limit = int(total_memory * user_server_config.get('memory_limit_percentage') / 100) - else: - if not server_config.get('memory_limit'): - ret = client.execute_command('cat /proc/meminfo') - if ret: - server_memory_stats = {} - memory_key_map = { - 'MemTotal': 'total', - 'MemFree': 'free', - 'MemAvailable': 'available', - 'Buffers': 'buffers', - 'Cached': 'cached' - } - for key in memory_key_map: - server_memory_stats[memory_key_map[key]] = 0 - for k, v in re.findall('(\w+)\s*:\s*(\d+\s*\w+)', ret.stdout): - if k in memory_key_map: - key = memory_key_map[k] - server_memory_stats[key] = parse_size(str(v)) - - if server_memory_stats['available'] < START_NEED_MEMORY: - stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(START_NEED_MEMORY))) - success = False - continue - - if server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached'] < MIN_MEMORY: - stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(MIN_MEMORY))) - success = False - continue + if k in memory_key_map: + key = memory_key_map[k] + server_memory_stats[key] = parse_size(str(v)) - memory_limit = max(MIN_MEMORY, server_memory_stats['available'] * 0.9) - server_config['memory_limit'] = format_size(memory_limit, 0) - cluster_config.update_server_conf(server, 'memory_limit', server_config['memory_limit'], False) + if user_server_config.get('memory_limit_percentage'): + if ip in ip_server_memory_info: + total_memory = parse_size(ip_server_memory_info[ip]['total']) + memory_limit = int(total_memory * user_server_config.get('memory_limit_percentage') / 100) + elif generate_check: + stdio.error(EC_OBSERVER_GET_MEMINFO_FAIL.format(server=server)) + success = False + continue + else: + memory_limit = MIN_MEMORY + elif not server_config.get('memory_limit'): + if generate_config_mini: + memory_limit = MINI_MEMORY_SIZE + update_server_conf(server, 'memory_limit', format_size(memory_limit, 0)) + update_server_conf(server, 'production_mode', False) + if auto_set_min_pool_memory: + min_pool_memory = 1073741824 + update_server_conf(server, '__min_full_resource_pool_memory', min_pool_memory) + else: + if ip in ip_server_memory_info: + server_memory_stats = ip_server_memory_info[ip] + if generate_check: + if server_memory_stats['available'] < START_NEED_MEMORY: + stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(START_NEED_MEMORY))) + success = False + continue + + if server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached'] < MIN_MEMORY: + stdio.error(EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(MIN_MEMORY))) + success = False + continue + memory_limit = max(MIN_MEMORY, int(server_memory_stats['available'] * 0.9)) + update_server_conf(server, 'memory_limit', format_size(memory_limit, 0)) auto_set_memory = True - else: - stdio.error("%s: fail to get memory info.\nPlease configure 'memory_limit' manually in configuration file") + elif generate_check: + stdio.error(EC_OBSERVER_GET_MEMINFO_FAIL.format(server=server)) success = False continue - else: - try: - memory_limit = parse_size(server_config.get('memory_limit')) - except: - stdio.error('memory_limit must be an integer') - return + else: + memory_limit = MIN_MEMORY + else: + memory_limit = parse_size(server_config.get('memory_limit')) if system_memory == 0: auto_set_system_memory = True - system_memory = get_system_memory(memory_limit) - cluster_config.update_server_conf(server, 'system_memory', format_size(system_memory, 0), False) - + system_memory = get_system_memory(memory_limit, min_pool_memory, generate_config_mini) + update_server_conf(server, 'system_memory', format_size(system_memory, 0)) + # cpu if not server_config.get('cpu_count'): ret = client.execute_command("grep -e 'processor\s*:' /proc/cpuinfo | wc -l") @@ -215,14 +241,15 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): server_config['cpu_count'] = max(MIN_CPU_COUNT, int(cpu_num - 2)) else: server_config['cpu_count'] = MIN_CPU_COUNT - cluster_config.update_server_conf(server, 'cpu_count', server_config['cpu_count'], False) + update_server_conf(server, 'cpu_count', server_config['cpu_count']) elif server_config['cpu_count'] < MIN_CPU_COUNT: - cluster_config.update_server_conf(server, 'cpu_count', MIN_CPU_COUNT, False) + update_server_conf(server, 'cpu_count', MIN_CPU_COUNT) stdio.warn('(%s): automatically adjust the cpu_count %s' % (server, MIN_CPU_COUNT)) - + # disk - if (not server_config.get('datafile_size') and not user_server_config.get('datafile_disk_percentage')) or \ - (not server_config.get('log_disk_size') and not user_server_config.get('log_disk_percentage')): + datafile_size = parse_size(server_config.get('datafile_size', 0)) + log_disk_size = parse_size(server_config.get('log_disk_size', 0)) + if not server_config.get('datafile_size') or not server_config.get('log_disk_size'): disk = {'/': 0} ret = client.execute_command('df --block-size=1024') if ret: @@ -257,22 +284,23 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): home_path_mount = mounts[dirs['home_path']] home_path_disk = disk[home_path_mount] - + data_dir_mount = mounts[dirs['data_dir']] data_dir_disk = disk[data_dir_mount] - + clog_dir_mount = mounts[dirs['clog_dir']] clog_dir_disk = disk[clog_dir_mount] auto_set_datafile_size = False auto_set_log_disk_size = False - datafile_size = parse_size(server_config.get('datafile_size', 0)) - log_disk_size = parse_size(server_config.get('log_disk_size', 0)) if not datafile_size: datafile_disk_percentage = int(user_server_config.get('datafile_disk_percentage', 0)) if datafile_disk_percentage: datafile_size = data_dir_mount['total'] * datafile_disk_percentage / 100 + elif generate_config_mini: + datafile_size = MINI_DATA_FILE_SIZE + update_server_conf(server, 'datafile_size', format_size(datafile_size, 0)) else: auto_set_datafile_size = True @@ -280,13 +308,16 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): log_disk_percentage = int(user_server_config.get('log_disk_percentage', 0)) if log_disk_percentage: log_disk_size = clog_dir_disk['total'] * log_disk_percentage / 100 + elif generate_config_mini: + log_disk_size = MINI_LOG_DISK_SIZE + update_server_conf(server, 'log_disk_size', format_size(log_disk_size, 0)) else: auto_set_log_disk_size = True if user_server_config.get('enable_syslog_recycle') is False: log_size = 1 << 30 # 默认先给1G普通日志空间 else: - log_size = (256 << 20) * user_server_config.get('max_syslog_file_count', max_syslog_file_count_default) * 4 + log_size = (256 << 20) * int(user_server_config.get('max_syslog_file_count', max_syslog_file_count_default)) * 4 if clog_dir_mount == data_dir_mount: min_log_size = log_size if clog_dir_mount == home_path_mount else 0 @@ -304,18 +335,30 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): min_log_disk_size = log_disk_size MIN_NEED += log_disk_size min_need = min_log_size + min_datafile_size + min_log_disk_size - + disk_free = data_dir_disk['avail'] if MIN_NEED > disk_free: - stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s).' % (ip, data_dir_mount, format_size(disk_free), format_size(MIN_NEED))) - success = False - continue - if min_need > disk_free: - if not auto_set_memory: - stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s).' % (ip, data_dir_mount, format_size(disk_free), format_size(min_need))) + if generate_check: + stdio.error(err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=data_dir_mount, avail=format_size(disk_free), need=format_size(MIN_NEED))) success = False continue - + else: + if auto_set_datafile_size: + datafile_size = MIN_MEMORY * 3 + if auto_set_log_disk_size: + log_disk_size = MIN_MEMORY * 3 + if auto_set_memory: + memory_limit = MIN_MEMORY + update_server_conf(server, 'memory_limit', format_size(memory_limit, 0)) + if auto_set_system_memory: + system_memory = get_system_memory(memory_limit, min_pool_memory, generate_config_mini) + update_server_conf(server, 'system_memory', format_size(system_memory, 0)) + elif min_need > disk_free: + if generate_check and not auto_set_memory: + stdio.error(err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=data_dir_mount, avail=format_size(disk_free), need=format_size(min_need))) + success = False + continue + disk_free = disk_free - log_size - SLOG_SIZE memory_factor = 6 if auto_set_datafile_size is False: @@ -325,21 +368,21 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): disk_free -= min_log_disk_size memory_factor -= 3 memory_limit = format_size(disk_free / max(1, memory_factor), 0) - cluster_config.update_server_conf(server, 'memory_limit', memory_limit, False) + update_server_conf(server, 'memory_limit', memory_limit) memory_limit = parse_size(memory_limit) if auto_set_system_memory: - system_memory = get_system_memory(memory_limit) - cluster_config.update_server_conf(server, 'system_memory', format_size(system_memory, 0), False) + system_memory = get_system_memory(memory_limit, min_pool_memory, generate_config_mini) + update_server_conf(server, 'system_memory', format_size(system_memory, 0)) log_disk_size = memory_limit * 3 - datafile_size = disk_free - log_disk_size + datafile_size = max(disk_free - log_disk_size, log_disk_size) else: log_disk_size = memory_limit * 3 - datafile_size = disk_free - log_size - SLOG_SIZE - log_disk_size + datafile_size = max(disk_free - log_size - SLOG_SIZE - log_disk_size, log_disk_size) if auto_set_datafile_size: - cluster_config.update_server_conf(server, 'datafile_size', format_size(datafile_size, 0), False) + update_server_conf(server, 'datafile_size', format_size(datafile_size, 0)) if auto_set_log_disk_size: - cluster_config.update_server_conf(server, 'log_disk_size', format_size(log_disk_size, 0), False) + update_server_conf(server, 'log_disk_size', format_size(log_disk_size, 0)) else: datafile_min_memory_limit = memory_limit if auto_set_datafile_size: @@ -347,18 +390,18 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): min_log_size = log_size if data_dir_mount == home_path_mount else 0 disk_free = data_dir_disk['avail'] min_need = min_log_size + datafile_size + SLOG_SIZE - if min_need > disk_free: + if generate_check and min_need > disk_free: if not auto_set_memory: - stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s).' % (ip, data_dir_mount, format_size(disk_free), format_size(min_need))) + stdio.error(err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=data_dir_mount, avail=format_size(disk_free), need=format_size(min_need))) success = False continue datafile_min_memory_limit = (disk_free - log_size - SLOG_SIZE) / 3 if datafile_min_memory_limit < min_memory: - stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s).' % (ip, data_dir_mount, format_size(disk_free), format_size(min_need))) + stdio.error(err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=data_dir_mount, avail=format_size(disk_free), need=format_size(min_need))) success = False continue datafile_min_memory_limit = parse_size(format_size(datafile_min_memory_limit, 0)) - disk_log_size = datafile_min_memory_limit * 3 + datafile_size = datafile_min_memory_limit * 3 log_disk_min_memory_limit = memory_limit if auto_set_log_disk_size: @@ -366,35 +409,146 @@ def generate_config(plugin_context, deploy_config, *args, **kwargs): min_log_size = log_size if clog_dir_mount == home_path_mount else 0 disk_free = clog_dir_disk['avail'] min_need = min_log_size + log_disk_size - if min_need > disk_free: + if generate_check and min_need > disk_free: if not auto_set_memory: - stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s).' % (ip, clog_dir_mount, format_size(disk_free), format_size(min_need))) + stdio.error(err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=data_dir_mount, avail=format_size(disk_free), need=format_size(min_need))) success = False continue log_disk_min_memory_limit = (disk_free - log_size) / 3 if log_disk_min_memory_limit < min_memory: - stdio.error('(%s) %s not enough disk space. (Avail: %s, Need: %s).' % (ip, clog_dir_mount, format_size(disk_free), format_size(min_need))) + stdio.error(err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=data_dir_mount, avail=format_size(disk_free), need=format_size(min_need))) success = False continue log_disk_min_memory_limit = parse_size(format_size(log_disk_min_memory_limit, 0)) log_disk_size = log_disk_min_memory_limit * 3 - + if auto_set_memory: - cluster_config.update_server_conf(server, 'memory_limit', format_size(memory_limit, 0), False) + update_server_conf(server, 'memory_limit', format_size(memory_limit, 0)) if auto_set_system_memory: - system_memory = get_system_memory(memory_limit) - cluster_config.update_server_conf(server, 'system_memory', format_size(system_memory, 0), False) - + system_memory = get_system_memory(memory_limit, min_pool_memory, generate_config_mini) + update_server_conf(server, 'system_memory', system_memory) + if auto_set_datafile_size: - cluster_config.update_server_conf(server, 'datafile_size', format_size(datafile_size, 0), False) + update_server_conf(server, 'datafile_size', format_size(datafile_size, 0)) if auto_set_log_disk_size: - cluster_config.update_server_conf(server, 'log_disk_size', format_size(log_disk_size, 0), False) + update_server_conf(server, 'log_disk_size', format_size(log_disk_size, 0)) if memory_limit < PRO_MEMORY_MIN: - cluster_config.update_server_conf(server, 'production_mode', False, False) - + update_server_conf(server, 'production_mode', False) + servers_info[server] = { + "memory_limit": memory_limit, + "system_memory": system_memory, + "min_pool_memory": min_pool_memory, + "log_disk_size": log_disk_size + } + + # ocp meta db + SYS_TENANT_LOG_DISK_SCALE = 1 + if has_ocp: + if 'ocp_meta_tenant_log_disk_size' not in global_config and 'log_disk_size' not in global_config.get('ocp_meta_tenant', {}): + if generate_config_mini: + update_global_conf('ocp_meta_tenant_log_disk_size', '6656M') + else: + meta_min_log_disk_size = 6 << 30 + expect_log_disk_size = (9 * 512 + 512 * len(cluster_config.servers) + 512 * 3) << 20 + max_available = 0 + sys_memory_size = None + sys_log_disk_size = None + if 'sys_tenant' in global_config: + if 'memory_size' in global_config['sys_tenant']: + sys_memory_size = global_config['sys_tenant']['memory_size'] + if 'log_disk_size' in global_config['sys_tenant']: + sys_log_disk_size = global_config['sys_tenant']['log_disk_size'] + for server in cluster_config.servers: + # server_config = cluster_config.get_server_conf_with_default(server) + server_info = servers_info.get(server) + if not server_info: + continue + memory_limit = server_info['memory_limit'] + system_memory = server_info['system_memory'] + log_disk_size = server_info['log_disk_size'] + min_pool_memory = server_info['min_pool_memory'] + if not sys_log_disk_size: + if not sys_memory_size: + sys_memory_size = max(min_pool_memory, min(int((memory_limit - system_memory) * 0.25), 16 << 30)) + sys_log_disk_size = sys_memory_size * SYS_TENANT_LOG_DISK_SCALE + max_available = max(max_available, log_disk_size - sys_log_disk_size) + if expect_log_disk_size > max_available: + expect_log_disk_size = meta_min_log_disk_size + if expect_log_disk_size > max_available and generate_check: + stdio.error(err.EC_OCP_EXPRESS_META_DB_NOT_ENOUGH_LOG_DISK_AVAILABLE.format(avail=max_available, need=expect_log_disk_size)) + success = False + cluster_config.update_global_conf('ocp_meta_tenant_log_disk_size', format_size(expect_log_disk_size, 0)) + if generate_config_mini and 'ocp_meta_tenant_memory_size' not in global_config and 'memory_size' not in global_config.get('ocp_meta_tenant', {}): + update_global_conf('ocp_meta_tenant_memory_size', '1536M') + + if generate_consistent_config: + generate_global_config = generate_configs['global'] + server_num = len(cluster_config.servers) + keys = ['memory_limit', 'datafile_size', 'system_memory', 'log_disk_size', 'cpu_count', 'production_mode'] + for key in keys: + servers = [] + values = [] + is_capacity_key = (key != 'cpu_count' and key != 'production_mode') + for server in cluster_config.servers: + if key in generate_configs.get(server, {}): + value = generate_configs[server][key] + servers.append(server) + values.append(parse_size(value) if is_capacity_key else value) + if values: + if len(values) != server_num and key in generate_global_config: + continue + value = min(values) + generate_global_config[key] = format_size(value, 0) if is_capacity_key else value + for server in servers: + del generate_configs[server][key] + + # merge_generate_config + merge_config = {} + generate_global_config = generate_configs['global'] + count_base = len(cluster_config.servers) - 1 + if count_base < 1: + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_global_config.update(generate_configs[server]) + generate_configs[server] = {} + else: + for server in cluster_config.servers: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + merged_server_config = {} + for key in generate_server_config: + if key in generate_global_config: + if generate_global_config[key] != generate_server_config[key]: + merged_server_config[key] = generate_server_config[key] + elif key in merge_config: + if merge_config[key]['value'] != generate_server_config[key]: + merged_server_config[key] = generate_server_config[key] + elif count_base == merge_config[key]['count']: + generate_global_config[key] = generate_server_config[key] + del merge_config[key] + else: + merge_config[key]['severs'].append(server) + merge_config[key]['count'] += 1 + else: + merge_config[key] = {'value': generate_server_config[key], 'severs': [server], 'count': 1} + generate_configs[server] = merged_server_config + + for key in merge_config: + config_st = merge_config[key] + for server in config_st['severs']: + if server not in generate_configs: + continue + generate_server_config = generate_configs[server] + generate_server_config[key] = config_st['value'] + + # summit_config + summit_config() + if success: stdio.stop_loading('succeed') return plugin_context.return_true() - stdio.stop_loading('fail') \ No newline at end of file + stdio.stop_loading('fail') diff --git a/plugins/oceanbase/4.0.0.0/init.py b/plugins/oceanbase/4.0.0.0/init.py index e396fcc..668b601 100644 --- a/plugins/oceanbase/4.0.0.0/init.py +++ b/plugins/oceanbase/4.0.0.0/init.py @@ -60,7 +60,7 @@ def init_dir(server, client, key, path, link_path=None): return False -def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def init(plugin_context, *args, **kwargs): global stdio, force cluster_config = plugin_context.cluster_config clients = plugin_context.clients @@ -78,8 +78,6 @@ def init(plugin_context, local_home_path, repository_dir, *args, **kwargs): server_config = cluster_config.get_server_conf(server) client = clients[server] home_path = server_config['home_path'] - remote_home_path = client.execute_command('echo ${OBD_HOME:-"$HOME"}/.obd').stdout.strip() - remote_repository_dir = repository_dir.replace(local_home_path, remote_home_path) if not server_config.get('data_dir'): server_config['data_dir'] = '%s/store' % home_path diff --git a/plugins/oceanbase/4.0.0.0/list_tenant.py b/plugins/oceanbase/4.0.0.0/list_tenant.py new file mode 100644 index 0000000..a200012 --- /dev/null +++ b/plugins/oceanbase/4.0.0.0/list_tenant.py @@ -0,0 +1,89 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import re + + +def parse_size(size): + _bytes = 0 + if isinstance(size, str): + size = size.strip() + if not isinstance(size, str) or size.isdigit(): + _bytes = int(size) + else: + units = {"B": 1, "K": 1 << 10, "M": 1 << 20, "G": 1 << 30, "T": 1 << 40} + match = re.match(r'^([1-9][0-9]*)\s*([B,K,M,G,T])$', size.upper()) + _bytes = int(match.group(1)) * units[match.group(2)] + return _bytes + + +def format_size(size, precision=1): + units = ['B', 'K', 'M', 'G', 'T', 'P'] + idx = 0 + if precision: + div = 1024.0 + format = '%.' + str(precision) + 'f%s' + else: + div = 1024 + format = '%d%s' + while idx < 5 and size >= 1024: + size /= 1024.0 + idx += 1 + return format % (size, units[idx]) + + +def list_tenant(plugin_context, cursor, *args, **kwargs): + + cluster_config = plugin_context.cluster_config + stdio = plugin_context.stdio + + stdio.start_loading('Select tenant') + tenant_infos = [] + sql = "select * from oceanbase.DBA_OB_TENANTS;" + tenants = cursor.fetchall(sql) + if tenants is False: + stdio.stop_loading('fail') + return + + for tenant in tenants: + sql = "select * from oceanbase.__all_unit_config where name = '%s'" + if tenant['TENANT_TYPE'] == 'META': + continue + unit_name = '%s_unit' % tenant['TENANT_NAME'] if tenant['TENANT_NAME'] != 'sys' else 'sys_unit_config' + res = cursor.fetchone(sql % unit_name) + if res is False: + stdio.stop_loading('fail') + return + tenant_infos.append(dict(tenant, **res)) + if tenant_infos: + stdio.print_list(tenant_infos, ['tenant_name', 'tenant_type', 'compatibility_mode', 'primary_zone', 'max_cpu', + 'min_cpu', 'memory_size', 'max_iops', 'min_iops', 'log_disk_size', + 'iops_weight'], + lambda x: [x['TENANT_NAME'], x['TENANT_TYPE'], x['COMPATIBILITY_MODE'], x['PRIMARY_ZONE'], + x['max_cpu'], x['min_cpu'], format_size(x['memory_size']), x['max_iops'], x['min_iops'], + format_size(x['log_disk_size']), x['iops_weight']], + title='tenant') + stdio.stop_loading('succeed') + return plugin_context.return_true() + + stdio.stop_loading('fail') + plugin_context.return_false() \ No newline at end of file diff --git a/plugins/oceanbase/4.0.0.0/major_freeze.py b/plugins/oceanbase/4.0.0.0/major_freeze.py index 55844f5..6e6b94c 100644 --- a/plugins/oceanbase/4.0.0.0/major_freeze.py +++ b/plugins/oceanbase/4.0.0.0/major_freeze.py @@ -25,33 +25,33 @@ def major_freeze(plugin_context, cursor, *args, **kwargs): - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - stdio = plugin_context.stdio tenant_name = kwargs.get('tenant') - tenant_id = execute(cursor, "select TENANT_ID from oceanbase.DBA_OB_TENANTS where tenant_name = '%s'" % tenant_name)["TENANT_ID"] + tenant_id = cursor.fetchone("select TENANT_ID from oceanbase.DBA_OB_TENANTS where tenant_name = '%s'" % tenant_name) + if tenant_id is False: + return + tenant_id = tenant_id["TENANT_ID"] # Major freeze stdio.start_loading('Merge') sql_frozen_scn = "select FROZEN_SCN, LAST_SCN from oceanbase.CDB_OB_MAJOR_COMPACTION where tenant_id = '%s'" % tenant_id - merge_version = execute(cursor, sql_frozen_scn)['FROZEN_SCN'] - execute(cursor, "alter system major freeze tenant = %s" % tenant_name) + merge_version = cursor.fetchone(sql_frozen_scn) + if merge_version is False: + return + merge_version = merge_version['FROZEN_SCN'] + if cursor.execute("alter system major freeze tenant = %s" % tenant_name) is False: + return while True: - current_version = execute(cursor, sql_frozen_scn).get("FROZEN_SCN") + current_version = cursor.fetchone(sql_frozen_scn) + if current_version is False: + return + current_version = current_version.get("FROZEN_SCN") if int(current_version) > int(merge_version): break time.sleep(5) while True: - ret = execute(cursor, sql_frozen_scn) + ret = cursor.fetchone(sql_frozen_scn) + if ret is False: + return if int(ret.get("FROZEN_SCN", 0)) / 1000 == int(ret.get("LAST_SCN", 0)) / 1000: break time.sleep(5) diff --git a/plugins/oceanbase/4.0.0.0/parameter.yaml b/plugins/oceanbase/4.0.0.0/parameter.yaml index b456431..44eef6c 100644 --- a/plugins/oceanbase/4.0.0.0/parameter.yaml +++ b/plugins/oceanbase/4.0.0.0/parameter.yaml @@ -1,5 +1,7 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING min_value: NULL max_value: NULL @@ -7,16 +9,20 @@ description_en: the directory for the work data file description_local: OceanBase工作目录 - name: cluster_id + name_local: 集群ID require: true + essential: true type: INT default: 1 min_value: 1 max_value: 4294901759 modify_limit: modify - need_restart: true + need_redeploy: true description_en: ID of the cluster description_local: 本OceanBase集群ID - name: data_dir + name_local: 数据目录 + essential: true type: STRING min_value: NULL max_value: NULL @@ -24,6 +30,8 @@ description_en: the directory for the data file description_local: 存储sstable等数据的目录 - name: redo_dir + name_local: 日志目录 + essential: true type: STRING min_value: NULL max_value: NULL @@ -52,6 +60,8 @@ description_en: the directory for the ilog file description_local: 存储ilog数据的目录 - name: devname + name_local: 网卡名 + essential: true type: STRING min_value: NULL max_value: NULL @@ -59,7 +69,9 @@ description_en: name of network adapter description_local: 服务进程绑定的网卡设备名 - name: rpc_port + name_local: 内部通信端口 require: true + essential: true type: INT default: 2882 min_value: 1025 @@ -69,7 +81,9 @@ description_en: the port number for RPC protocol. description_local: 集群内部通信的端口号 - name: mysql_port + name_local: 服务端口 require: true + essential: true type: INT default: 2881 min_value: 1025 @@ -239,7 +253,9 @@ description_en: enable server supports SSL connection, takes effect only after server restart with all ca/cert/key file. description_local: 是否开启SSL连接功能 - name: datafile_size + name_local: 数据文件大小 require: false + essential: true type: CAPACITY default: 0 min_value: 0M @@ -247,8 +263,26 @@ modify_limit: decrease section: SSTABLE need_restart: false - description_en: size of the data file. - description_local: 数据文件大小。一般不要设置。 + description_en: size of the data file. Please enter an capacity, such as 20G + description_local: 数据文件大小。请输入带容量带单位的整数,如20G +- name: log_disk_percentage + require: false + type: INT + default: 0 + min_value: 0 + max_value: 99 + description_en: the percentage of disk space used by the clog files. + description_local: Redo 日志占用其所在磁盘总空间的百分比。 +- name: log_disk_size + name_local: Redo 日志大小 + require: false + essential: true + type: CAPACITY + default: 0 + min_value: 0M + max_value: NULL + description_en: the size of disk space used by the clog files. Please enter an capacity, such as 20G + description_local: Redo 日志磁盘的大小。请输入带容量带单位的整数,如24G - name: merge_stat_sampling_ratio require: false type: INT @@ -600,6 +634,7 @@ description_local: 本地存储配置文件的多个目录,为了冗余存储多份配置文件 - name: enable_syslog_recycle require: false + essential: true type: BOOL default: false min_value: NULL @@ -608,6 +643,17 @@ need_restart: false description_en: specifies whether log file recycling is turned on description_local: 是否自动回收系统日志 +- name: max_syslog_file_count + require: false + essential: true + type: INT + default: 0 + min_value: 0 + max_value: NULL + section: OBSERVER + need_restart: false + description_en: specifies the maximum number of the log files that can co-exist before the log file recycling kicks in. Each log file can occupy at most 256MB disk space. When this value is set to 0, no log file will be removed. + description_local: 系统日志自动回收复用时,最多保留多少个。值0表示不自动清理。 - name: px_task_size require: false type: CAPACITY @@ -919,15 +965,6 @@ need_restart: false description_en: URL for OBConfig service description_local: OBConfig服务的URL地址 -- name: system_memory - type: CAPACITY - default: 30G - min_value: 0M - max_value: NULL - section: OBSERVER - need_restart: false - description_en: the memory reserved for internal use which cannot be allocated to any outer-tenant, and should be determined to guarantee every server functions normally. - description_local: 系统预留内存大小,不能分配给普通租户使用 - name: cpu_quota_concurrency require: false type: DOUBLE @@ -999,7 +1036,9 @@ description_en: the time interval between the schedules of the partition load-balancing task. description_local: 负载均衡等后台任务线程空闲时的唤醒间隔时间 - name: memory_limit + name_local: 最大运行内存 require: false + essential: true type: CAPACITY default: 0 min_value: NULL @@ -1007,8 +1046,19 @@ modify_limit: decrease section: OBSERVER need_restart: false - description_en: the size of the memory reserved for internal use(for testing purpose) - description_local: 可用总内存大小。用于调试,不要设置。 + description_en: the size of the memory reserved for internal use(for testing purpose). Please enter an capacity, such as 8G + description_local: 可用总内存大小。请输入带容量带单位的整数,如8G +- name: system_memory + name_local: 集群系统内存 + essential: true + type: CAPACITY + default: 30G + min_value: 0M + max_value: NULL + section: OBSERVER + need_restart: false + description_en: the memory reserved for internal use which cannot be allocated to any outer-tenant, and should be determined to guarantee every server functions normally. Please enter an capacity, such as 2G + description_local: 系统预留内存大小,不能分配给普通租户使用。请输入带容量带单位的整数,如2G - name: __min_full_resource_pool_memory require: true type: INT @@ -1169,7 +1219,9 @@ description_en: disable write to memstore when observer memstore free memory(plus memory hold by blockcache) lower than this limit, description_local: 当全局剩余内存小于这个百分比时,暂停普通租户写入(sys租户不受影响) - name: cpu_count + name_local: 系统CPU总数 require: false + essential: true type: INT default: 0 min_value: 0 @@ -1178,16 +1230,6 @@ need_restart: true description_en: the number of CPUs in the system. If this parameter is set to zero, the number will be set according to sysconf; otherwise, this parameter is used. description_local: 系统CPU总数,如果设置为0,将自动检测 -- name: max_syslog_file_count - require: false - type: INT - default: 0 - min_value: 0 - max_value: NULL - section: OBSERVER - need_restart: false - description_en: specifies the maximum number of the log files that can co-exist before the log file recycling kicks in. Each log file can occupy at most 256MB disk space. When this value is set to 0, no log file will be removed. - description_local: 系统日志自动回收复用时,最多保留多少个。值0表示不自动清理。 - name: appname require: false type: STRING @@ -1566,22 +1608,6 @@ need_restart: false description_en: password of observer root user description_local: sys租户root用户的密码 -- name: log_disk_percentage - require: false - type: INT - default: 0 - min_value: 0 - max_value: 99 - description_en: the percentage of disk space used by the clog files. - description_local: Redo 日志占用其所在磁盘总空间的百分比。 -- name: log_disk_size - require: false - type: CAPACITY - default: 0 - min_value: 0M - max_value: NULL - description_en: the size of disk space used by the clog files. - description_local: Redo 日志磁盘的大小。 # todo: 等文档更新 - name: sql_login_thread_count require: false @@ -1814,3 +1840,68 @@ need_redeploy: false description_en: Production mode switch, default True. Adjust the memory_limit and __min_full_resource_pool_memory The lower bound of memory is adjusted to 16G and 2147483648 description_local: 生产模式开关, 默认开启。开启后调整memory limit 和 __min_full_resource_pool_memory 下界调整为 16G 和 2147483648 +- name: ocp_meta_tenant + require: false + type: DICT + default: + tenant_name: ocp + max_cpu: 1 + memory_size: 2147483648 + need_redeploy: true + description_en: The tenant specifications for ocp meta db + description_local: ocp express的元数据库使用的租户规格 +- name: ocp_meta_tenant_max_cpu + name_local: OCP express元数据库租户的CPU数 + essential: true + require: false + type: INT + default: 1 + need_redeploy: true + description_en: The tenant cpu count for ocp meta db + description_local: ocp express的元数据库使用的CPU数量 +- name: ocp_meta_tenant_memory_size + name_local: OCP express元数据库租户内存 + essential: true + require: false + type: CAPACITY + default: 2G + need_redeploy: true + description_en: The tenant memory size for ocp meta db + description_local: ocp express的元数据库使用的租户内存大小 +- name: ocp_meta_tenant_log_disk_size + name_local: OCP express元数据库租户日志磁盘大小 + essential: true + require: false + type: CAPACITY + default: 6656M + need_redeploy: true + description_en: The tenant log disk size for ocp meta db + description_local: ocp express的元数据库使用的租户日志磁盘大小 +- name: ocp_meta_db + require: false + type: STRING + default: ocp_express + need_redeploy: true + description_en: The database name for ocp meta db + description_local: ocp express的元数据库使用的数据库名 +- name: ocp_meta_username + require: false + type: STRING + default: meta + need_redeploy: true + description_en: The database name for ocp meta db + description_local: ocp express的元数据库使用的数据库名 +- name: ocp_meta_password + require: false + type: STRING + default: oceanbase + need_redeploy: true + description_en: The database name for ocp meta db + description_local: ocp express的元数据库使用的数据库名 +- name: ocp_agent_monitor_password + require: false + type: STRING + default: '' + need_redeploy: true + description_en: The password for obagent monitor user + description_local: obagent 监控用户的密码 \ No newline at end of file diff --git a/plugins/oceanbase/4.0.0.0/restart.py b/plugins/oceanbase/4.0.0.0/restart.py index fc00ca6..e449f10 100644 --- a/plugins/oceanbase/4.0.0.0/restart.py +++ b/plugins/oceanbase/4.0.0.0/restart.py @@ -28,11 +28,22 @@ class Restart(object): def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None): self.local_home_path = local_home_path - self.plugin_context = plugin_context + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + self.components = plugin_context.components self.clients = plugin_context.clients self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context self.repository = repository self.start_plugin = start_plugin self.reload_plugin = reload_plugin @@ -47,24 +58,42 @@ def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, self.cursor = None for server in self.cluster_config.servers: self.now_clients[server] = self.clients[server] + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) def close(self): if self.db: self.cursor.close() - self.db.close() self.cursor = None self.db = None def connect(self): if self.cursor is None or self.execute_sql('select version()', error=False) is False: - self.stdio.verbose('Call %s for %s' % (self.connect_plugin, self.repository)) self.sub_io.start_loading('Connect to observer') - ret = self.connect_plugin(self.components, self.clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + ret = self.call_plugin(self.connect_plugin) if not ret: self.sub_io.stop_loading('fail') return False self.sub_io.stop_loading('succeed') - self.close() + if self.cursor: + self.close() self.cursor = ret.get_return('cursor') self.db = ret.get_return('connect') while self.execute_sql('use oceanbase', error=False) is False: @@ -73,18 +102,13 @@ def connect(self): return True def execute_sql(self, query, args=None, one=True, error=True): - msg = query % tuple(args) if args is not None else query - self.stdio.verbose("query: %s. args: %s" % (query, args)) - try: - self.stdio.verbose('execute sql: %s' % msg) - self.cursor.execute(query, args) - result = self.cursor.fetchone() if one else self.cursor.fetchall() - result and self.stdio.verbose(result) - return result - except: - msg = 'execute sql exception: %s' % msg if error else '' - self.stdio.exception(msg) - return False + exc_level = 'error' if error else 'verbose' + if one: + result = self.cursor.fetchone(query, args, exc_level=exc_level) + else: + result = self.cursor.fetchall(query, args, exc_level=exc_level) + result and self.stdio.verbose(result) + return result def broken_sql(self, sql, sleep_time=3): while True: @@ -135,12 +159,12 @@ def stop_zone(self, zone): def rollback(self): if self.new_clients: self.stdio.start_loading('Rollback') - self.stop_plugin(self.components, self.now_clients, self.new_cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io) + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.now_clients, cluster_config=cluster_config) for server in self.cluster_config.servers: client = self.clients[server] new_client = self.now_clients[server] server_config = self.cluster_config.get_server_conf(server) - home_path = server_config['home_path'] chown_cmd = 'sudo chown -R %s:' % client.config.username for key in ['home_path', 'data_dir', 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir']: if key in server_config: @@ -156,11 +180,10 @@ def dir_read_check(self, client, path): def _restart(self): clients = self.clients - self.stdio.verbose('Call %s for %s' % (self.stop_plugin, self.repository)) - if not self.stop_plugin(self.components, clients, self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io): + if not self.call_plugin(self.stop_plugin, clients=clients): self.stdio.stop_loading('stop_loading', 'fail') return False - + if self.new_clients: self.stdio.verbose('use new clients') for server in self.cluster_config.servers: @@ -178,10 +201,10 @@ def _restart(self): clients = self.new_clients cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config - self.stdio.verbose('Call %s for %s' % (self.start_plugin, self.repository)) - if not self.start_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, local_home_path=self.local_home_path, repository_dir=self.repository.repository_dir): + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, repository=self.repository): self.stdio.stop_loading('stop_loading', 'fail') return False + self.close() return True def rolling(self, zones_servers): @@ -205,13 +228,13 @@ def rolling(self, zones_servers): where svr_ip = %s and svr_port = %s and refreshed_schema_version > 1 ) as b on a.tenant_id = b.tenant_id where b.tenant_id is null''' - if self.execute_sql(sql, args=(server.ip, config['rpc_port'])).get('cnt'): + if self.execute_sql(sql, args=(server.ip, config['rpc_port']), error=False).get('cnt'): break else: break time.sleep(3) - while self.execute_sql("select * from oceanbase.__all_virtual_clog_stat where table_id = 1099511627777 and status != 'ACTIVE'"): + while self.execute_sql("select * from oceanbase.__all_virtual_clog_stat where table_id = 1099511627777 and status != 'ACTIVE'", error=False): time.sleep(3) self.stop_zone(zone) @@ -245,7 +268,7 @@ def restart(self): if self.connect(): self.stdio.start_loading('Server check') servers = self.execute_sql("select * from oceanbase.__all_server", one=False, error=False) - if len(self.cluster_config.servers) == len(servers): + if isinstance(servers, list) and len(self.cluster_config.servers) == len(servers): for server in servers: if server['status'] != 'active' or server['stop_time'] > 0 or server['start_service_time'] == 0: break @@ -256,7 +279,6 @@ def restart(self): if zone not in zones_servers: zones_servers[zone] = [] zones_servers[zone].append(server) - servers = self.cluster_config.servers self.stdio.stop_loading('succeed') ret = False try: @@ -266,11 +288,9 @@ def restart(self): ret = self.un_rolling() if ret and self.connect(): - self.display_plugin(self.components, self.new_clients if self.new_clients else self.clients, self.new_cluster_config if self.new_cluster_config else self.cluster_config, self.plugin_context.cmd, self.plugin_context.options, self.sub_io, cursor=self.cursor) + self.call_plugin(self.display_plugin, clients=self.now_clients, cluster_config=self.new_cluster_config if self.new_cluster_config else self.cluster_config, cursor=self.cursor) if self.new_cluster_config: - self.stdio.verbose('Call %s for %s' % (self.reload_plugin, self.repository)) - self.reload_plugin(self.components, self.clients, self.cluster_config, [], {}, self.sub_io, - cursor=self.cursor, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir) + self.call_plugin(self.reload_plugin, clients=self.now_clients, cursor=self.cursor, new_cluster_config=self.new_cluster_config, repository_dir=self.repository.repository_dir) except Exception as e: self.stdio.exception('Run Exception: %s' % e) finally: @@ -282,7 +302,8 @@ def restart(self): return ret -def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, rollback=False, *args, **kwargs): +def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, new_cluster_config=None, new_clients=None, rollback=False, *args, **kwargs): + repository = kwargs.get('repository') task = Restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, repository, new_cluster_config, new_clients) call = task.rollback if rollback else task.restart if call(): diff --git a/plugins/oceanbase/4.0.0.0/start.py b/plugins/oceanbase/4.0.0.0/start.py index 605d8c2..08d64cd 100644 --- a/plugins/oceanbase/4.0.0.0/start.py +++ b/plugins/oceanbase/4.0.0.0/start.py @@ -20,13 +20,12 @@ from __future__ import absolute_import, division, print_function -import os import json import time import requests from copy import deepcopy -from _errno import EC_OBSERVER_FAIL_TO_START +from _errno import EC_OBSERVER_FAIL_TO_START, EC_OBSERVER_FAIL_TO_START_WITH_ERR, EC_OBSERVER_FAILED_TO_REGISTER, EC_OBSERVER_FAILED_TO_REGISTER_WITH_DETAILS from collections import OrderedDict @@ -81,7 +80,7 @@ def __exit__(self, *args, **kwargs): self.client.del_env(env_key) -def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): +def start(plugin_context, *args, **kwargs): cluster_config = plugin_context.cluster_config options = plugin_context.options clients = plugin_context.clients @@ -101,10 +100,10 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): try: cfg_url = init_config_server(obconfig_url, appname, cluster_id, getattr(options, 'force_delete', False), stdio) if not cfg_url: - stdio.error('failed to register cluster. %s may have been registered in %s.' % (appname, obconfig_url)) + stdio.error(EC_OBSERVER_FAILED_TO_REGISTER_WITH_DETAILS.format(appname, obconfig_url)) return except: - stdio.exception('failed to register cluster') + stdio.exception(EC_OBSERVER_FAILED_TO_REGISTER.format()) return stdio.start_loading('Start observer') @@ -123,6 +122,9 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): if not server_config.get('data_dir'): server_config['data_dir'] = '%s/store' % home_path + if client.execute_command('ls %s/clog/tenant_1/' % server_config['data_dir']).stdout.strip(): + need_bootstrap = False + remote_pid_path = '%s/run/observer.pid' % home_path remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() if remote_pid: @@ -153,12 +155,13 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): }) not_cmd_opt = [ 'home_path', 'obconfig_url', 'root_password', 'proxyro_password', - 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir', '$_zone_idc', 'production_mode' + 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir', '$_zone_idc', 'production_mode', + 'ocp_meta_tenant', 'ocp_meta_username', 'ocp_meta_password', 'ocp_meta_db', 'ocp_agent_monitor_password' ] get_value = lambda key: "'%s'" % server_config[key] if isinstance(server_config[key], str) else server_config[key] opt_str = [] for key in server_config: - if key not in not_cmd_opt and key not in not_opt_str: + if key not in not_cmd_opt and key not in not_opt_str and not key.startswith('ocp_meta_tenant_'): value = get_value(key) opt_str.append('%s=%s' % (key, value)) if cfg_url: @@ -186,7 +189,7 @@ def start(plugin_context, local_home_path, repository_dir, *args, **kwargs): ret = client.execute_command(clusters_cmd[server]) if not ret: stdio.stop_loading('fail') - stdio.error(EC_OBSERVER_FAIL_TO_START.format(server=server) + ': ' + ret.stderr) + stdio.error(EC_OBSERVER_FAIL_TO_START_WITH_ERR.format(server=server, stderr=ret.stderr)) return stdio.stop_loading('succeed') diff --git a/plugins/oceanbase/4.0.0.0/start_check.py b/plugins/oceanbase/4.0.0.0/start_check.py index 98b9758..f3df2ae 100644 --- a/plugins/oceanbase/4.0.0.0/start_check.py +++ b/plugins/oceanbase/4.0.0.0/start_check.py @@ -23,12 +23,11 @@ import os import re import time +import copy +from math import sqrt + +import _errno as err -from _errno import ( - EC_OBSERVER_NOT_ENOUGH_DISK_4_CLOG, EC_CONFIG_CONFLICT_PORT, - EC_OBSERVER_NOT_ENOUGH_MEMORY, EC_ULIMIT_CHECK, WC_ULIMIT_CHECK, - EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE, EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED -) stdio = None success = True @@ -36,7 +35,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -83,27 +82,117 @@ def get_mount_path(disk, _path): return _mount_path -def _start_check(plugin_context, strict_check=False, *args, **kwargs): - def alert(*arg, **kwargs): +def get_system_memory(memory_limit, min_pool_memory): + if memory_limit <= 8 << 30: + system_memory = 2 << 30 + elif memory_limit <= 16 << 30: + system_memory = 3 << 30 + elif memory_limit <= 32 << 30: + system_memory = 5 << 30 + elif memory_limit <= 48 << 30: + system_memory = 7 << 30 + elif memory_limit <= 64 << 30: + system_memory = 10 << 30 + else: + memory_limit_gb = memory_limit >> 30 + system_memory = int(3 * (sqrt(memory_limit_gb) - 3)) << 30 + return max(system_memory, min_pool_memory) + + +def get_disk_info_by_path(path, client, stdio): + disk_info = {} + ret = client.execute_command('df --block-size=1024 {}'.format(path)) + if ret: + for total, used, avail, puse, path in re.findall(r'(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', ret.stdout): + disk_info[path] = {'total': int(total) << 10, 'avail': int(avail) << 10, 'need': 0} + stdio.verbose('get disk info for path {}, total: {} avail: {}'.format(path, disk_info[path]['total'], disk_info[path]['avail'])) + return disk_info + + +def get_disk_info(all_paths, client, stdio): + overview_ret = True + disk_info = get_disk_info_by_path('', client, stdio) + if not disk_info: + overview_ret = False + disk_info = get_disk_info_by_path('/', client, stdio) + if not disk_info: + disk_info['/'] = {'total': 0, 'avail': 0, 'need': 0} + all_path_success = {} + for path in all_paths: + all_path_success[path] = False + cur_path = path + while cur_path not in disk_info: + disk_info_for_current_path = get_disk_info_by_path(cur_path, client, stdio) + if disk_info_for_current_path: + disk_info.update(disk_info_for_current_path) + all_path_success[path] = True + break + else: + cur_path = os.path.dirname(cur_path) + if overview_ret or all(all_path_success.values()): + return disk_info + + +def start_check(plugin_context, init_check_status=False, strict_check=False, work_dir_check=False, work_dir_empty_check=True, generate_configs={}, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def alert(item, error, suggests=[]): global success if strict_check: success = False - stdio.error(*arg, **kwargs) + check_fail(item, error, suggests) + stdio.error(error) else: - stdio.warn(*arg, **kwargs) - def error(*arg, **kwargs): + stdio.warn(error) + def error(item, _error, suggests=[]): global success if plugin_context.dev_mode: - stdio.warn(*arg, **kwargs) + stdio.warn(_error) else: success = False - stdio.error(*arg, **kwargs) - def critical(*arg, **kwargs): + check_fail(item, _error, suggests) + stdio.error(_error) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) - global stdio + check_fail(item, error, suggests) + stdio.error(error) + + global stdio, success + success = True + check_status = {} cluster_config = plugin_context.cluster_config + plugin_context.set_variable('start_check_status', check_status) + + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + 'mem': err.CheckStatus(), + 'disk': err.CheckStatus(), + 'ulimit': err.CheckStatus(), + 'aio': err.CheckStatus(), + 'net': err.CheckStatus(), + 'ntp': err.CheckStatus(), + 'ocp meta db': err.CheckStatus() + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + clients = plugin_context.clients stdio = plugin_context.stdio servers_clients = {} @@ -112,30 +201,105 @@ def critical(*arg, **kwargs): servers_disk = {} servers_clog_mount = {} servers_net_inferface = {} - server_num = len(cluster_config.servers) - + servers_dirs = {} + servers_check_dirs = {} + servers_log_disk_size = {} + servers_min_pool_memory = {} PRO_MEMORY_MIN = 16 << 30 PRO_POOL_MEM_MIN = 2147483648 START_NEED_MEMORY = 3 << 30 + global_generate_config = generate_configs.get('global', {}) stdio.start_loading('Check before start observer') + + need_bootstrap = True for server in cluster_config.servers: ip = server.ip client = clients[server] + server_generate_config = generate_configs.get(server, {}) servers_clients[ip] = client server_config = cluster_config.get_server_conf_with_default(server) home_path = server_config['home_path'] - remote_pid_path = '%s/run/observer.pid' % home_path - remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s' % remote_pid): - continue + if not precheck: + if need_bootstrap: + data_dir = server_config['data_dir'] if server_config.get('data_dir') else '%s/store' % home_path + if client.execute_command('ls %s/clog/tenant_1/' % data_dir).stdout.strip(): + need_bootstrap = False + remote_pid_path = '%s/run/observer.pid' % home_path + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + continue + + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + original_server_conf = cluster_config.get_server_conf(server) + + if not server_config.get('data_dir'): + server_config['data_dir'] = '%s/store' % home_path + if not server_config.get('redo_dir'): + server_config['redo_dir'] = server_config['data_dir'] + if not server_config.get('clog_dir'): + server_config['clog_dir'] = '%s/clog' % server_config['redo_dir'] + if not server_config.get('ilog_dir'): + server_config['ilog_dir'] = '%s/ilog' % server_config['redo_dir'] + if not server_config.get('slog_dir'): + server_config['slog_dir'] = '%s/slog' % server_config['redo_dir'] + if server_config['redo_dir'] == server_config['data_dir']: + keys = ['home_path', 'data_dir', 'clog_dir', 'ilog_dir', 'slog_dir'] + else: + keys = ['home_path', 'data_dir', 'redo_dir', 'clog_dir', 'ilog_dir', 'slog_dir'] + + for key in keys: + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + if key not in original_server_conf: + continue + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False if ip not in servers_port: servers_disk[ip] = {} servers_port[ip] = {} servers_clog_mount[ip] = {} servers_net_inferface[ip] = {} - servers_memory[ip] = {'num': 0, 'percentage': 0, 'server_num': 0} + servers_memory[ip] = {'num': 0, 'percentage': 0, 'servers': {}} memory = servers_memory[ip] ports = servers_port[ip] disk = servers_disk[ip] @@ -145,54 +309,48 @@ def critical(*arg, **kwargs): for key in ['mysql_port', 'rpc_port']: port = int(server_config[key]) if port in ports: - critical(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key'])) + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) continue ports[port] = { 'server': server, 'key': key } if get_port_socket_inode(client, port): - critical('%s:%s port is already used' % (ip, port)) + critical('port', err.EC_CONFLICT_PORT.format(server=ip, port=port), [err.SUG_USE_OTHER_PORT.format()]) - __min_full_resource_pool_memory = server_config.get('__min_full_resource_pool_memory') + servers_min_pool_memory[server] = __min_full_resource_pool_memory = server_config.get('__min_full_resource_pool_memory') if server_config.get('production_mode') and __min_full_resource_pool_memory < PRO_POOL_MEM_MIN: - error('(%s): when production_mode is True, __min_full_resource_pool_memory can not be less then %s' % (server, PRO_POOL_MEM_MIN)) + error('mem', err.EC_OBSERVER_PRODUCTION_MODE_LIMIT.format(server=server, key="__min_full_resource_pool_memory", limit=PRO_POOL_MEM_MIN), [err.SUB_SET_NO_PRODUCTION_MODE.format()]) - memory['server_num'] += 1 + memory_limit = 0 + percentage = 0 if 'memory_limit' in server_config: - try: - memory_limit = parse_size(server_config['memory_limit']) - if server_config.get('production_mode') and memory_limit < PRO_MEMORY_MIN: - error('(%s): when production_mode is True, memory_limit can not be less then %s' % (server, format_size(PRO_MEMORY_MIN))) - memory['num'] += memory_limit - except: - error('memory_limit must be an integer') - return + memory_limit = parse_size(server_config['memory_limit']) + if server_config.get('production_mode') and memory_limit < PRO_MEMORY_MIN: + error('mem', err.EC_OBSERVER_PRODUCTION_MODE_LIMIT.format(server=server, key='memory_limit', limit=format_size(PRO_MEMORY_MIN)), [err.SUB_SET_NO_PRODUCTION_MODE.format()]) + memory['num'] += memory_limit elif 'memory_limit_percentage' in server_config: - try: - memory['percentage'] += int(parse_size(server_config['memory_limit_percentage'])) - except: - error('memory_limit_percentage must be an integer') - return + percentage = int(parse_size(server_config['memory_limit_percentage'])) + memory['percentage'] += percentage else: - memory['percentage'] += 80 + percentage = 80 + memory['percentage'] += percentage + memory['servers'][server] = { + 'num': memory_limit, + 'percentage': percentage, + 'system_memory': parse_size(server_config.get('system_memory', 0)) + } data_path = server_config['data_dir'] if server_config.get('data_dir') else os.path.join(server_config['home_path'], 'store') redo_dir = server_config['redo_dir'] if server_config.get('redo_dir') else data_path clog_dir = server_config['clog_dir'] if server_config.get('clog_dir') else os.path.join(redo_dir, 'clog') if not client.execute_command('ls %s/sstable/block_file' % data_path): - if data_path in disk: - critical('Same Path: %s in %s and %s' % (data_path, server, disk[data_path]['server'])) - continue - if clog_dir in clog_mount: - critical('Same Path: %s in %s and %s' % (clog_dir, server, clog_mount[clog_dir]['server'])) - continue - disk[data_path] = { - 'server': server - } - clog_mount[clog_dir] = { - 'server': server - } + disk[data_path] = {'server': server} + clog_mount[clog_dir] = {'server': server} if 'datafile_size' in server_config and server_config['datafile_size'] and parse_size(server_config['datafile_size']): # if need is string, it means use datafile_size disk[data_path]['need'] = server_config['datafile_size'] @@ -210,38 +368,51 @@ def critical(*arg, **kwargs): devname = server_config.get('devname') if devname: if not client.execute_command("grep -e '^ *%s:' /proc/net/dev" % devname): - critical('%s No such net interface: %s' % (server, devname)) + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + suggest.auto_fix = 'devname' not in global_generate_config and 'devname' not in server_generate_config + critical('net', err.EC_NO_SUCH_NET_DEVICE.format(server=server, devname=devname), suggests=[suggest]) if devname not in inferfaces: inferfaces[devname] = [] inferfaces[devname].append(ip) + + ip_server_memory_info = {} for ip in servers_disk: + ip_servers = servers_memory[ip]['servers'].keys() + server_num = len(ip_servers) client = servers_clients[ip] ret = client.execute_command('cat /proc/sys/fs/aio-max-nr /proc/sys/fs/aio-nr') if not ret: - alert('(%s) failed to get fs.aio-max-nr and fs.aio-nr' % ip) + for server in ip_servers: + alert('aio', err.EC_FAILED_TO_GET_AIO_NR.format(ip=ip), [err.SUG_CONNECT_EXCEPT.format()]) else: try: max_nr, nr = ret.stdout.strip().split('\n') max_nr, nr = int(max_nr), int(nr) need = server_num * 20000 + RECD_AIO = 1048576 if need > max_nr - nr: - critical('(%s) Insufficient AIO remaining (Avail: %s, Need: %s), The recommended value of fs.aio-max-nr is 1048576' % (ip, max_nr - nr, need)) - elif int(max_nr) < 1048576: - alert('(%s) The recommended value of fs.aio-max-nr is 1048576 (Current value: %s)' % (ip, max_nr)) + for server in ip_servers: + critical('aio', err.EC_AIO_NOT_ENOUGH.format(ip=ip, avail=max_nr - nr, need=need), [err.SUG_SYSCTL.format(var='fs.aio-max-nr', value=max(RECD_AIO, need), ip=ip)]) + elif int(max_nr) < RECD_AIO: + for server in ip_servers: + alert('aio', err.WC_AIO_NOT_ENOUGH.format(ip=ip, current=max_nr), [err.SUG_SYSCTL.format(var='fs.aio-max-nr', value=RECD_AIO, ip=ip)]) except: - alert('(%s) failed to get fs.aio-max-nr and fs.aio-nr' % ip) + for server in ip_servers: + alert('aio', err.EC_FAILED_TO_GET_AIO_NR.format(ip=ip), [err.SUG_UNSUPPORT_OS.format()]) stdio.exception('') ret = client.execute_command('ulimit -a') ulimits_min = { 'open files': { 'need': lambda x: 20000 * x, - 'recd': lambda x: 655350 + 'recd': lambda x: 655350, + 'name': 'nofile' }, 'max user processes': { 'need': lambda x: 4096, - 'recd': lambda x: 4096 * x + 'recd': lambda x: 4096 * x, + 'name': 'nproc' }, } ulimits = {} @@ -253,16 +424,20 @@ def critical(*arg, **kwargs): if value == 'unlimited': continue if not value or not (value.strip().isdigit()): - alert('(%s) failed to get %s' % (ip, key)) + for server in ip_servers: + alert('ulimit', '(%s) failed to get %s' % (ip, key), [err.SUG_UNSUPPORT_OS.format()]) else: value = int(value) need = ulimits_min[key]['need'](server_num) if need > value: - critical(EC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value)) + for server in ip_servers: + critical('ulimit', err.EC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value), [err.SUG_ULIMIT.format(name=ulimits_min[key]['name'], value=need, ip=ip)]) else: need = ulimits_min[key]['recd'](server_num) if need > value: - alert(WC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value)) + for server in ip_servers: + alert('ulimit', err.WC_ULIMIT_CHECK.format(server=ip, key=key, need=need, now=value), [err.SUG_ULIMIT.format(name=ulimits_min[key]['name'], value=need, ip=ip)]) + # memory ret = client.execute_command('cat /proc/meminfo') @@ -282,40 +457,45 @@ def critical(*arg, **kwargs): key = memory_key_map[k] server_memory_stats[key] = parse_size(str(v)) - min_start_need = servers_memory[ip]['server_num'] * START_NEED_MEMORY - total_use = servers_memory[ip]['percentage'] * server_memory_stats['total'] / 100 + servers_memory[ip]['num'] + ip_server_memory_info[ip] = server_memory_stats + server_memory_stat = servers_memory[ip] + min_start_need = server_num * START_NEED_MEMORY + total_use = server_memory_stat['percentage'] * server_memory_stats['total'] / 100 + server_memory_stat['num'] if min_start_need > server_memory_stats['available']: - error(EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(min_start_need))) + for server in ip_servers: + error('mem', err.EC_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(min_start_need)), [err.SUG_OBSERVER_NOT_ENOUGH_MEMORY_ALAILABLE.format(ip=ip)]) elif total_use > server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached']: - error(EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(total_use))) + for server in ip_servers: + server_generate_config = generate_configs.get(server, {}) + suggest = err.SUG_OBSERVER_REDUCE_MEM.format() + suggest.auto_fix = True + for key in ['memory_limit', 'memory_limit_percentage']: + if key in global_generate_config or key in server_generate_config: + suggest.auto_fix = False + break + error('mem', err.EC_OBSERVER_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(total_use)), [suggest]) elif total_use > server_memory_stats['free']: - alert(EC_OBSERVER_NOT_ENOUGH_MEMORY.format(ip=ip, free=format_size(server_memory_stats['free']), need=format_size(total_use))) + for server in ip_servers: + alert('mem', err.EC_OBSERVER_NOT_ENOUGH_MEMORY.format(ip=ip, free=format_size(server_memory_stats['free']), need=format_size(total_use)), [err.SUG_OBSERVER_REDUCE_MEM.format()]) + else: + server_memory_config = server_memory_stat['servers'] + for server in server_memory_config: + if server_memory_config[server]['system_memory']: + memory_limit = server_memory_config[server]['num'] + if not memory_limit: + server_memory_config[server]['num'] = memory_limit = server_memory_config[server]['percentage'] * server_memory_stats['total'] + + factor = 0.75 + suggest = err.SUG_OBSERVER_SYS_MEM_TOO_LARGE.format(factor=factor) + suggest.auto_fix = 'system_memory' not in global_generate_config and 'system_memory' not in generate_configs.get(server, {}) + if memory_limit < server_memory_config[server]['system_memory']: + critical('mem', err.EC_OBSERVER_SYS_MEM_TOO_LARGE.format(server=server), [suggest]) + elif memory_limit * factor < server_memory_config[server]['system_memory']: + alert('mem', err.WC_OBSERVER_SYS_MEM_TOO_LARGE.format(server=server, factor=factor), [suggest]) + # disk - disk = {'/': 0} - ret = client.execute_command('df --block-size=1024') - if ret: - for total, used, avail, puse, path in re.findall('(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', ret.stdout): - disk[path] = { - 'total': int(total) << 10, - 'avail': int(avail) << 10, - 'need': 0, - } all_path = set(list(servers_disk[ip].keys()) + list(servers_clog_mount[ip].keys())) - for include_dir in all_path: - while include_dir not in disk: - ret = client.execute_command('df --block-size=1024 %s' % include_dir) - if ret: - for total, used, avail, puse, path in re.findall('(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', - ret.stdout): - disk[path] = { - 'total': int(total) << 10, - 'avail': int(avail) << 10, - 'need': 0, - } - break - else: - include_dir = os.path.dirname(include_dir) - + disk = get_disk_info(all_paths=all_path, client=client, stdio=stdio) stdio.verbose('disk: {}'.format(disk)) for path in servers_disk[ip]: mount_path = get_mount_path(disk, path) @@ -335,11 +515,7 @@ def critical(*arg, **kwargs): # slog need 10G disk[mount_path]['need'] += max(disk[mount_path]['total'] - slog_size, 0) * need / 100 else: - try: - disk[mount_path]['need'] += parse_size(need) - except: - critical('datafile_size must be an integer') - return + disk[mount_path]['need'] += parse_size(need) disk[mount_path]['need'] += slog_size disk[mount_path]['is_data_disk'] = True @@ -356,22 +532,104 @@ def critical(*arg, **kwargs): stdio.verbose('clog percentage: {}'.format(need)) if isinstance(need, int): # log_disk_percentage - disk[mount_path]['need'] += disk[mount_path]['total'] * need / 100 + log_disk_size = disk[mount_path]['total'] * need / 100 else: - try: - # log_disk_size - disk[mount_path]['need'] += parse_size(need) - except: - critical('log_disk_size must be valid size string') - return + # log_disk_size + log_disk_size = parse_size(need) + servers_log_disk_size[servers_clog_mount[ip][path]['server']] = log_disk_size + disk[mount_path]['need'] += log_disk_size disk[mount_path]['is_clog_disk'] = True for p in disk: avail = disk[p]['avail'] need = disk[p]['need'] + suggests = [] if disk[p].get('is_data_disk') and disk[p].get('is_clog_disk'): - alert('(%s) clog and data use the same disk (%s)' % (ip, p)) + suggests.append(err.SUG_OBSERVER_SAME_DISK.format()) + for server in ip_servers: + alert('disk', err.WC_OBSERVER_SAME_DISK.format(ip=ip, disk=p), suggests) if need > avail: - critical('(%s) %s not enough disk space. (Avail: %s, Need: %s)' % (ip, p, format_size(avail), format_size(need))) + suggest_temps = { + 'data': { + 'tmplate': err.SUG_OBSERVER_NOT_ENOUGH_DISK, + 'keys': ['datafile_size', 'datafile_disk_percentage'] + } + } + if suggests: + suggest_temps['mem'] = { + 'tmplate': err.SUG_OBSERVER_REDUCE_MEM, + 'keys': ['memory_limit', 'memory_limit_percentage'] + } + suggest_temps['redo'] = { + 'tmplate': err.SUG_OBSERVER_REDUCE_REDO, + 'keys': ['log_disk_size', 'log_disk_percentage'] + } + for server in ip_servers: + tmp_suggests = [] + server_generate_config = generate_configs.get(server, {}) + for item in suggest_temps: + suggest = suggest_temps[item]['tmplate'].format() + suggest.auto_fix = True + for key in suggest_temps[item]['keys']: + if key in global_generate_config or key in server_generate_config: + suggest.auto_fix = False + break + tmp_suggests.append(suggest) + tmp_suggests = sorted(tmp_suggests, key=lambda suggest: suggest.auto_fix, reverse=True) + critical('disk', err.EC_OBSERVER_NOT_ENOUGH_DISK.format(ip=ip, disk=p, avail=format_size(avail), need=format_size(need)), tmp_suggests + suggests) + + global_conf = cluster_config.get_global_conf() + has_ocp = 'ocp-express' in plugin_context.components + if not has_ocp and any([key.startswith('ocp_meta') for key in global_conf]): + has_ocp = True + if has_ocp and need_bootstrap: + global_conf_with_default = copy.deepcopy(cluster_config.get_global_conf_with_default()) + ocp_meta_tenant_prefix = 'ocp_meta_tenant_' + for key in global_conf_with_default: + if key.startswith(ocp_meta_tenant_prefix): + global_conf_with_default['ocp_meta_tenant'][key.replace(ocp_meta_tenant_prefix, '', 1)] = global_conf_with_default[key] + meta_db_memory_size = parse_size(global_conf_with_default['ocp_meta_tenant'].get('memory_size')) + servers_sys_memory = {} + if meta_db_memory_size: + sys_memory_size = None + if 'sys_tenant' in global_conf and 'memory_size' in global_conf['sys_tenant']: + sys_memory_size = global_conf['sys_tenant']['memory_size'] + for server in cluster_config.servers: + if server.ip not in servers_memory or server not in servers_memory[server.ip]['servers'] or server not in servers_min_pool_memory: + stdio.verbose('skip server {} for missing some memory info.'.format(server)) + continue + memory_limit = servers_memory[server.ip]['servers'][server]['num'] + system_memory = servers_memory[server.ip]['servers'][server]['system_memory'] + min_pool_memory = servers_min_pool_memory[server] + if system_memory == 0: + system_memory = get_system_memory(memory_limit, min_pool_memory) + if not sys_memory_size: + sys_memory_size = servers_sys_memory[server] = max(min_pool_memory, min((memory_limit - system_memory) * 0.25, parse_size('16G'))) + if meta_db_memory_size + system_memory + sys_memory_size <= memory_limit: + break + else: + suggest = err.SUG_OCP_EXPRESS_REDUCE_META_DB_MEM.format() + suggest.auto_fix = True + if 'ocp_meta_tenant_memory_size' in global_generate_config: + suggest.auto_fix = False + error('ocp meta db', err.EC_OCP_EXPRESS_META_DB_NOT_ENOUGH_MEM.format(), [suggest]) + + meta_db_log_disk_size = global_conf_with_default['ocp_meta_tenant'].get('log_disk_size') + meta_db_log_disk_size = parse_size(meta_db_log_disk_size) if meta_db_log_disk_size else meta_db_log_disk_size + if not meta_db_log_disk_size and meta_db_memory_size: + meta_db_log_disk_size = meta_db_memory_size * 3 + if meta_db_log_disk_size: + for server in cluster_config.servers: + log_disk_size = servers_log_disk_size[server] + sys_log_disk_size = servers_sys_memory.get(server, 0) + if meta_db_log_disk_size + sys_log_disk_size <= log_disk_size: + break + else: + suggest = err.SUG_OCP_EXPRESS_REDUCE_META_DB_LOG_DISK.format() + suggest.auto_fix = True + if 'ocp_meta_tenant_log_disk_size' in global_generate_config: + suggest.auto_fix = False + error('ocp meta db', err.EC_OCP_EXPRESS_META_DB_NOT_ENOUGH_LOG_DISK.format(), [suggest]) + if success: for ip in servers_net_inferface: @@ -385,24 +643,34 @@ def critical(*arg, **kwargs): interfaces = ['lo'] if len(interfaces) > 1: servers = ','.join(str(server) for server in servers_net_inferface[ip][None]) - critical('%s has more than one network inferface. Please set `devname` for (%s)' % (ip, servers)) + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + for server in ip_servers: + critical('net', err.EC_OBSERVER_MULTI_NET_DEVICE.format(ip=ip, server=servers), [suggest]) else: servers_net_inferface[ip][interfaces[0]] = servers_net_inferface[ip][None] del servers_net_inferface[ip][None] + if success: for ip in servers_net_inferface: client = servers_clients[ip] for devname in servers_net_inferface[ip]: if client.is_localhost() and devname != 'lo' or (not client.is_localhost() and devname == 'lo'): - critical('%s %s fail to ping %s. Please check configuration `devname`' % (server, devname, ip)) - continue + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + suggest.auto_fix = client.is_localhost() and 'devname' not in global_generate_config and 'devname' not in server_generate_config + for server in ip_servers: + critical('net', err.EC_OBSERVER_PING_FAILED.format(ip1=ip, devname=devname, ip2=ip), [suggest]) + continue for _ip in servers_clients: if ip == _ip: continue if not client.execute_command('ping -W 1 -c 1 -I %s %s' % (devname, _ip)): - critical('%s %s fail to ping %s. Please check configuration `devname`' % (server, devname, _ip)) + suggest = err.SUG_NO_SUCH_NET_DEVIC.format(ip=ip) + suggest.auto_fix = 'devname' not in global_generate_config and 'devname' not in server_generate_config + for server in ip_servers: + critical('net', err.EC_OBSERVER_PING_FAILED.format(ip1=ip, devname=devname, ip2=_ip), [suggest]) break + if success: times = [] for ip in servers_clients: @@ -410,12 +678,14 @@ def critical(*arg, **kwargs): delta = time_delta(client) stdio.verbose('%s time delta %s' % (ip, delta)) times.append(delta) - if times and max(times) - min(times) > 200: - critical('Cluster NTP is out of sync') - + if times and max(times) - min(times) > 500: + critical('ntp', err.EC_OBSERVER_TIME_OUT_OF_SYNC.format(), [err.SUG_OBSERVER_TIME_OUT_OF_SYNC.format()]) + for server in cluster_config.servers: + status = check_status[server] + for key in status: + if status[key].status == err.CheckStatus.WAIT: + status[key].status = err.CheckStatus.PASS -def start_check(plugin_context, strict_check=False, *args, **kwargs): - _start_check(plugin_context, strict_check) if success: stdio.stop_loading('succeed') plugin_context.return_true() diff --git a/plugins/oceanbase/4.0.0.0/upgrade.py b/plugins/oceanbase/4.0.0.0/upgrade.py new file mode 100644 index 0000000..3b146dd --- /dev/null +++ b/plugins/oceanbase/4.0.0.0/upgrade.py @@ -0,0 +1,544 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os +import time +import tool +import datetime +from ssh import LocalClient + + +class Exector(object): + + def __init__(self, tmp_prefix, host, port, user, pwd, exector_path, stdio): + self.tmp_prefix = tmp_prefix + self._host = host + self._port = port + self._user = user + self._pwd = pwd + self._cmd = None + self.stdio = stdio + self._exector = os.path.join(exector_path, 'executer27/bin/executer') + + @property + def host(self): + return self._host + + @property + def port(self): + return self._port + + @property + def user(self): + return self._user + + @property + def pwd(self): + return self._pwd + + @property + def exector(self): + return self._exector + + @property + def cmd(self): + if self._cmd is None: + self._cmd = '%s %%s -h %s -P %s -u %s %s' % (self._exector, self.host, self.port, self.user, "-p '%s'" % self.pwd if self.pwd else '') + return self._cmd + + @host.setter + def host(self, value): + self._host = value + self._cmd = None + + @port.setter + def port(self, value): + self._port = value + self._cmd = None + + @user.setter + def user(self, value): + self._user = value + self._cmd = None + + @pwd.setter + def pwd(self, value): + self._pwd = value + self._cmd = None + + @pwd.setter + def exector(self, exector_path): + self._exector = os.path.join(exector_path, 'bin/executer27') + self._cmd = None + + def create_temp(self, repository, direct_upgrade=False): + tmp_path = os.path.join('/tmp', self.tmp_prefix, repository.md5) + if not os.path.exists(tmp_path): + relative_dir = 'etc/direct_upgrade' if direct_upgrade else 'etc' + script_dir = os.path.join(repository.repository_dir, relative_dir) + LocalClient.put_dir(script_dir, tmp_path) + return tmp_path + + def clear_temp(self): + tmp_path = os.path.join('/tmp', self.tmp_prefix) + tool.DirectoryUtil.rm(tmp_path) + + def exec_script(self, name, repository, direct_upgrade=False, can_skip=False): + script_dir = self.create_temp(repository, direct_upgrade) + path = os.path.join(script_dir, name) + self.stdio.verbose('exec %s %s' % (repository, name)) + if os.path.exists(path): + cmd = self.cmd % path + self.stdio.start_loading('Exec %s %s' % (repository, name)) + if LocalClient.execute_command(cmd, stdio=self.stdio): + self.stdio.stop_loading('succeed') + return True + else: + self.stdio.stop_loading('fail') + return False + else: + if can_skip: + self.stdio.print('skip %s %s' % (repository, name)) + return True + else: + self.stdio.error('No such file: %s' % path) + return False + + +class Upgrader(object): + + def __init__(self, plugin_context, search_py_script_plugin, apply_param_plugin, upgrade_ctx, upgrade_repositories, local_home_path, exector_path, install_repository_to_servers, unuse_lib_repository): + self._search_py_script_plugin = search_py_script_plugin + self.apply_param_plugin = apply_param_plugin + self.plugin_context = plugin_context + self.components = plugin_context.components + self.clients = plugin_context.clients + self.cluster_config = plugin_context.cluster_config + self.stdio = plugin_context.stdio + self._connect_plugin = None + self._start_plugin = None + self._stop_plugin = None + self._display_plugin = None + self.install_repository_to_servers = install_repository_to_servers + self.unuse_lib_repository = unuse_lib_repository + self.local_home_path = local_home_path + self.exector_path = exector_path + self.components = plugin_context.components + self.exector = None + self.db = None + self.cursor = None + self.repositories = upgrade_repositories + self.upgrade_ctx = upgrade_ctx + self.route = upgrade_ctx.get('route') + self.route_index = upgrade_ctx.get('index') + self.process_index = upgrade_ctx.get('process_index', 0) + self.process_route_index = upgrade_ctx.get('process_route_index', self.route_index) + self.process = [ + self.exec_upgrade_checker, + self.upgrade_mode_on, + self.exec_upgrade_pre, + self.upgrade_zone, + self.upgrade_virtual_schema, + self.exec_upgrade_post, + self.upgrade_mode_off, + self.root_inspect, + self.exec_upgrade_post_checker + ] + self.process_total = len(self.process) + key = [] + for server in self.cluster_config.servers: + config = self.cluster_config.get_server_conf_with_default(server) + port = config.get('rpc_port') + key.append('%s:%s' % (server.ip, port)) + self.tmp_prefix = '_'.join(key) + + def search_py_script_plugin(self, index, name): + repository = self.repositories[index] + return self._search_py_script_plugin([repository], name)[repository] + + @property + def connect_plugin(self): + if self._connect_plugin is None: + self._connect_plugin = self.search_py_script_plugin(self.route_index - 1, 'connect') + return self._connect_plugin + + @property + def start_plugin(self): + if self._start_plugin is None: + self._start_plugin = self.search_py_script_plugin(self.next_stage, 'start') + return self._start_plugin + + @property + def stop_plugin(self): + if self._stop_plugin is None: + self._stop_plugin = self.search_py_script_plugin(self.route_index - 1, 'stop') + return self._stop_plugin + + @property + def display_plugin(self): + if self._display_plugin is None: + self._display_plugin = self.search_py_script_plugin(self.route_index - 1, 'display') + return self._display_plugin + + def _clear_plugin(self): + self._connect_plugin = None + self._start_plugin = None + self._stop_plugin = None + self._display_plugin = None + + def call_plugin(self, plugin, *args, **kwargs): + return plugin(self.plugin_context.namespace, self.plugin_context.namespaces, self.plugin_context.deploy_name, + self.plugin_context.repositories, self.plugin_context.components, self.plugin_context.clients, + self.plugin_context.cluster_config, self.plugin_context.cmds, self.plugin_context.options, + self.plugin_context.stdio, *args, **kwargs) + + def run(self): + total = len(self.route) + self.apply_param_plugin(self.repositories[self.route_index - 1]) + while self.route_index < total: + self.call_plugin(self.start_plugin, local_home_path=None, repository_dir=None) + self.close() + if not self.connect(): + return False + self.stdio.verbose('upgrade %s to %s' % (self.repositories[self.route_index], self.repositories[self.next_stage])) + while self.process_index < self.process_total: + try: + if not self.process[self.process_index](): + self._dump() + return False + self.process_index += 1 + self.process_route_index = self.route_index + except Exception as e: + self._dump() + self.stdio.exception(str(e)) + return False + self.process_index = 0 + self.route_index = self.next_stage + 1 + self.exector.clear_temp() + self.stdio.verbose('set route index from %s to %s' % (self.route_index, self.next_stage + 1)) + break + self._dump() + return True + + def _dump(self): + self.upgrade_ctx['route'] = self.route + self.upgrade_ctx['index'] = self.route_index + self.upgrade_ctx['process_index'] = self.process_index + self.upgrade_ctx['process_route_index'] = self.process_route_index + + def close(self): + if self.db: + self.cursor.close() + self.cursor = None + self.db = None + self.exector = None + + def connect(self): + if self.cursor is None or self.execute_sql('select version()', error=False) is False: + ret = self.call_plugin(self.connect_plugin) + if not ret: + return False + if self.cursor: + self.close() + self.cursor = ret.get_return('cursor') + self.db = ret.get_return('connect') + while self.execute_sql('use oceanbase', error=False) is False: + time.sleep(2) + self.execute_sql('set session ob_query_timeout=1000000000') + server = ret.get_return('server') + host = server.ip + port = self.db.port + user = 'root' + pwd = self.cluster_config.get_global_conf().get('root_password', '') + self.exector = Exector(self.tmp_prefix, host, port, user, pwd if pwd is not None else '', self.exector_path, self.stdio) + return True + + def execute_sql(self, query, args=None, one=True, error=True): + exc_level = 'error' if error else 'verbose' + if one: + result = self.cursor.fetchone(query, args, exc_level=exc_level) + else: + result = self.cursor.fetchall(query, args, exc_level=exc_level) + result and self.stdio.verbose(result) + return result + + @property + def next_stage(self): + next_stage = self.route_index + total = len(self.route) - 1 + while next_stage < total: + node = self.route[next_stage] + if node.get('require_from_binary'): + break + next_stage += 1 + return next_stage + + def _exec_script_dest_only(self, name, can_skip=True): + self.stdio.start_loading('Exec %s' % name) + next_stage = self.next_stage + repository = self.repositories[next_stage] + self.stdio.verbose('exec %s %s' % (repository, name)) + if not self.exector.exec_script(name, repository, direct_upgrade=self.route[next_stage].get('direct_upgrade'), can_skip=can_skip): + return False + self.stdio.stop_loading('succeed') + return True + + def _exec_script_all_repositories(self, name, can_skip=False): + self.stdio.start_loading('Exec %s' % name) + next_stage = self.next_stage + cur_repository = self.repositories[self.route_index - 1] + while self.process_route_index <= next_stage: + repository = self.repositories[self.process_route_index] + if cur_repository.version == repository.version: + self.stdio.verbose('skip %s %s' % (repository, name)) + else: + self.stdio.verbose('exec %s %s' % (repository, name)) + if not self.exector.exec_script(name, repository, direct_upgrade=self.route[self.process_route_index].get('direct_upgrade'), can_skip=can_skip): + self.stdio.stop_loading('fail') + return False + self.process_route_index += 1 + self.stdio.stop_loading('succeed') + return True + + def execute_upgrade_sql(self, query, args=None, one=True): + if self.execute_sql(query, args, one) is False: + return False + self.process_route_index = self.route_index + return True + + def exec_upgrade_checker(self): + return self._exec_script_dest_only('upgrade_checker.py') + + def upgrade_mode_on(self): + self.stdio.start_loading('Enable upgrade mode') + if self.execute_upgrade_sql('alter system begin upgrade') is False: + self.stdio.stop_loading('fail') + return False + time.sleep(2) + sql = "select value from oceanbase.__all_virtual_sys_parameter_stat where name = 'enable_upgrade_mode' and value = 'False'" + while True: + if not self.execute_sql(sql, error=False): + self.stdio.stop_loading('succeed') + return True + time.sleep(2) + + def exec_upgrade_pre(self): + return self._exec_script_all_repositories('upgrade_pre.py') + + def broken_sql(self, sql, sleep_time=3): + while True: + ret = self.execute_sql(sql, error=False) + if ret is None: + break + time.sleep(sleep_time) + + def wait(self): + if not self.connect(): + return False + self.stdio.verbose('server cneck') + self.broken_sql("select * from oceanbase.DBA_OB_SERVERS where STATUS != 'ACTIVE' or STOP_TIME is not NULL or START_SERVICE_TIME is NULL") + self.broken_sql("select * from GV$OB_LOG_STAT where in_sync = 'NO'") + return True + + def start_zone(self, zone=None): + if not self.connect(): + return False + if zone: + self.stdio.verbose('start zone %s' % zone) + start_sql = "alter system start zone %s" % zone + check_sql = "select * from oceanbase.__all_zone where name = 'status' and zone = '%s' and info != 'ACTIVE'" % zone + while True: + if self.execute_sql(start_sql, error=False) is None: + break + if self.execute_sql(check_sql, error=False) is None: + break + time.sleep(3) + self.wait() + return True + + def stop_zone(self, zone): + if not self.wait(): + return False + + self.stdio.verbose('stop zone %s' % zone) + stop_sql = "alter system stop zone %s" % zone + check_sql = "select * from oceanbase.__all_zone where name = 'status' and zone = '%s' and info = 'ACTIVE'" % zone + while True: + if self.execute_sql(stop_sql, error=False) is None: + break + if self.execute_sql(check_sql, error=False): + break + time.sleep(3) + return True + + def upgrade_zone(self): + zones_servers = {} + for server in self.cluster_config.servers: + config = self.cluster_config.get_server_conf_with_default(server) + zone = config['zone'] + if zone not in zones_servers: + zones_servers[zone] = [] + zones_servers[zone].append(server) + servers = self.cluster_config.servers + try: + if len(zones_servers) > 2: + ret = self.rolling_upgrade(zones_servers) + else: + ret = self.un_rolling_upgrade() + if ret: + self._clear_plugin() + return True + return False + except Exception as e: + self.stdio.exception('Run Exception: %s' % e) + return False + finally: + self.cluster_config.servers = servers + + def un_rolling_upgrade(self): + self.stdio.start_loading('Upgrade') + repository = self.repositories[self.next_stage] + repository_dir = repository.repository_dir + self.install_repository_to_servers(self.components, self.cluster_config, repository, self.clients, + self.unuse_lib_repository) + + if not self.call_plugin(self.stop_plugin): + self.stdio.stop_loading('stop_loading', 'fail') + return False + + self.apply_param_plugin(repository) + if not self.call_plugin(self.start_plugin, local_home_path=self.local_home_path, repository_dir=repository_dir): + self.stdio.stop_loading('stop_loading', 'fail') + return False + self.close() + self.wait() + self.stdio.stop_loading('succeed') + return True + + def rolling_upgrade(self, zones_servers): + self.stdio.start_loading('Rotation upgrade') + all_servers = self.cluster_config.servers + repository = self.repositories[self.next_stage] + repository_dir = repository.repository_dir + pre_zone = None + for zone in zones_servers: + self.cluster_config.servers = zones_servers[zone] + if not self.start_zone(pre_zone): + self.stdio.stop_loading('stop_loading', 'fail') + return False + while True: + for server in zones_servers[zone]: + config = self.cluster_config.get_server_conf(server) + sql = ''' + select count(*) from oceanbase.DBA_OB_TENANTS as a left join ( + select tenant_id, refreshed_schema_version + from GV$OB_SERVER_SCHEMA_INFO + where svr_ip = %s and svr_port = %s and refreshed_schema_version > 1 + ) as b on a.tenant_id = b.tenant_id + where b.tenant_id is null''' + if self.execute_sql(sql, args=(server.ip, config['rpc_port'])).get('cnt'): + break + else: + break + time.sleep(3) + + # while self.execute_sql("select * from oceanbase.__all_virtual_clog_stat where table_id = 1099511627777 and status != 'ACTIVE'"): + # time.sleep(3) + + self.stop_zone(zone) + + self.stdio.print('upgrade zone "%s"' % zone) + self.install_repository_to_servers(self.components, self.cluster_config, repository, self.clients, self.unuse_lib_repository) + + + if pre_zone: + self.apply_param_plugin(self.repositories[self.route_index - 1]) + if not self.call_plugin(self.stop_plugin): + self.stdio.stop_loading('stop_loading', 'fail') + return False + + self.apply_param_plugin(repository) + if not self.call_plugin(self.start_plugin, local_home_path=self.local_home_path, repository_dir=repository_dir): + self.stdio.stop_loading('stop_loading', 'fail') + return False + self.close() + pre_zone = zone + + if not self.start_zone(pre_zone): + self.stdio.stop_loading('stop_loading', 'fail') + return False + self.stdio.stop_loading('succeed') + return True + + def upgrade_virtual_schema(self): + return self.execute_upgrade_sql('alter system upgrade virtual schema') + + def exec_upgrade_post(self): + return self._exec_script_all_repositories('upgrade_post.py') + + def upgrade_mode_off(self): + self.stdio.start_loading('Disable upgrade mode') + if self.execute_upgrade_sql('alter system end upgrade') is False: + self.stdio.stop_loading('fail') + return False + time.sleep(2) + sql = "select value from oceanbase.__all_virtual_sys_parameter_stat where name = 'enable_upgrade_mode' and value = 'True'" + while True: + if not self.execute_sql(sql, error=False): + self.stdio.stop_loading('succeed') + return True + time.sleep(2) + + def root_inspect(self): + self.stdio.start_loading('Root inspection') + if self.execute_sql("alter system run job 'root_inspection'") is False: + self.stdio.stop_loading('fail') + return False + self.stdio.stop_loading('succeed') + return True + + def exec_upgrade_post_checker(self): + return self._exec_script_dest_only('upgrade_post_checker.py') + + +def upgrade(plugin_context, search_py_script_plugin, apply_param_plugin, install_repository_to_servers, unuse_lib_repository, *args, **kwargs): + + upgrade_ctx = kwargs.get('upgrade_ctx') + local_home_path = kwargs.get('local_home_path') + upgrade_repositories = kwargs.get('upgrade_repositories') + exector_path = getattr(plugin_context.options, 'executer_path', '/usr/obd/lib/executer') + + upgrader = Upgrader( + plugin_context=plugin_context, + search_py_script_plugin=search_py_script_plugin, + apply_param_plugin=apply_param_plugin, + upgrade_ctx=upgrade_ctx, + upgrade_repositories=upgrade_repositories, + local_home_path=local_home_path, + exector_path=exector_path, + install_repository_to_servers=install_repository_to_servers, + unuse_lib_repository=unuse_lib_repository) + if upgrader.run(): + if upgrader.route_index >= len(upgrader.route): + upgrader.call_plugin(upgrader.display_plugin, upgrader.cursor, *args, **kwargs) + plugin_context.return_true() + diff --git a/plugins/oceanbase/4.0.0.0/upgrade_check.py b/plugins/oceanbase/4.0.0.0/upgrade_check.py new file mode 100644 index 0000000..891b92f --- /dev/null +++ b/plugins/oceanbase/4.0.0.0/upgrade_check.py @@ -0,0 +1,82 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os + + +def upgrade_check(plugin_context, current_repository, route, cursor, *args, **kwargs): + + repositories = plugin_context.repositories + options = plugin_context.options + stdio = plugin_context.stdio + cluster_config = plugin_context.cluster_config + + skip_check = getattr(options, 'skip_check', False) + + can_skip = ['upgrade_checker.py', 'upgrade_post_checker.py'] + large_upgrade_need = ['upgrade_pre.py', 'upgrade_post.py'] + zones = set() + for server in cluster_config.servers: + config = cluster_config.get_server_conf_with_default(server) + zone = config['zone'] + zones.add(zone) + + if len(zones) > 2: + tenants = cursor.fetchall("""select a.tenant_name, a.tenant_id, zone_list from oceanbase.DBA_OB_TENANTS as a left join ( + select zone_list, tenant_id from oceanbase.DBA_OB_RESOURCE_POOLS ) as b + on a.tenant_id = b.tenant_id where a.tenant_name not like 'META$%'""") + if tenants is False: + return + for tenant in tenants: + zone_list = tenant.get('zone_list', '').split(';') + if len(zone_list) < 3: + stdio.error('Tenant %s does not meet rolling upgrade conditions (zone number greater than 2).' % tenant.get('tenant_name')) + return + + succeed = True + n, i = len(route), 1 + while i < n: + cant_use = False + node = route[i] + repository = repositories[i] + stdio.verbose('route %s-%s use %s. file check begin.' % (node.get('version'), node.get('release'), repository)) + script_dir = os.path.join(repository.repository_dir, 'etc/direct_upgrade') if node.get('direct_upgrade') else os.path.join(repository.repository_dir, 'etc') + if skip_check is False: + for name in can_skip: + path = os.path.join(script_dir, name) + if not os.path.isfile(path): + succeed = False + stdio.error('No such file: %s . You can use --skip-check to skip this check or --disable to ban this package' % path) + + if repository.version != current_repository.version: + for name in large_upgrade_need: + path = os.path.join(script_dir, name) + if not os.path.isfile(path): + cant_use = True + succeed = False + stdio.error('No such file: %s .' % path) + if cant_use: + stdio.error('%s cannot be used for the upgrade. You can use the --disable option to disable the image.' % repository) + i += 1 + + if succeed: + plugin_context.return_true() diff --git a/plugins/ocp-express/1.0/bootstrap.py b/plugins/ocp-express/1.0/bootstrap.py new file mode 100644 index 0000000..f73e406 --- /dev/null +++ b/plugins/ocp-express/1.0/bootstrap.py @@ -0,0 +1,57 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os + + +def bootstrap(plugin_context, cursor, start_env=None, *args, **kwargs): + stdio = plugin_context.stdio + clients = plugin_context.clients + + if not start_env: + raise Exception("start env is needed") + success = True + for server in start_env: + server_config = start_env[server] + data = { + "cluster": { + "name": server_config["cluster_name"], + "obClusterId": server_config["ob_cluster_id"], + "rootSysPassword": server_config["root_sys_password"], + "serverAddresses": server_config["server_addresses"], + }, + "agentUsername": server_config["agent_username"], + "agentPassword": server_config["agent_password"] + } + if server not in cursor or not cursor[server].init(data, stdio=stdio): + stdio.error("failed to send init request to {} ocp express".format(server)) + success = False + else: + client = clients[server] + bootstrap_flag = os.path.join(server_config['home_path'], '.bootstrapped') + client.execute_command('touch %s' % bootstrap_flag) + + if success: + return plugin_context.return_true() + else: + return plugin_context.return_false() + diff --git a/plugins/ocp-express/1.0/connect.py b/plugins/ocp-express/1.0/connect.py new file mode 100644 index 0000000..8c6bda3 --- /dev/null +++ b/plugins/ocp-express/1.0/connect.py @@ -0,0 +1,111 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import json +import requests +from requests.auth import HTTPBasicAuth + +import _errno as err + + +class OcpExpressCursor(object): + + class Response(object): + + def __init__(self, code, content): + self.code = code + self.content = content + + def __bool__(self): + return self.code == 200 + + def __init__(self, ip, port, username=None, password=None): + self.ip = ip + self.port = port + self.username = username + self.password = password + self.url_prefix = "http://{ip}:{port}/".format(ip=self.ip, port=self.port) + if self.username: + self.auth = HTTPBasicAuth(username=username, password=password) + else: + self.auth = None + + def status(self, stdio=None): + resp = self._request('GET', 'api/v1/status', stdio=stdio) + if resp: + return resp.content.get("status") == "ok" + return False + + def init(self, data, stdio=None): + return self._request("POST", 'api/v1/init', data=data, stdio=stdio) + + def _request(self, method, api, data=None, retry=5, stdio=None): + url = self.url_prefix + api + headers = {"Content-Type": "application/json"} + try: + if data is not None: + data = json.dumps(data) + stdio.verbose('send http request method: {}, url: {}, data: {}'.format(method, url, data)) + resp = requests.request(method, url, auth=self.auth, data=data, verify=False, headers=headers) + return_code = resp.status_code + content = resp.content + except Exception as e: + if retry: + retry -= 1 + return self._request(method=method, api=api, data=data, retry=retry, stdio=stdio) + stdio.exception("") + return_code = 500 + content = str(e) + if return_code != 200: + stdio.verbose("request ocp-express failed: %s" % content) + try: + content = json.loads(content.decode()) + except: + pass + return self.Response(code=return_code, content=content) + + +def connect(plugin_context, target_server=None, *args, **kwargs): + cluster_config = plugin_context.cluster_config + stdio = plugin_context.stdio + if target_server: + servers = [target_server] + stdio.start_loading('Connect to ocp-express ({})'.format(target_server)) + else: + servers = cluster_config.servers + stdio.start_loading('Connect to ocp-express') + cursors = {} + for server in servers: + config = cluster_config.get_server_conf(server) + username = 'system' + password = config['system_password'] + stdio.verbose('connect ocp-express ({}:{} by user {})'.format(server.ip, config['port'], username)) + cursor = OcpExpressCursor(ip=server.ip, port=config['port'], username=username, password=password) + if cursor.status(stdio=stdio): + cursors[server] = cursor + if not cursors: + stdio.error(err.EC_FAIL_TO_CONNECT.format(component=cluster_config.name)) + stdio.stop_loading('fail') + return plugin_context.return_false() + + stdio.stop_loading('succeed') + return plugin_context.return_true(connect=cursors, cursor=cursors) diff --git a/plugins/ocp-express/1.0/destroy.py b/plugins/ocp-express/1.0/destroy.py new file mode 100644 index 0000000..ffd77ef --- /dev/null +++ b/plugins/ocp-express/1.0/destroy.py @@ -0,0 +1,56 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import _errno as err + +global_ret = True + + +def destroy(plugin_context, *args, **kwargs): + def clean(path): + client = clients[server] + ret = client.execute_command('rm -fr %s' % path, timeout=-1) + if not ret: + global global_ret + global_ret = False + stdio.warn(err.EC_CLEAN_PATH_FAILED.format(server=server, path=path)) + else: + stdio.verbose('%s:%s cleaned' % (server, path)) + + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + global global_ret + stdio.start_loading('ocp-express work dir cleaning') + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + stdio.verbose('%s work path cleaning', server) + home_path = server_config['home_path'] + clean(home_path) + log_dir = server_config.get('log_dir') + if log_dir: + clean(log_dir) + if global_ret: + stdio.stop_loading('succeed') + plugin_context.return_true() + else: + stdio.stop_loading('fail') diff --git a/plugins/ocp-express/1.0/display.py b/plugins/ocp-express/1.0/display.py new file mode 100644 index 0000000..e4cb51f --- /dev/null +++ b/plugins/ocp-express/1.0/display.py @@ -0,0 +1,50 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +from tool import NetUtil + + +def display(plugin_context, cursor, *args, **kwargs): + cluster_config = plugin_context.cluster_config + stdio = plugin_context.stdio + servers = cluster_config.servers + results = [] + for server in servers: + api_cursor = cursor.get(server) + ip = server.ip + if ip == '127.0.0.1': + ip = NetUtil.get_host_ip() + url = 'http://{}:{}'.format(ip, api_cursor.port) + results.append({ + 'ip': ip, + 'port': api_cursor.port, + 'user': "admin", + 'password': "oceanbase", + 'url': url, + 'status': 'active' if api_cursor and api_cursor.status(stdio) else 'inactive' + }) + stdio.print_list(results, ['url', 'username', 'default_password', 'status'], lambda x: [x['url'], 'admin', 'oceanbase', x['status']], title='ocp-express') + active_result = [r for r in results if r['status'] == 'active'] + info_dict = active_result[0] if len(active_result) > 0 else None + if info_dict is not None: + info_dict['type'] = 'web' + return plugin_context.return_true(info=info_dict) diff --git a/plugins/ocp-express/1.0/file_map.yaml b/plugins/ocp-express/1.0/file_map.yaml new file mode 100644 index 0000000..41844a1 --- /dev/null +++ b/plugins/ocp-express/1.0/file_map.yaml @@ -0,0 +1,6 @@ +- src_path: ./home/admin/ocp-express/lib/ocp-express-server.jar + target_path: lib/ocp-express-server.jar + type: file +- src_path: ./home/admin/ocp-express/conf + target_path: conf + type: dir \ No newline at end of file diff --git a/plugins/ocp-express/1.0/generate_config.py b/plugins/ocp-express/1.0/generate_config.py new file mode 100644 index 0000000..f43e51a --- /dev/null +++ b/plugins/ocp-express/1.0/generate_config.py @@ -0,0 +1,64 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + + +def generate_config(plugin_context, auto_depend=False, generate_config_mini=False, return_generate_keys=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=['memory_size', 'log_dir', 'logging_file_max_history']) + + cluster_config = plugin_context.cluster_config + stdio = plugin_context.stdio + depend_comps = [['obagent'], ['oceanbase', 'oceanbase-ce'], ['obproxy', 'obproxy-ce']] + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) + stdio.start_loading('Generate ocp express configuration') + min_memory_size = '752M' + + if auto_depend: + for comps in depend_comps: + for comp in comps: + if cluster_config.add_depend_component(comp): + break + global_config = cluster_config.get_global_conf() + if generate_config_mini: + if 'memory_size' not in global_config: + cluster_config.update_global_conf('memory_size', min_memory_size) + + auto_set_memory = False + if 'memory_size' not in global_config: + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + if 'memory_size' not in server_config: + auto_set_memory = True + if auto_set_memory: + observer_num = 0 + for comp in ['oceanbase', 'oceanbase-ce']: + if comp in cluster_config.depends: + observer_num = len(cluster_config.get_depend_servers(comp)) + if not observer_num: + stdio.warn('The component oceanbase/oceanbase-ce is not in the depends, the memory size cannot be calculated, and a fixed value of {} is used'.format(min_memory_size)) + cluster_config.update_global_conf('memory_size', min_memory_size) + else: + cluster_config.update_global_conf('memory_size', '%dM' % (512 + (observer_num + 3) * 60)) + + stdio.stop_loading('succeed') + plugin_context.return_true() \ No newline at end of file diff --git a/plugins/ocp-express/1.0/init.py b/plugins/ocp-express/1.0/init.py new file mode 100644 index 0000000..0f3757a --- /dev/null +++ b/plugins/ocp-express/1.0/init.py @@ -0,0 +1,124 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os.path + +import _errno as err + + +def _clean(server, client, path, stdio=None): + ret = client.execute_command('rm -fr %s' % path, timeout=-1) + if not ret: + stdio.warn(err.EC_CLEAN_PATH_FAILED.format(server=server, path=path)) + return False + else: + stdio.verbose('%s:%s cleaned' % (server, path)) + return True + + +def init(plugin_context, *args, **kwargs): + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + + global_ret = True + force = getattr(plugin_context.options, 'force', False) + clean = getattr(plugin_context.options, 'clean', False) + + stdio.start_loading('Initializes ocp-express work home') + servers_dirs = {} + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + ip = server.ip + if ip not in servers_dirs: + servers_dirs[ip] = {} + dirs = servers_dirs[ip] + home_path = server_config['home_path'] + keys = ['home_path', 'log_dir'] + for key in keys: + if key not in server_config: + continue + path = server_config[key] + if path in dirs: + global_ret = False + stdio.error(err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key'])) + continue + dirs[path] = { + 'server': server, + 'key': key, + } + need_clean = force + if clean and not force: + if client.execute_command( + 'bash -c \'if [[ "$(ls -d {0} 2>/dev/null)" != "" && ! -O {0} ]]; then exit 0; else exit 1; fi\''.format( + home_path)): + owner = client.execute_command("ls -ld %s | awk '{print $3}'" % home_path).stdout.strip() + global_ret = False + err_msg = ' {} is not empty, and the owner is {}'.format(home_path, owner) + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=err_msg)) + continue + need_clean = True + if need_clean: + port = server_config['port'] + client.execute_command("pkill -9 -u `whoami` -f 'java -jar {home_path}/lib/ocp-express-server.jar --port {port}'".format(home_path=home_path, port=port)) + if not _clean(server, client, home_path, stdio=stdio): + global_ret = False + continue + if client.execute_command('mkdir -p %s' % home_path): + ret = client.execute_command('ls %s' % (home_path)) + if not ret or ret.stdout.strip(): + global_ret = False + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=home_path))) + continue + else: + global_ret = False + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=err.InitDirFailedErrorMessage.CREATE_FAILED.format(path=home_path))) + continue + if not client.execute_command("bash -c 'mkdir -p %s/{run,bin,lib}'" % home_path): + global_ret = False + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='home path', msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=home_path))) + if 'log_dir' in server_config: + log_dir = server_config['log_dir'] + if client.execute_command('mkdir -p %s' % log_dir): + ret = client.execute_command('ls %s' % log_dir) + if not ret or ret.stdout.strip(): + global_ret = False + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='log dir', msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=log_dir))) + continue + else: + global_ret = False + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='log dir', msg=err.InitDirFailedErrorMessage.CREATE_FAILED.format(path=log_dir))) + continue + else: + log_dir = os.path.join(home_path, 'log') + if not client.execute_command('mkdir -p %s' % log_dir): + global_ret = False + stdio.error(err.EC_FAIL_TO_INIT_PATH.format(server=server, key='log dir', msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=log_dir))) + continue + link_path = os.path.join(home_path, 'log') + client.execute_command("if [ ! '%s' -ef '%s' ]; then ln -sf %s %s; fi" % (log_dir, link_path, log_dir, link_path)) + if global_ret: + stdio.stop_loading('succeed') + plugin_context.return_true() + else: + stdio.stop_loading('fail') diff --git a/plugins/ocp-express/1.0/parameter.yaml b/plugins/ocp-express/1.0/parameter.yaml new file mode 100644 index 0000000..5006a19 --- /dev/null +++ b/plugins/ocp-express/1.0/parameter.yaml @@ -0,0 +1,348 @@ +- name: home_path + name_local: 工作目录 + require: true + essential: true + type: STRING + need_redeploy: true + description_en: the directory for the work data + description_local: OCP express server工作目录 +- name: log_dir + name_local: 日志目录 + type: STRING + require: false + essential: true + need_redeploy: true + description_en: The directory for logging file. The default value is $home_path/log. + description_local: OCP express server日志目录, 默认为工作目录下的log +- name: java_bin + name_local: java路径 + type: STRING + require: true + essential: true + default: java + need_restart: true + description_en: The path of java binary + description_local: OCP express 使用的java可执行文件的路径 +- name: memory_size + name_local: 进程内存 + require: true + essential: true + type: CAPACITY + min_value: 512M + need_restart: true + description_en: the memroy size of ocp express server. Please enter an capacity, such as 2G + description_local: OCP express server进程内存大小。请输入带容量带单位的整数,如2G +- name: logging_file_max_size + name_local: 单个日志文件大小 + type: STRING + require: false + essential: true + default: 100MB + need_restart: true + description_local: 单个日志文件大小 + description_en: When logging_file_name is configured, specify the log file size through this configuration +- name: logging_file_total_size_cap + name_local: 日志总大小 + type: STRING + require: true + essential: true + default: 1GB + need_restart: true + description_local: 日志文件总大小 + description_en: When logging_file_name is configured, specify the total log file size through this configuration +- name: port + name_local: 端口 + require: true + essential: true + type: INT + default: 8180 + need_restart: true + description_en: the port of ocp server. + description_local: OCP server使用的端口 +- name: system_password + require: true + type: STRING + default: oceanbase + need_restart: true + description_en: The password name for ocp server + description_local: OCP server中system用户的密码 +- name: jdbc_url + require: false + type: STRING + need_redeploy: true + description_en: The jdbc connection url for ocp meta db + description_local: OCP使用的元数据库的jdbc连接串 +- name: jdbc_username + require: false + type: STRING + need_redeploy: true + description_en: The username name for ocp meta db + description_local: OCP使用的元数据库的用户名 +- name: jdbc_password + require: false + type: STRING + default: + need_redeploy: true + description_en: The password name for ocp meta db + description_local: OCP使用的元数据库的密码 +# bootstrap parameters +- name: cluster_name + requrire: false + type: STRING + default: obcluster + need_restart: true + description_en: The cluster name of observer + description_local: Oceanbase数据库的集群名称 +- name: ob_cluster_id + require: false + type: INT + min_value: 1 + max_value: 4294901759 + need_restart: true + description_en: ID of the cluster + description_local: OceanBase集群ID +- name: root_sys_password + require: false + type: STRING + default: + need_restart: true + description_en: password of observer root user + description_local: sys租户root用户的密码 +- name: server_addresses + require: false + type: LIST + need_restart: true + description_en: the servers info for oceanbase cluster + description_local: Oceanbase集群的节点信息 +- name: 'session_timeout' + type: 'STRING' + require: false + need_restart: true + description_local: '登陆会话/Session超时的时间,默认是30m,最少60s。如果不加后缀单位,则默认是秒。重启生效。' + description_en: 'Session timeout interval, default is 30m, at least 60s. If the suffix unit is not added, the default is seconds. Restart OCP to take effect.' +- name: 'login_encrypt_enabled' + type: 'STRING' + require: false + need_restart: true + description_local: '登录信息是否开启加密传输,默认开启,重启生效' + description_en: 'Switch to enable encrypted transmission of login information, enabled by default. Restart OCP to take effect.' +- name: 'login_encrypt_public_key' + type: 'STRING' + require: false + need_restart: true + description_local: '加密登录信息的公钥,建议部署后修改此配置,修改后重启生效' + description_en: 'The public key for login encryption, It is recommended to modify this configuration after deployment. Restart OCP to take effect.' +- name: 'login_encrypt_private_key' + type: 'STRING' + require: false + need_restart: true + description_local: '加密登录信息的私钥,建议部署后修改此配置,修改后重启生效' + description_en: 'The private key for encryption. It is recommended to modify this configuration after deployment. Restart OCP to take effect.' +- name: 'enable_basic_auth' + type: 'STRING' + require: false + need_restart: true + description_local: '是否启用Basic Auth登陆模式,通常供程序和SDK等客户端场景使用,默认true。本配置与ocp.iam.auth可同时开启。重启生效。' + description_en: 'Whether to enable Basic Authentication, usually for client programs and SDKs to call server APIs. The default is true. This configuration and ocp.iam.auth can be enabled together. Restart OCP to take effect.' +- name: 'enable_csrf' + type: 'STRING' + require: false + need_restart: true + description_local: '是否启用CSRF跨站点请求伪造安全保护,通常基于网页登陆的方式都推荐要启用,默认true。重启生效。' + description_en: 'Whether to enable CSRF cross-site request forgery security protection. It is recommended to enable it, the default is true. Restart OCP to take effect.' +- name: 'vault_key' + type: 'STRING' + require: false + need_restart: true + description_local: '密码箱加密密钥' + description_en: 'vault secret key' +- name: 'druid_name' + type: 'STRING' + require: false + need_restart: true + description_local: 'metadb的druid连接池名称。重启生效' + description_en: 'metadb druid connection pool name. Restart to take effect' +- name: 'druid_init_size' + type: 'STRING' + require: false + need_restart: true + description_local: '初始化时建立物理连接的个数。重启生效' + description_en: 'The number of physical connections established during initialization. Restart to take effect' +- name: 'druid_min_idle' + type: 'STRING' + require: false + need_restart: true + description_local: '最小连接池数量。重启生效' + description_en: 'Minimum number of connections. Restart to take effect' +- name: 'druid_max_active' + type: 'STRING' + require: false + need_restart: true + description_local: '最大连接池数量。重启生效' + description_en: 'The maximum number of connections. Restart to take effect' +- name: 'druid_test_while_idle' + type: 'STRING' + require: false + need_restart: true + description_local: '建议配置为true,不影响性能,并且保证安全性。申请连接的时候检测。重启生效' + description_en: 'It is recommended to set it to true, which will not affect performance and ensure safety. Detect when applying for connection. Restart to take effect' +- name: 'druid_validation_query' + type: 'STRING' + require: false + need_restart: true + description_local: '用来检测连接是否有效的sql。重启生效' + description_en: 'SQL used to detect whether the connection is valid. Restart to take effect' +- name: 'druid_max_wait' + type: 'STRING' + require: false + need_restart: true + description_local: '获取连接时最大等待时间,单位毫秒。重启生效' + description_en: 'Maximum waiting time when getting a connection, in milliseconds. Restart to take effect' +- name: 'druid_keep_alive' + type: 'STRING' + require: false + need_restart: true + description_local: '连接池中的minIdle数量以内的连接,空闲时间超过minEvictableIdleTimeMillis(缺省值1800秒),则会执行keepAlive操作。重启生效' + description_en: 'For connections within the number of minIdle in the connection pool, if the idle time exceeds minEvictableIdleTimeMillis (the default value is 1800 seconds), the keepAlive operation will be performed. Restart to take effect' +- name: 'logging_pattern_console' + type: 'STRING' + require: false + need_restart: true + description_local: '用于控制台输出的日志格式' + description_en: 'Log format for console output' +- name: 'logging_pattern_file' + type: 'STRING' + require: false + need_restart: true + description_local: '用于文件输出的日志格式' + description_en: 'Log format used for file output' +- name: 'logging_file_clean_when_start' + type: 'STRING' + require: false + need_restart: true + description_local: '启动时删除压缩的日志文件' + description_en: 'Clean the archive log files on startup' +- name: 'logging_file_max_history' + name_local: 日志保留天数 + type: INT + require: false + essential: true + need_restart: true + min_value: 1 + max_value: 2147483647 + description_local: '最多保留的归档日志文件的天数,默认不限制' + description_en: 'When logging.file is configured, set the maximum of retention days the log archive log files to keep. The default value is unlimited' +- name: 'default_timezone' + type: 'STRING' + require: false + need_restart: true + description_local: '系统默认时区,若不设置则使用 system default time zone,重启生效' + description_en: 'System default time zone, if not set, use system default time zone, restart to take effect' +- name: 'default_lang' + type: 'STRING' + require: false + need_restart: true + description_local: '系统默认语言(非前端语言设置),若不设置则使用 zh-CN,重启生效' + description_en: 'System default language (non-front-end language setting), if not set, use zh-CN, restart to take effect' +- name: 'ocp.idempotent.client-token.expire.time' + type: 'STRING' + require: false + need_restart: true + description_local: '幂等请求token的缓存过期时间,默认14d' + description_en: 'Expire time of idempotent client token, the default is 14d' +- name: 'obsdk_sql_query_limit' + type: 'STRING' + require: false + need_restart: true + description_local: '基于 obsdk 的采集查询,SQL 查询行数限制,默认 10000' + description_en: 'Sql query row limit for obsdk based collect' +- name: 'exporter_inactive_threshold' + type: 'INT' + require: false + need_restart: true + description_local: 'exporter地址判定为失效的连续不可用时间(秒)' + description_en: 'consecutive failure time of exporter address that is regarded as inactive (seconds)' +- name: 'ocp.monitor.host.exporters' + type: 'STRING' + require: false + need_restart: true + description_local: '主机监控exporter' + description_en: 'exporters of ocp host' +- name: 'ocp.monitor.ob.exporters' + type: 'STRING' + require: false + need_restart: true + description_local: 'OB监控exporter' + description_en: 'exporters of ob' +- name: 'monitor_collect_interval' + type: 'STRING' + require: false + need_restart: true + description_local: '秒级别监控采集间隔,默认 1s,支持配置选项是 1s, 5s, 10s, 15s' + description_en: 'The parameter determines the second-level monitoring and collection interval. The supported configuration options are 1s, 5s, 10s, 15s. Default value is 1s' +- name: 'montior_retention_days' + type: 'STRING' + require: false + need_restart: true + description_local: '监控数据保存天数,key 是监控数据的表名,value 是保存的天数,修改后重启生效.' + description_en: 'Retention days for monitor data, key is table name for monitor data, value is the retention days. Restart to take effect.' +- name: 'obsdk_cache_size' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk连接器池容量,取值范围10~200,默认值100' + description_en: 'Obsdk connector holder capacity, value range 10~200, default value 100' +- name: 'obsdk_max_idle' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk空闲连接器的过期时间,单位秒,取值范围300~18000,默认值3600' + description_en: 'The expiration time of the obsdk idle connector, in seconds, the value range is 300~18000, and the default value is 3600' +- name: 'obsdk_cleanup_period' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk过期连接器的清理周期,单位秒,取值范围30~1800,默认值300' + description_en: 'The interval for obsdk to clean up the expired connector, in seconds, the value range is 30~1800, and the default value is 300' +- name: 'obsdk_print_sql' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk中sql打印开关,默认开启' + description_en: 'Sql print switch in obsdk, enabled by default' +- name: 'obsdk_slow_query_threshold' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk中慢查询日志阈值,单位毫秒,默认值 1000' + description_en: 'Slow query log threshold in obsdk, in milliseconds, the default value is 1000' +- name: 'obsdk_init_timeout' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk中连接器初始化超时时间,单位毫秒,默认值 3000' + description_en: 'Timeout of connector initialization in obsdk, in milliseconds, the default value is 5000' +- name: 'obsdk_init_core_size' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk中连接器初始化的线程个数' + description_en: 'The thread count of connector initialization in obsdk, the default value is 16' +- name: 'obsdk_global_timeout' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk中运维命令全局超时时间,单位毫秒,取值范围10000~7200000,默认值 300000' + description_en: 'Global timeout of operation in obsdk, in milliseconds, the value range is 10000~7200000, and the default value is 300000' +- name: 'obsdk_connect_timeout' + type: 'STRING' + require: false + need_restart: true + description_local: 'obsdk建立Socket连接的超时时间,单位:ms' + description_en: 'The timeout period for obsdk to connect to ob, unit: ms' +- name: 'obsdk_read_timeout' + type: 'STRING' + require: false + need_restart: true + description_local: 'Obsdk的Socket读取数据的超时时间,单位:ms' + description_en: 'Obsdk socket read data timeout time, unit: ms' \ No newline at end of file diff --git a/plugins/ocp-express/1.0/reload.py b/plugins/ocp-express/1.0/reload.py new file mode 100644 index 0000000..0712894 --- /dev/null +++ b/plugins/ocp-express/1.0/reload.py @@ -0,0 +1,27 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + + +def reload(plugin_context, *args, **kwargs): + stdio = plugin_context.stdio + stdio.verbose('Nothing to do for ocp express reload') + return plugin_context.return_true() diff --git a/plugins/ocp-express/1.0/restart.py b/plugins/ocp-express/1.0/restart.py new file mode 100644 index 0000000..9de2d40 --- /dev/null +++ b/plugins/ocp-express/1.0/restart.py @@ -0,0 +1,157 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os + + +class Restart(object): + + def __init__(self, plugin_context, local_home_path, start_check_plugin, start_plugin, reload_plugin, stop_plugin, connect_plugin, + display_plugin, repository, new_cluster_config=None, new_clients=None, bootstrap_plugin=None, + repository_dir_map=None): + self.local_home_path = local_home_path + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + + self.components = plugin_context.components + self.clients = plugin_context.clients + self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode + self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context + self.repository = repository + self.start_check_plugin = start_check_plugin + self.start_plugin = start_plugin + self.reload_plugin = reload_plugin + self.connect_plugin = connect_plugin + self.display_plugin = display_plugin + self.bootstrap_plugin = bootstrap_plugin + self.stop_plugin = stop_plugin + self.new_clients = new_clients + self.new_cluster_config = new_cluster_config + self.sub_io = self.stdio.sub_io() + self.dbs = None + self.cursors = None + self.repository_dir_map = repository_dir_map + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) + + def connect(self, cluster_config): + if self.cursors is None: + self.sub_io.start_loading('Connect to ocp express') + ret = self.call_plugin(self.connect_plugin, cluster_config=cluster_config) + if not ret: + self.sub_io.stop_loading('fail') + return False + self.sub_io.stop_loading('succeed') + self.cursors = ret.get_return('cursor') + self.dbs = ret.get_return('connect') + return True + + def dir_read_check(self, client, path): + if not client.execute_command('cd %s' % path): + dirpath, name = os.path.split(path) + return self.dir_read_check(client, dirpath) and client.execute_command('sudo chmod +1 %s' % path) + return True + + def restart(self): + clients = self.clients + if not self.call_plugin(self.stop_plugin, clients=clients): + self.stdio.stop_loading('stop_loading', 'fail') + return False + + if self.new_clients: + self.stdio.verbose('use new clients') + for server in self.cluster_config.servers: + new_client = self.new_clients[server] + server_config = self.cluster_config.get_server_conf(server) + for key in ['home_path', 'data_dir']: + if key in server_config: + path = server_config[key] + if not new_client.execute_command('sudo chown -R %s: %s' % (new_client.config.username, path)): + self.stdio.stop_loading('stop_loading', 'fail') + return False + self.dir_read_check(new_client, path) + clients = self.new_clients + + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + + need_bootstrap = self.bootstrap_plugin is not None + if not self.call_plugin(self.start_check_plugin, clients=clients, cluster_config=cluster_config): + self.stdio.stop_loading('stop_loading', 'fail') + return False + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, need_bootstrap=need_bootstrap, repository_dir_map=self.repository_dir_map): + self.rollback() + self.stdio.stop_loading('stop_loading', 'fail') + return False + + if self.connect(cluster_config): + if self.bootstrap_plugin: + self.call_plugin(self.bootstrap_plugin, cursor=self.cursors) + return self.call_plugin(self.display_plugin, cursor=self.cursors) + return False + + def rollback(self): + if self.new_clients: + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.new_clients, cluster_config=cluster_config) + for server in self.cluster_config.servers: + client = self.clients[server] + new_client = self.new_clients[server] + server_config = self.cluster_config.get_server_conf(server) + home_path = server_config['home_path'] + new_client.execute_command('sudo chown -R %s: %s' % (client.config.username, home_path)) + + +def restart(plugin_context, local_home_path, start_check_plugin, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, + new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, repository_dir_map=None, *args, + **kwargs): + repository = kwargs.get('repository') + task = Restart(plugin_context=plugin_context, local_home_path=local_home_path, start_check_plugin=start_check_plugin, start_plugin=start_plugin, bootstrap_plugin=bootstrap_plugin, reload_plugin=reload_plugin, stop_plugin=stop_plugin, connect_plugin=connect_plugin, + display_plugin=display_plugin, repository=repository, new_cluster_config=new_cluster_config, new_clients=new_clients, repository_dir_map=repository_dir_map) + call = task.rollback if rollback else task.restart + if call(): + plugin_context.return_true() diff --git a/plugins/ocp-express/1.0/start.py b/plugins/ocp-express/1.0/start.py new file mode 100644 index 0000000..b2d245c --- /dev/null +++ b/plugins/ocp-express/1.0/start.py @@ -0,0 +1,453 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os +import re +import time +import base64 +import sys +from copy import deepcopy + +from tool import FileUtil, YamlLoader + +from Crypto import Random +from Crypto.Hash import SHA +from Crypto.PublicKey import RSA +from Crypto.Signature import PKCS1_v1_5 as PKCS1_signature +from Crypto.Cipher import PKCS1_OAEP as PKCS1_cipher + +PRI_KEY_FILE = '.ocp-express' +PUB_KEY_FILE = '.ocp-express.pub' + + +if sys.version_info.major == 2: + import MySQLdb as mysql +else: + import pymysql as mysql +from _stdio import SafeStdio + + +def parse_size(size): + _bytes = 0 + if not isinstance(size, str) or size.isdigit(): + _bytes = int(size) + else: + units = {"B": 1, "K": 1<<10, "M": 1<<20, "G": 1<<30, "T": 1<<40} + match = re.match(r'(0|[1-9][0-9]*)\s*([B,K,M,G,T])', size.upper()) + _bytes = int(match.group(1)) * units[match.group(2)] + return _bytes + + +def format_size(size, precision=1): + units = ['B', 'K', 'M', 'G'] + units_num = len(units) - 1 + idx = 0 + if precision: + div = 1024.0 + format = '%.' + str(precision) + 'f%s' + limit = 1024 + else: + div = 1024 + limit = 1024 + format = '%d%s' + while idx < units_num and size >= limit: + size /= div + idx += 1 + return format % (size, units[idx]) + + +class Cursor(SafeStdio): + + def __init__(self, ip, port, user='root', tenant='sys', password='', database=None, stdio=None): + self.stdio = stdio + self.ip = ip + self.port = port + self._user = user + self.tenant = tenant + self.password = password + self.database = database + self.cursor = None + self.db = None + self._connect() + + @property + def user(self): + if "@" in self._user: + return self._user + if self.tenant: + return "{}@{}".format(self._user, self.tenant) + else: + return self._user + + def _connect(self): + self.stdio.verbose('connect %s -P%s -u%s -p%s' % (self.ip, self.port, self.user, self.password)) + if sys.version_info.major == 2: + self.db = mysql.connect(host=self.ip, user=self.user, port=int(self.port), passwd=str(self.password), database=self.database) + self.cursor = self.db.cursor(cursorclass=mysql.cursors.DictCursor) + else: + self.db = mysql.connect(host=self.ip, user=self.user, port=int(self.port), password=str(self.password), database=self.database, + cursorclass=mysql.cursors.DictCursor) + self.cursor = self.db.cursor() + + +def generate_key(client, key_dir, stdio): + rsa = RSA.generate(1024) + private_key = rsa + public_key = rsa.publickey() + client.write_file(private_key.exportKey(pkcs=8), os.path.join(key_dir, PRI_KEY_FILE), mode='wb', stdio=stdio) + client.write_file(public_key.exportKey(pkcs=8), os.path.join(key_dir, PUB_KEY_FILE), mode='wb', stdio=stdio) + return private_key, public_key + + +def get_key(client, key_dir, stdio): + private_key_file = os.path.join(key_dir, PRI_KEY_FILE) + ret = client.execute_command("cat {}".format(private_key_file)) + if not ret: + return generate_key(client, key_dir, stdio) + private_key = RSA.importKey(ret.stdout.strip()) + public_key_file = os.path.join(key_dir, PUB_KEY_FILE) + ret = client.execute_command("cat {}".format(public_key_file)) + if not ret: + return generate_key(client, key_dir, stdio) + public_key = RSA.importKey(ret.stdout.strip()) + return private_key, public_key + + +def get_plain_public_key(public_key): + if isinstance(public_key, RSA.RsaKey): + public_key = public_key.exportKey(pkcs=8).decode() + elif isinstance(public_key, bytes): + public_key = public_key.decode() + public_key = public_key.replace("-----BEGIN PRIVATE KEY-----", "").replace("-----END PRIVATE KEY-----", "").replace("-----BEGIN PUBLIC KEY-----", "").replace("-----END PUBLIC KEY-----", "").replace("\n", "") + return public_key + + +def rsa_private_sign(passwd, private_key): + signer = PKCS1_cipher.new(private_key) + sign = signer.encrypt(passwd.encode("utf-8")) + # digest = SHA.new() + # digest.update(passwd.encode("utf8")) + # sign = signer.sign(digest) + signature = base64.b64encode(sign) + signature = signature.decode('utf-8') + return signature + + +def get_port_socket_inode(client, port, stdio): + port = hex(port)[2:].zfill(4).upper() + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + res = client.execute_command(cmd) + if not res or not res.stdout.strip(): + return False + stdio.verbose(res.stdout) + return res.stdout.strip().split('\n') + + +def confirm_port(client, pid, port, stdio): + socket_inodes = get_port_socket_inode(client, port, stdio) + if not socket_inodes: + return False + ret = client.execute_command("ls -l /proc/%s/fd/ |grep -E 'socket:\[(%s)\]'" % (pid, '|'.join(socket_inodes))) + if ret and ret.stdout.strip(): + return True + return False + + +def get_missing_required_parameters(parameters): + results = [] + for key in ["jdbc_url", "jdbc_password", "jdbc_username", "cluster_name", "ob_cluster_id", "root_sys_password", + "server_addresses", "agent_username", "agent_password"]: + if parameters.get(key) is None: + results.append(key) + return results + + +def prepare_parameters(cluster_config, stdio): + # depends config + env = {} + depend_observer = False + depend_info = {} + ob_servers_conf = {} + root_servers = [] + for comp in ["oceanbase", "oceanbase-ce"]: + ob_zones = {} + if comp in cluster_config.depends: + depend_observer = True + observer_globals = cluster_config.get_depend_config(comp) + ocp_meta_keys = [ + "ocp_meta_tenant", "ocp_meta_db", "ocp_meta_username", "ocp_meta_password", "appname", "cluster_id", "root_password" + ] + for key in ocp_meta_keys: + value = observer_globals.get(key) + if value is not None: + depend_info[key] = value + ob_servers = cluster_config.get_depend_servers(comp) + for ob_server in ob_servers: + ob_servers_conf[ob_server] = ob_server_conf = cluster_config.get_depend_config(comp, ob_server) + if 'server_ip' not in depend_info: + depend_info['server_ip'] = ob_server.ip + depend_info['mysql_port'] = ob_server_conf['mysql_port'] + zone = ob_server_conf['zone'] + if zone not in ob_zones: + ob_zones[zone] = ob_server + root_servers = ob_zones.values() + break + for comp in ['obproxy', 'obproxy-ce']: + if comp in cluster_config.depends: + obproxy_servers = cluster_config.get_depend_servers(comp) + obproxy_server = obproxy_servers[0] + obproxy_server_config = cluster_config.get_depend_config(comp, obproxy_server) + depend_info['server_ip'] = obproxy_server.ip + depend_info['mysql_port'] = obproxy_server_config['listen_port'] + break + if 'obagent' in cluster_config.depends: + obagent_servers = cluster_config.get_depend_servers('obagent') + server_addresses = [] + for obagent_server in obagent_servers: + obagent_server_config_without_default = cluster_config.get_depend_config('obagent', obagent_server, with_default=False) + obagent_server_config = cluster_config.get_depend_config('obagent', obagent_server) + username = obagent_server_config['http_basic_auth_user'] + password = obagent_server_config['http_basic_auth_password'] + if 'obagent_username' not in depend_info: + depend_info['obagent_username'] = username + elif depend_info['obagent_username'] != username: + stdio.error('The http basic auth of obagent is inconsistent') + return + if 'obagent_password' not in depend_info: + depend_info['obagent_password'] = password + elif depend_info['obagent_password'] != password: + stdio.error('The http basic auth of obagent is inconsistent') + return + if obagent_server_config_without_default.get('sql_port'): + sql_port = obagent_server_config['sql_port'] + elif ob_servers_conf.get(obagent_server) and ob_servers_conf[obagent_server].get('mysql_port'): + sql_port = ob_servers_conf[obagent_server]['mysql_port'] + else: + continue + if obagent_server_config_without_default.get('rpc_port'): + svr_port = obagent_server_config['rpc_port'] + elif ob_servers_conf.get(obagent_server) and ob_servers_conf[obagent_server].get('rpc_port'): + svr_port = ob_servers_conf[obagent_server]['rpc_port'] + else: + continue + server_addresses.append({ + "address": obagent_server.ip, + "svrPort": svr_port, + "sqlPort": sql_port, + "withRootServer": obagent_server in root_servers, + "agentMgrPort": obagent_server_config.get('mgragent_http_port', 0), + "agentMonPort": obagent_server_config.get('monagent_http_port', 0) + }) + depend_info['server_addresses'] = server_addresses + + for server in cluster_config.servers: + server_config = deepcopy(cluster_config.get_server_conf_with_default(server)) + original_server_config = cluster_config.get_original_server_conf(server) + missed_keys = get_missing_required_parameters(original_server_config) + if missed_keys: + if 'jdbc_url' in missed_keys and depend_observer: + server_config['jdbc_url'] = 'jdbc:oceanbase://{}:{}/{}'.format(depend_info['server_ip'], depend_info['mysql_port'], depend_info['ocp_meta_db']) + if 'jdbc_username' in missed_keys and depend_observer: + server_config['jdbc_username'] = "{}@{}".format(depend_info['ocp_meta_username'], + depend_info.get('ocp_meta_tenant', {}).get("tenant_name")) + depends_key_maps = { + "jdbc_password": "ocp_meta_password", + "cluster_name": "appname", + "ob_cluster_id": "cluster_id", + "root_sys_password": "root_password", + "agent_username": "obagent_username", + "agent_password": "obagent_password", + "server_addresses": "server_addresses" + } + for key in depends_key_maps: + if key in missed_keys: + if depend_info.get(depends_key_maps[key]) is not None: + server_config[key] = depend_info[depends_key_maps[key]] + env[server] = server_config + return env + + +def start(plugin_context, start_env=None, *args, **kwargs): + cluster_config = plugin_context.cluster_config + options = plugin_context.options + clients = plugin_context.clients + stdio = plugin_context.stdio + + if not start_env: + start_env = prepare_parameters(cluster_config, stdio) + if not start_env: + return plugin_context.return_false() + + + + exclude_keys = ["home_path", "port", "jdbc_url", "jdbc_username", "jdbc_password", "cluster_name", "ob_cluster_id", + "root_sys_password", "server_addresses", "agent_username", "agent_password", "system_password", "memory_size"] + + repository_dir = None + for repository in plugin_context.repositories: + if repository.name == cluster_config.name: + repository_dir = repository.repository_dir + break + with FileUtil.open(os.path.join(repository_dir, 'conf/ocp-express-config-mapper.yaml')) as f: + data = YamlLoader(stdio=stdio).load(f) + config_mapper = data.get('config_mapper', {}) + server_pid = {} + success = True + stdio.start_loading("Start ocp-express") + for server in cluster_config.servers: + client = clients[server] + server_config = start_env[server] + home_path = server_config['home_path'] + jdbc_url = server_config['jdbc_url'] + jdbc_username = server_config['jdbc_username'] + jdbc_password = server_config['jdbc_password'] + system_password = server_config["system_password"] + port = server_config['port'] + pid_path = os.path.join(home_path, 'run/ocp-express.pid') + pids = client.execute_command("cat %s" % pid_path).stdout.strip() + bootstrap_flag = os.path.join(home_path, '.bootstrapped') + if pids and all([client.execute_command('ls /proc/%s' % pid) for pid in pids.split('\n')]): + server_pid[server] = pids + continue + if getattr(options, 'without_parameter', False) and client.execute_command('ls %s' % bootstrap_flag): + use_parameter = False + else: + use_parameter = True + # check meta db connect before start + matched = re.match(r"^jdbc:\S+://(\S+?)(|:\d+)/(\S+)", jdbc_url) + if matched: + ip = matched.group(1) + sql_port = matched.group(2)[1:] + database = matched.group(3) + connected = False + retries = 10 + while not connected and retries: + retries -= 1 + try: + Cursor(ip=ip, port=sql_port, user=jdbc_username, password=jdbc_password, database=database, stdio=stdio) + connected = True + except: + time.sleep(1) + if not connected: + success = False + stdio.error("{}: failed to connect meta db".format(server)) + continue + else: + stdio.verbose('unmatched jdbc url, skip meta db connection check') + if server_config.get('encrypt_password', False): + private_key, public_key = get_key(client, os.path.join(home_path, 'conf'), stdio) + public_key_str = get_plain_public_key(public_key) + jdbc_password = rsa_private_sign(jdbc_password, private_key) + system_password = rsa_private_sign(system_password, private_key) + else: + public_key_str = "" + memory_size = server_config['memory_size'] + jvm_memory_option = "-Xms{0} -Xmx{0}".format(format_size(parse_size(memory_size) * 0.5, 0).lower()) + extra_options = { + "ocp.iam.encrypted-system-password": system_password + } + extra_options_str = ' '.join(["-D{}={}".format(k, v) for k, v in extra_options.items()]) + java_bin = server_config['java_bin'] + cmd = '{java_bin} -jar {jvm_memory_option} -DJDBC_URL={jdbc_url} -DJDBC_USERNAME={jdbc_username} -DJDBC_PASSWORD={jdbc_password} ' \ + '-DPUBLIC_KEY={public_key} {extra_options_str} {home_path}/lib/ocp-express-server.jar --port={port}'.format( + java_bin=java_bin, + home_path=home_path, + port=port, + jdbc_url=jdbc_url, + jdbc_username=jdbc_username, + jdbc_password=jdbc_password, + public_key=public_key_str, + jvm_memory_option=jvm_memory_option, + extra_options_str=extra_options_str, + ) + if "log_dir" not in server_config: + log_dir = os.path.join(home_path, 'log') + else: + log_dir = server_config["log_dir"] + server_config["logging_file_name"] = os.path.join(log_dir, 'ocp-express.log') + if use_parameter: + cmd += ' --bootstrap --progress-log={}'.format(os.path.join(log_dir, 'bootstrap.log')) + for key in server_config: + if key not in exclude_keys and key in config_mapper: + cmd += ' --with-property={}:{}'.format(config_mapper[key], server_config[key]) + client.execute_command("cd {}; bash -c '{} > /dev/null 2>&1 &'".format(home_path, cmd)) + ret = client.execute_command("ps -aux | grep '%s' | grep -v grep | awk '{print $2}' " % cmd) + if ret: + server_pid[server] = ret.stdout.strip() + if not server_pid[server]: + stdio.error("failed to start {} ocp express".format(server)) + success = False + continue + client.write_file(server_pid[server], os.path.join(home_path, 'run/ocp-express.pid')) + if success: + stdio.stop_loading('succeed') + else: + stdio.stop_loading('fail') + return plugin_context.return_false() + + stdio.start_loading("ocp-express program health check") + failed = [] + servers = cluster_config.servers + count = 60 + while servers and count: + count -= 1 + tmp_servers = [] + for server in servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + stdio.verbose('%s program health check' % server) + pids_stat = {} + for pid in server_pid[server].split("\n"): + pids_stat[pid] = None + if not client.execute_command('ls /proc/{}'.format(pid)): + pids_stat[pid] = False + continue + confirm = confirm_port(client, pid, int(server_config["port"]), stdio) + if confirm: + pids_stat[pid] = True + break + if any(pids_stat.values()): + for pid in pids_stat: + if pids_stat[pid]: + stdio.verbose('%s ocp-express[pid: %s] started', server, pid) + continue + if all([stat is False for stat in pids_stat.values()]): + failed.append('failed to start {} ocp-express'.format(server)) + elif count: + tmp_servers.append(server) + stdio.verbose('failed to start %s ocp-express, remaining retries: %d' % (server, count)) + else: + failed.append('failed to start {} ocp-express'.format(server)) + servers = tmp_servers + if servers and count: + time.sleep(3) + if failed: + stdio.stop_loading('failed') + for msg in failed: + stdio.error(msg) + return plugin_context.return_false() + else: + stdio.stop_loading('succeed') + plugin_context.return_true(need_bootstrap=True) + + return False + diff --git a/plugins/ocp-express/1.0/start_check.py b/plugins/ocp-express/1.0/start_check.py new file mode 100644 index 0000000..d654a13 --- /dev/null +++ b/plugins/ocp-express/1.0/start_check.py @@ -0,0 +1,527 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import re +import os + +from copy import deepcopy +from _rpm import Version +import _errno as err + +success = True + + +def get_missing_required_parameters(parameters): + results = [] + for key in ["jdbc_url", "jdbc_password", "jdbc_username", "cluster_name", "ob_cluster_id", "root_sys_password", + "server_addresses", "agent_username", "agent_password"]: + if parameters.get(key) is None: + results.append(key) + return results + + +def get_port_socket_inode(client, port): + port = hex(port)[2:].zfill(4).upper() + cmd = "bash -c 'cat /proc/net/{udp*,tcp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + res = client.execute_command(cmd) + if not res or not res.stdout.strip(): + return False + return res.stdout.strip().split('\n') + + +def password_check(passwd): + pattern = r"((?=(.*\d){2,})(?=(.*[a-z]){2,})(?=(.*[A-Z]){2,})(?=(.*[~!@#%^&*_\-+=|(){}\[\]:;,.?/]){2,})[0-9a-zA-Z~!@#%^&*_\-+=|(){}\[\]:;,.?/]{8,32})" + if re.match(pattern, passwd): + return True + + +def parse_size(size): + _bytes = 0 + if not isinstance(size, str) or size.isdigit(): + _bytes = int(size) + else: + units = {"B": 1, "K": 1<<10, "M": 1<<20, "G": 1<<30, "T": 1<<40} + match = re.match(r'(0|[1-9][0-9]*)\s*([B,K,M,G,T])', size.upper()) + _bytes = int(match.group(1)) * units[match.group(2)] + return _bytes + + +def format_size(size, precision=1): + units = ['B', 'K', 'M', 'G'] + units_num = len(units) - 1 + idx = 0 + if precision: + div = 1024.0 + format = '%.' + str(precision) + 'f%s' + limit = 1024 + else: + div = 1024 + limit = 1024 + format = '%d%s' + while idx < units_num and size >= limit: + size /= div + idx += 1 + return format % (size, units[idx]) + + +def get_mount_path(disk, _path): + _mount_path = '/' + for p in disk: + if p in _path: + if len(p) > len(_mount_path): + _mount_path = p + return _mount_path + + +def get_disk_info_by_path(path, client, stdio): + disk_info = {} + ret = client.execute_command('df --block-size=1024 {}'.format(path)) + if ret: + for total, used, avail, puse, path in re.findall(r'(\d+)\s+(\d+)\s+(\d+)\s+(\d+%)\s+(.+)', ret.stdout): + disk_info[path] = {'total': int(total) << 10, 'avail': int(avail) << 10, 'need': 0} + stdio.verbose('get disk info for path {}, total: {} avail: {}'.format(path, disk_info[path]['total'], disk_info[path]['avail'])) + return disk_info + + +def get_disk_info(all_paths, client, stdio): + overview_ret = True + disk_info = get_disk_info_by_path('', client, stdio) + if not disk_info: + overview_ret = False + disk_info = get_disk_info_by_path('/', client, stdio) + if not disk_info: + disk_info['/'] = {'total': 0, 'avail': 0, 'need': 0} + all_path_success = {} + for path in all_paths: + all_path_success[path] = False + cur_path = path + while cur_path not in disk_info: + disk_info_for_current_path = get_disk_info_by_path(cur_path, client, stdio) + if disk_info_for_current_path: + disk_info.update(disk_info_for_current_path) + all_path_success[path] = True + break + else: + cur_path = os.path.dirname(cur_path) + if overview_ret or all(all_path_success.values()): + return disk_info + + +def prepare_parameters(cluster_config, stdio): + # depends config + env = {} + depend_observer = False + depend_info = {} + ob_servers_conf = {} + root_servers = [] + for comp in ["oceanbase", "oceanbase-ce"]: + ob_zones = {} + if comp in cluster_config.depends: + depend_observer = True + observer_globals = cluster_config.get_depend_config(comp) + ocp_meta_keys = [ + "ocp_meta_tenant", "ocp_meta_db", "ocp_meta_username", "ocp_meta_password", "appname", "cluster_id", "root_password" + ] + for key in ocp_meta_keys: + value = observer_globals.get(key) + if value is not None: + depend_info[key] = value + ob_servers = cluster_config.get_depend_servers(comp) + for ob_server in ob_servers: + ob_servers_conf[ob_server] = ob_server_conf = cluster_config.get_depend_config(comp, ob_server) + if 'server_ip' not in depend_info: + depend_info['server_ip'] = ob_server.ip + depend_info['mysql_port'] = ob_server_conf['mysql_port'] + zone = ob_server_conf['zone'] + if zone not in ob_zones: + ob_zones[zone] = ob_server + root_servers = ob_zones.values() + break + for comp in ['obproxy', 'obproxy-ce']: + if comp in cluster_config.depends: + obproxy_servers = cluster_config.get_depend_servers(comp) + obproxy_server = obproxy_servers[0] + obproxy_server_config = cluster_config.get_depend_config(comp, obproxy_server) + depend_info['server_ip'] = obproxy_server.ip + depend_info['mysql_port'] = obproxy_server_config['listen_port'] + break + if 'obagent' in cluster_config.depends: + obagent_servers = cluster_config.get_depend_servers('obagent') + server_addresses = [] + for obagent_server in obagent_servers: + obagent_server_config_without_default = cluster_config.get_depend_config('obagent', obagent_server, with_default=False) + obagent_server_config = cluster_config.get_depend_config('obagent', obagent_server) + username = obagent_server_config['http_basic_auth_user'] + password = obagent_server_config['http_basic_auth_password'] + if 'obagent_username' not in depend_info: + depend_info['obagent_username'] = username + elif depend_info['obagent_username'] != username: + stdio.error('The http basic auth of obagent is inconsistent') + return + if 'obagent_password' not in depend_info: + depend_info['obagent_password'] = password + elif depend_info['obagent_password'] != password: + stdio.error('The http basic auth of obagent is inconsistent') + return + if obagent_server_config_without_default.get('sql_port'): + sql_port = obagent_server_config['sql_port'] + elif ob_servers_conf.get(obagent_server) and ob_servers_conf[obagent_server].get('mysql_port'): + sql_port = ob_servers_conf[obagent_server]['mysql_port'] + else: + continue + if obagent_server_config_without_default.get('rpc_port'): + svr_port = obagent_server_config['rpc_port'] + elif ob_servers_conf.get(obagent_server) and ob_servers_conf[obagent_server].get('rpc_port'): + svr_port = ob_servers_conf[obagent_server]['rpc_port'] + else: + continue + server_addresses.append({ + "address": obagent_server.ip, + "svrPort": svr_port, + "sqlPort": sql_port, + "withRootServer": obagent_server in root_servers, + "agentMgrPort": obagent_server_config.get('mgragent_http_port', 0), + "agentMonPort": obagent_server_config.get('monagent_http_port', 0) + }) + depend_info['server_addresses'] = server_addresses + + for server in cluster_config.servers: + server_config = deepcopy(cluster_config.get_server_conf_with_default(server)) + original_server_config = cluster_config.get_original_server_conf(server) + missed_keys = get_missing_required_parameters(original_server_config) + if missed_keys: + if 'jdbc_url' in missed_keys and depend_observer: + server_config['jdbc_url'] = 'jdbc:oceanbase://{}:{}/{}'.format(depend_info['server_ip'], depend_info['mysql_port'], depend_info['ocp_meta_db']) + if 'jdbc_username' in missed_keys and depend_observer: + server_config['jdbc_username'] = "{}@{}".format(depend_info['ocp_meta_username'], depend_info.get('ocp_meta_tenant', {}).get("tenant_name")) + depends_key_maps = { + "jdbc_password": "ocp_meta_password", + "cluster_name": "appname", + "ob_cluster_id": "cluster_id", + "root_sys_password": "root_password", + "agent_username": "obagent_username", + "agent_password": "obagent_password", + "server_addresses": "server_addresses" + } + for key in depends_key_maps: + if key in missed_keys: + if depend_info.get(depends_key_maps[key]) is not None: + server_config[key] = depend_info[depends_key_maps[key]] + env[server] = server_config + return env + + +def start_check(plugin_context, init_check_status=False, work_dir_check=False, work_dir_empty_check=True, strict_check=False, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def alert(item, error, suggests=[]): + global success + if strict_check: + success = False + check_fail(item, error, suggests) + stdio.error(error) + else: + stdio.warn(error) + def error(item, _error, suggests=[]): + global success + if plugin_context.dev_mode: + stdio.warn(_error) + else: + success = False + check_fail(item, _error, suggests) + stdio.error(_error) + def critical(item, error, suggests=[]): + global success + success = False + check_fail(item, error, suggests) + stdio.error(error) + + cluster_config = plugin_context.cluster_config + option = plugin_context.options + clients = plugin_context.clients + stdio = plugin_context.stdio + global success + success = True + + check_status = {} + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + 'java': err.CheckStatus(), + 'disk': err.CheckStatus(), + 'mem': err.CheckStatus(), + 'oceanbase version': err.CheckStatus(), + 'obagent version': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + + stdio.start_loading('Check before start ocp-express') + env = prepare_parameters(cluster_config, stdio) + if not env: + return plugin_context.return_false() + versions_check = { + "oceanbase version": { + 'comps': ['oceanbase', 'oceanbase-ce'], + 'min_version': Version('4.0') + }, + "obagent version": { + 'comps': ['obagent'], + 'min_version': Version('1.3.0') + } + } + repo_versions = {} + for repository in plugin_context.repositories: + repo_versions[repository.name] = repository.version + + for check_item in versions_check: + for comp in versions_check[check_item]['comps']: + if comp not in cluster_config.depends: + continue + depend_comp_version = repo_versions.get(comp) + if depend_comp_version is None: + stdio.verbose('failed to get {} version, skip version check'.format(comp)) + continue + min_version = versions_check[check_item]['min_version'] + if depend_comp_version < min_version: + critical(check_item, err.EC_OCP_EXPRESS_DEPENDS_COMP_VERSION.format(ocp_express_version=cluster_config.version, comp=comp, comp_version=min_version)) + + server_port = {} + servers_dirs = {} + servers_check_dirs = {} + for server in cluster_config.servers: + client = clients[server] + server_config = env[server] + missed_keys = get_missing_required_parameters(server_config) + if missed_keys: + stdio.error(err.EC_NEED_CONFIG.format(server=server, component=cluster_config.name, miss_keys=missed_keys)) + success = False + continue + home_path = server_config['home_path'] + if not precheck: + remote_pid_path = '%s/run/ocp-express.pid' % home_path + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is running, skip' % server) + wait_2_pass() + continue + + if work_dir_check: + ip = server.ip + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + original_server_conf = cluster_config.get_server_conf(server) + + keys = ['home_path', 'log_dir'] + for key in keys: + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + if key not in original_server_conf: + continue + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False + + port = server_config['port'] + ip = server.ip + if ip not in server_port: + server_port[ip] = {} + ports = server_port[ip] + if port in server_port[ip]: + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], + key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) + continue + ports[port] = { + 'server': server, + 'key': 'port' + } + if get_port_socket_inode(client, port): + critical( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + continue + check_pass('port') + + # java version check + for server in cluster_config.servers: + client = clients[server] + server_config = env[server] + java_bin = server_config['java_bin'] + ret = client.execute_command('{} -version'.format(java_bin)) + if not ret: + critical('java', err.EC_OCP_EXPRESS_JAVA_NOT_FOUND.format(server=server), [err.SUG_OCP_EXPRESS_INSTALL_JAVA_WITH_VERSION.format(version='1.8.0')]) + continue + version_pattern = r'version\s+\"(\d+\.\d+.\d+)' + found = re.search(version_pattern, ret.stdout) or re.search(version_pattern, ret.stderr) + if not found: + error('java', err.EC_OCP_EXPRESS_JAVA_VERSION_ERROR.format(server=server, version='1.8.0'), [err.SUG_OCP_EXPRESS_INSTALL_JAVA_WITH_VERSION.format(version='1.8.0'),]) + continue + java_major_version = found.group(1) + if Version(java_major_version) != Version('1.8.0'): + critical('java', err.EC_OCP_EXPRESS_JAVA_VERSION_ERROR.format(server=server, version='1.8.0'), [err.SUG_OCP_EXPRESS_INSTALL_JAVA_WITH_VERSION.format(version='1.8.0'),]) + continue + + servers_memory = {} + servers_disk = {} + servers_client = {} + ip_servers = {} + + for server in cluster_config.servers: + client = clients[server] + server_config = env[server] + memory_size = parse_size(server_config['memory_size']) + if server_config.get('log_dir'): + log_dir = server_config['log_dir'] + else: + log_dir = os.path.join(server_config['home_path'], 'log') + need_size = parse_size(server_config['logging_file_total_size_cap']) + ip = server.ip + if ip not in servers_client: + servers_client[ip] = client + if ip not in servers_memory: + servers_memory[ip] = { + 'need': memory_size, + 'server_num': 1 + } + else: + servers_memory[ip]['need'] += memory_size + servers_memory[ip]['server_num'] += 1 + if ip not in servers_disk: + servers_disk[ip] = {} + if log_dir not in servers_disk[ip]: + servers_disk[ip][log_dir] = need_size + else: + servers_disk[ip][log_dir] += need_size + if ip not in ip_servers: + ip_servers[ip] = [server] + else: + ip_servers[ip].append(server) + # memory check + for ip in servers_memory: + client = servers_client[ip] + memory_needed = servers_memory[ip]['need'] + ret = client.execute_command('cat /proc/meminfo') + if ret: + server_memory_stats = {} + memory_key_map = { + 'MemTotal': 'total', + 'MemFree': 'free', + 'MemAvailable': 'available', + 'Buffers': 'buffers', + 'Cached': 'cached' + } + for key in memory_key_map: + server_memory_stats[memory_key_map[key]] = 0 + + for k, v in re.findall('(\w+)\s*:\s*(\d+\s*\w+)', ret.stdout): + if k in memory_key_map: + key = memory_key_map[k] + server_memory_stats[key] = parse_size(str(v)) + mem_suggests = [err.SUG_OCP_EXPRESS_REDUCE_MEM.format()] + if memory_needed * 0.5 > server_memory_stats['available']: + for server in ip_servers[ip]: + error('mem', err.EC_OCP_EXPRESS_NOT_ENOUGH_MEMORY_AVAILABLE.format(ip=ip, available=format_size(server_memory_stats['available']), need=format_size(memory_needed)), suggests=mem_suggests) + elif memory_needed > server_memory_stats['free'] + server_memory_stats['buffers'] + server_memory_stats['cached']: + for server in ip_servers[ip]: + error('mem', err.EC_OCP_EXPRESS_NOT_ENOUGH_MEMORY_CACHED.format(ip=ip, free=format_size(server_memory_stats['free']), cached=format_size(server_memory_stats['buffers'] + server_memory_stats['cached']), need=format_size(memory_needed)), suggests=mem_suggests) + elif memory_needed > server_memory_stats['free']: + for server in ip_servers[ip]: + alert('mem', err.EC_OCP_EXPRESS_NOT_ENOUGH_MEMORY.format(ip=ip, free=format_size(server_memory_stats['free']), need=format_size(memory_needed)), suggests=mem_suggests) + # disk check + for ip in servers_disk: + client = servers_client[ip] + disk_info = get_disk_info(all_paths=servers_disk[ip], client=client, stdio=stdio) + if disk_info: + for path in servers_disk[ip]: + disk_needed = servers_disk[ip][path] + mount_path = get_mount_path(disk_info, path) + if disk_needed > disk_info[mount_path]['avail']: + for server in ip_servers[ip]: + error('disk', err.EC_OCP_EXPRESS_NOT_ENOUGH_DISK.format(ip=ip, disk=mount_path, need=format_size(disk_needed), avail=format_size(disk_info[mount_path]['avail'])), suggests=[err.SUG_OCP_EXPRESS_REDUCE_DISK.format()]) + else: + stdio.warn(err.WC_OCP_EXPRESS_FAILED_TO_GET_DISK_INFO.format(ip)) + plugin_context.set_variable('start_env', env) + + for server in cluster_config.servers: + wait_2_pass() + + if success: + stdio.stop_loading('succeed') + plugin_context.return_true() + else: + stdio.stop_loading('fail') diff --git a/plugins/ocp-express/1.0/status.py b/plugins/ocp-express/1.0/status.py new file mode 100644 index 0000000..72b7aab --- /dev/null +++ b/plugins/ocp-express/1.0/status.py @@ -0,0 +1,39 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os + + +def status(plugin_context, *args, **kwargs): + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + cluster_status = {} + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + cluster_status[server] = 0 + pid_path = os.path.join(server_config['home_path'], 'run/ocp-express.pid') + pids = client.execute_command('cat {}'.format(pid_path)).stdout.strip().split('\n') + for pid in pids: + if pid and client.execute_command('ls /proc/{}'.format(pid)): + cluster_status[server] = 1 + return plugin_context.return_true(cluster_status=cluster_status) \ No newline at end of file diff --git a/plugins/ocp-express/1.0/stop.py b/plugins/ocp-express/1.0/stop.py new file mode 100644 index 0000000..c7e98bf --- /dev/null +++ b/plugins/ocp-express/1.0/stop.py @@ -0,0 +1,107 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + + +from __future__ import absolute_import, division, print_function + +import os +import time + + +def get_port_socket_inode(client, port, stdio): + port = hex(port)[2:].zfill(4).upper() + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + res = client.execute_command(cmd) + if not res or not res.stdout.strip(): + return False + stdio.verbose(res.stdout) + return res.stdout.strip().split('\n') + + +def confirm_port(client, pid, port, stdio): + socket_inodes = get_port_socket_inode(client, port, stdio) + if not socket_inodes: + return False + ret = client.execute_command("ls -l /proc/%s/fd/ |grep -E 'socket:\[(%s)\]'" % (pid, '|'.join(socket_inodes))) + if ret and ret.stdout.strip(): + return True + return False + + +def stop(plugin_context, *args, **kwargs): + cluster_config = plugin_context.cluster_config + clients = plugin_context.clients + stdio = plugin_context.stdio + servers = {} + stdio.start_loading('Stop ocp-express') + success = True + for server in cluster_config.servers: + server_config = cluster_config.get_server_conf(server) + client = clients[server] + home_path = server_config['home_path'] + pid_path = os.path.join(home_path, 'run/ocp-express.pid') + pids = client.execute_command('cat {}'.format(pid_path)).stdout.strip().split('\n') + for pid in pids: + if pid and client.execute_command('ls /proc/{}'.format(pid)): + if client.execute_command('ls /proc/%s/fd' % pid): + stdio.verbose('{} ocp-express[pid: {}] stopping...'.format(server, pid)) + client.execute_command('kill -9 {}'.format(pid)) + servers[server] = { + 'client': client, + 'port': server_config['port'], + 'pid': pid, + 'path': pid_path + } + else: + stdio.verbose('failed to stop ocp-express[pid:{}] in {}, permission deny'.format(pid, server)) + success = False + else: + stdio.verbose('{} ocp-express is not running'.format(server)) + if not success: + stdio.stop_loading('fail') + return plugin_context.return_true() + + count = 10 + check = lambda client, pid, port: confirm_port(client, pid, port, stdio) if count < 5 else get_port_socket_inode(client, port, stdio) + time.sleep(1) + while count and servers: + tmp_servers = {} + for server in servers: + data = servers[server] + client = clients[server] + stdio.verbose('%s check whether the port is released' % server) + for key in ['port']: + if data[key] and check(data['client'], data['pid'], data[key]): + tmp_servers[server] = data + break + data[key] = '' + else: + client.execute_command('rm -rf %s' % data['path']) + stdio.verbose('%s ocp-server is stopped', server) + servers = tmp_servers + count -= 1 + if count and servers: + time.sleep(3) + if servers: + stdio.stop_loading('fail') + for server in servers: + stdio.warn('%s port not released'% server) + else: + stdio.stop_loading('succeed') + return plugin_context.return_true() diff --git a/plugins/prometheus/2.37.1/display.py b/plugins/prometheus/2.37.1/display.py index 1940f82..bff435a 100644 --- a/plugins/prometheus/2.37.1/display.py +++ b/plugins/prometheus/2.37.1/display.py @@ -20,8 +20,7 @@ from __future__ import absolute_import, division, print_function -import socket -from tool import YamlLoader +from tool import YamlLoader, NetUtil yaml = YamlLoader() @@ -39,8 +38,7 @@ def display(plugin_context, cursor, *args, **kwargs): password = api_cursor.password ip = server.ip if ip == '127.0.0.1': - hostname = socket.gethostname() - ip = socket.gethostbyname(hostname) + ip = NetUtil.get_host_ip() url = '%s://%s:%s' % (protocol, ip, server_config['port']) results.append({ 'ip': ip, @@ -51,4 +49,8 @@ def display(plugin_context, cursor, *args, **kwargs): 'status': 'active' if api_cursor and api_cursor.connect(stdio) else 'inactive' }) stdio.print_list(results, ['url', 'user', 'password', 'status'], lambda x: [x['url'], x['user'], x['password'], x['status']], title='prometheus') - return plugin_context.return_true() + active_result = [r for r in results if r['status'] == 'active'] + info_dict = active_result[0] if len(active_result) > 0 else None + if info_dict is not None: + info_dict['type'] = 'web' + return plugin_context.return_true(info=info_dict) diff --git a/plugins/prometheus/2.37.1/generate_config.py b/plugins/prometheus/2.37.1/generate_config.py index c4ddebc..f2eef6e 100644 --- a/plugins/prometheus/2.37.1/generate_config.py +++ b/plugins/prometheus/2.37.1/generate_config.py @@ -21,23 +21,18 @@ from __future__ import absolute_import, division, print_function -def generate_config(plugin_context, deploy_config, auto_depend=False, *args, **kwargs): +def generate_config(plugin_context, auto_depend=False, return_generate_keys=False, *args, **kwargs): + if return_generate_keys: + return plugin_context.return_true(generate_keys=[]) + cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio - success = True have_depend = False depends = ['obagent'] + generate_configs = {'global': {}} + plugin_context.set_variable('generate_configs', generate_configs) stdio.start_loading('Generate prometheus configuration') - for server in cluster_config.servers: - server_config = cluster_config.get_server_conf(server) - if not server_config.get('home_path'): - stdio.error("prometheus %s: missing configuration 'home_path' in configuration file" % server) - success = False - if not success: - stdio.stop_loading('fail') - return - for comp in cluster_config.depends: if comp in depends: have_depend = True @@ -46,5 +41,6 @@ def generate_config(plugin_context, deploy_config, auto_depend=False, *args, **k for depend in depends: if cluster_config.add_depend_component(depend): break + stdio.stop_loading('succeed') plugin_context.return_true() \ No newline at end of file diff --git a/plugins/prometheus/2.37.1/init.py b/plugins/prometheus/2.37.1/init.py index 2b0828c..61cc0b9 100644 --- a/plugins/prometheus/2.37.1/init.py +++ b/plugins/prometheus/2.37.1/init.py @@ -35,7 +35,7 @@ def _clean(server, client, path, stdio=None): return True -def init(plugin_context, repositories_dir_map, *args, **kwargs): +def init(plugin_context, *args, **kwargs): cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio diff --git a/plugins/prometheus/2.37.1/parameter.yaml b/plugins/prometheus/2.37.1/parameter.yaml index cb45aca..e7d070d 100644 --- a/plugins/prometheus/2.37.1/parameter.yaml +++ b/plugins/prometheus/2.37.1/parameter.yaml @@ -1,5 +1,7 @@ - name: home_path + name_local: 工作目录 require: true + essential: true type: STRING min_value: NULL max_value: NULL @@ -7,7 +9,9 @@ description_en: the directory for the work data file description_local: Prometheus工作目录 - name: data_dir + name_local: 数据目录 require: false + essential: true type: STRING need_redeploy: true description_en: Base path for metrics storage. @@ -21,6 +25,8 @@ description_local: - name: port require: true + name_local: 服务端口 + essential: true default: 9090 type: INT need_restart: true diff --git a/plugins/prometheus/2.37.1/restart.py b/plugins/prometheus/2.37.1/restart.py index 0be45a3..5c6ccc4 100644 --- a/plugins/prometheus/2.37.1/restart.py +++ b/plugins/prometheus/2.37.1/restart.py @@ -29,11 +29,22 @@ def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, display_plugin, repository, new_cluster_config=None, new_clients=None, bootstrap_plugin=None, repository_dir_map=None): self.local_home_path = local_home_path - self.plugin_context = plugin_context + + self.namespace = plugin_context.namespace + self.namespaces = plugin_context.namespaces + self.deploy_name = plugin_context.deploy_name + self.repositories = plugin_context.repositories + self.plugin_name = plugin_context.plugin_name + self.components = plugin_context.components self.clients = plugin_context.clients self.cluster_config = plugin_context.cluster_config + self.cmds = plugin_context.cmds + self.options = plugin_context.options + self.dev_mode = plugin_context.dev_mode self.stdio = plugin_context.stdio + + self.plugin_context = plugin_context self.repository = repository self.start_plugin = start_plugin self.reload_plugin = reload_plugin @@ -47,13 +58,30 @@ def __init__(self, plugin_context, local_home_path, start_plugin, reload_plugin, self.dbs = None self.cursors = None self.repository_dir_map = repository_dir_map + + def call_plugin(self, plugin, **kwargs): + args = { + 'namespace': self.namespace, + 'namespaces': self.namespaces, + 'deploy_name': self.deploy_name, + 'cluster_config': self.cluster_config, + 'repositories': self.repositories, + 'repository': self.repository, + 'components': self.components, + 'clients': self.clients, + 'cmd': self.cmds, + 'options': self.options, + 'stdio': self.sub_io + } + args.update(kwargs) + + self.stdio.verbose('Call %s for %s' % (plugin, self.repository)) + return plugin(**args) def connect(self, cluster_config): if self.cursors is None: - self.stdio.verbose('Call %s for %s' % (self.connect_plugin, self.repository)) self.sub_io.start_loading('Connect to prometheus') - ret = self.connect_plugin(self.components, self.clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io) + ret = self.connect_plugin(self.namespace, self.namespaces, self.deploy_name, self.repositories, self.components, self.clients, cluster_config, self.cmds, self.options, self.sub_io) if not ret: self.sub_io.stop_loading('fail') return False @@ -70,16 +98,13 @@ def dir_read_check(self, client, path): def restart(self): clients = self.clients - self.stdio.verbose('Call %s for %s' % (self.stop_plugin, self.repository)) - if not self.stop_plugin(self.components, clients, self.cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io): + if not self.call_plugin(self.stop_plugin, clients=clients): self.stdio.stop_loading('stop_loading', 'fail') return False if self.new_clients: self.stdio.verbose('use new clients') for server in self.cluster_config.servers: - client = clients[server] new_client = self.new_clients[server] server_config = self.cluster_config.get_server_conf(server) for key in ['home_path', 'data_dir']: @@ -92,30 +117,22 @@ def restart(self): clients = self.new_clients cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config - self.stdio.verbose('Call %s for %s' % (self.start_plugin, self.repository)) need_bootstrap = self.bootstrap_plugin is not None - if not self.start_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io, local_home_path=self.local_home_path, - repository_dir=self.repository.repository_dir, need_bootstrap=need_bootstrap, repository_dir_map=self.repository_dir_map): + if not self.call_plugin(self.start_plugin, clients=clients, cluster_config=cluster_config, local_home_path=self.local_home_path, need_bootstrap=need_bootstrap, repository_dir_map=self.repository_dir_map): self.rollback() self.stdio.stop_loading('stop_loading', 'fail') return False if self.connect(cluster_config): if self.bootstrap_plugin: - self.stdio.verbose('Call %s for %s' % (self.bootstrap_plugin, self.repository)) - self.bootstrap_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io, cursor=self.cursors) - self.stdio.verbose('Call %s for %s' % (self.display_plugin, self.repository)) - ret = self.display_plugin(self.components, clients, cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io, cursor=self.cursors) - return ret + self.call_plugin(self.bootstrap_plugin, cursor=self.cursors) + return self.call_plugin(self.display_plugin, cursor=self.cursors) return False def rollback(self): if self.new_clients: - self.stop_plugin(self.components, self.new_clients, self.new_cluster_config, self.plugin_context.cmd, - self.plugin_context.options, self.sub_io) + cluster_config = self.new_cluster_config if self.new_cluster_config else self.cluster_config + self.call_plugin(self.stop_plugin, clients=self.new_clients, cluster_config=cluster_config) for server in self.cluster_config.servers: client = self.clients[server] new_client = self.new_clients[server] @@ -125,8 +142,9 @@ def rollback(self): def restart(plugin_context, local_home_path, start_plugin, reload_plugin, stop_plugin, connect_plugin, display_plugin, - repository, new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, repository_dir_map=None, *args, + new_cluster_config=None, new_clients=None, rollback=False, bootstrap_plugin=None, repository_dir_map=None, *args, **kwargs): + repository = kwargs.get('repository') task = Restart(plugin_context=plugin_context, local_home_path=local_home_path, start_plugin=start_plugin, reload_plugin=reload_plugin, stop_plugin=stop_plugin, connect_plugin=connect_plugin, display_plugin=display_plugin, repository=repository, new_cluster_config=new_cluster_config, new_clients=new_clients, repository_dir_map=repository_dir_map) call = task.rollback if rollback else task.restart diff --git a/plugins/prometheus/2.37.1/start.py b/plugins/prometheus/2.37.1/start.py index 042aa60..878ec1a 100644 --- a/plugins/prometheus/2.37.1/start.py +++ b/plugins/prometheus/2.37.1/start.py @@ -28,6 +28,7 @@ import bcrypt from tool import YamlLoader, FileUtil +from _rpm import Version prometheusd_path = os.path.join(os.path.split(__file__)[0], 'prometheusd.sh') @@ -73,14 +74,13 @@ def prometheusd(home_path, client, server, args, start_only=False, stdio=None): return True -def load_config_from_obagent(cluster_config, repository_dir_map, stdio, client, server, server_config, yaml): +def load_config_from_obagent(cluster_config, obagent_repo, stdio, client, server, server_config, yaml): stdio.verbose('load config from obagent') server_home_path = server_config['home_path'] port = server_config['port'] - address = server_config['address'] + address = server_config['address'] obagent_servers = cluster_config.get_depend_servers('obagent') - obagent_repo_dir = repository_dir_map['obagent'] - prometheus_conf_dir = os.path.join(obagent_repo_dir, 'conf/prometheus_config') + prometheus_conf_dir = os.path.join(obagent_repo.repository_dir, 'conf/prometheus_config') prometheus_conf_path = os.path.join(prometheus_conf_dir, 'prometheus.yaml') rules_dir = os.path.join(prometheus_conf_dir, 'rules') remote_rules_dir = os.path.join(server_home_path, 'rules') @@ -88,9 +88,13 @@ def load_config_from_obagent(cluster_config, repository_dir_map, stdio, client, obagent_targets = [] http_basic_auth_user = None http_basic_auth_password = None + watershed = Version('1.3.0') for obagent_server in obagent_servers: obagent_server_config = cluster_config.get_depend_config('obagent', obagent_server) - server_port = obagent_server_config['server_port'] + if obagent_repo.version < watershed: + server_port = obagent_server_config['server_port'] + else: + server_port = obagent_server_config['monagent_http_port'] obagent_targets.append('{}:{}'.format(obagent_server.ip, server_port)) if http_basic_auth_user is None: http_basic_auth_user = obagent_server_config['http_basic_auth_user'] @@ -134,7 +138,7 @@ def load_config_from_obagent(cluster_config, repository_dir_map, stdio, client, raise e -def start(plugin_context, local_home_path, repository_dir, repository_dir_map=None, *args, **kwargs): +def start(plugin_context, *args, **kwargs): def generate_or_update_config(): prometheus_conf_content = None @@ -148,9 +152,9 @@ def generate_or_update_config(): stdio.exception('') stdio.warn('{}: invalid prometheus config {}, regenerate a new config.'.format(server, runtime_prometheus_conf)) if prometheus_conf_content is None: - if 'obagent' in cluster_config.depends and repository_dir_map: + if obagent_repo: try: - prometheus_conf_content = load_config_from_obagent(cluster_config, repository_dir_map, stdio, client, server, server_config, yaml=yaml) + prometheus_conf_content = load_config_from_obagent(cluster_config, obagent_repo, stdio, client, server, server_config, yaml=yaml) except Exception as e: stdio.exception(e) return False @@ -188,6 +192,13 @@ def check_parameter(key): yaml = YamlLoader(stdio=stdio) pid_path = {} cmd_args_map = {} + obagent_repo = None + if 'obagent' in cluster_config.depends: + for repository in plugin_context.repositories: + if repository.name == 'obagent': + stdio.verbose('obagent version: {}'.format(repository.version)) + obagent_repo = repository + break stdio.start_loading('Start promethues') if not os.path.exists(prometheusd_path): diff --git a/plugins/prometheus/2.37.1/start_check.py b/plugins/prometheus/2.37.1/start_check.py index edb915e..9192687 100644 --- a/plugins/prometheus/2.37.1/start_check.py +++ b/plugins/prometheus/2.37.1/start_check.py @@ -20,7 +20,9 @@ from __future__ import absolute_import, division, print_function -from _errno import EC_CONFIG_CONFLICT_PORT +import os +import _errno as err + stdio = None success = True @@ -28,7 +30,7 @@ def get_port_socket_inode(client, port): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) if not res or not res.stdout.strip(): return False @@ -36,14 +38,29 @@ def get_port_socket_inode(client, port): return res.stdout.strip().split('\n') -def start_check(plugin_context, strict_check=False, *args, **kwargs): - - def critical(*arg, **kwargs): +def start_check(plugin_context, init_check_status=False, work_dir_check=False, work_dir_empty_check=True, precheck=False, *args, **kwargs): + def check_pass(item): + status = check_status[server] + if status[item].status == err.CheckStatus.WAIT: + status[item].status = err.CheckStatus.PASS + def check_fail(item, error, suggests=[]): + status = check_status[server][item] + if status.status == err.CheckStatus.WAIT: + status.error = error + status.suggests = suggests + status.status = err.CheckStatus.FAIL + def wait_2_pass(): + status = check_status[server] + for item in status: + check_pass(item) + def critical(item, error, suggests=[]): global success success = False - stdio.error(*arg, **kwargs) - + check_fail(item, error, suggests) + stdio.error(error) + global stdio, success + success = True cluster_config = plugin_context.cluster_config clients = plugin_context.clients stdio = plugin_context.stdio @@ -52,6 +69,20 @@ def critical(*arg, **kwargs): depends = ['obagent'] username = None password = None + check_status = {} + servers_dirs = {} + servers_check_dirs = {} + plugin_context.set_variable('start_check_status', check_status) + for server in cluster_config.servers: + check_status[server] = { + 'port': err.CheckStatus(), + } + if work_dir_check: + check_status[server]['dir'] = err.CheckStatus() + + if init_check_status: + return plugin_context.return_true(start_check_status=check_status) + for comp in cluster_config.depends: if comp in depends: for server in cluster_config.get_depend_servers(comp): @@ -72,14 +103,69 @@ def critical(*arg, **kwargs): ip = server.ip client = clients[server] servers_clients[ip] = client - server_config = cluster_config.get_server_conf(server) + server_config = cluster_config.get_server_conf_with_default(server) home_path = server_config['home_path'] - remote_pid_path = '%s/run/prometheus.pid' % home_path - remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() - if remote_pid: - if client.execute_command('ls /proc/%s' % remote_pid): - continue + if not precheck: + remote_pid_path = '%s/run/prometheus.pid' % home_path + remote_pid = client.execute_command('cat %s' % remote_pid_path).stdout.strip() + if remote_pid: + if client.execute_command('ls /proc/%s' % remote_pid): + stdio.verbose('%s is runnning, skip' % server) + wait_2_pass() + continue + + if work_dir_check: + stdio.verbose('%s dir check' % server) + if ip not in servers_dirs: + servers_dirs[ip] = {} + servers_check_dirs[ip] = {} + dirs = servers_dirs[ip] + check_dirs = servers_check_dirs[ip] + original_server_conf = cluster_config.get_server_conf(server) + if not server_config.get('data_dir'): + server_config['data_dir'] = '%s/data' % home_path + + keys = ['home_path', 'data_dir'] + for key in keys: + path = server_config.get(key) + suggests = [err.SUG_CONFIG_CONFLICT_DIR.format(key=key, server=server)] + if path in dirs and dirs[path]: + critical('dir', err.EC_CONFIG_CONFLICT_DIR.format(server1=server, path=path, server2=dirs[path]['server'], key=dirs[path]['key']), suggests) + dirs[path] = { + 'server': server, + 'key': key, + } + if key not in original_server_conf: + continue + empty_check = work_dir_empty_check + while True: + if path in check_dirs: + if check_dirs[path] != True: + critical('dir', check_dirs[path], suggests) + break + + if client.execute_command('bash -c "[ -a %s ]"' % path): + is_dir = client.execute_command('[ -d {} ]'.format(path)) + has_write_permission = client.execute_command('[ -w {} ]'.format(path)) + if is_dir and has_write_permission: + if empty_check: + ret = client.execute_command('ls %s' % path) + if not ret or ret.stdout.strip(): + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_EMPTY.format(path=path)) + else: + check_dirs[path] = True + else: + check_dirs[path] = True + else: + if not is_dir: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.NOT_DIR.format(path=path)) + else: + check_dirs[path] = err.EC_FAIL_TO_INIT_PATH.format(server=server, key=key, msg=err.InitDirFailedErrorMessage.PERMISSION_DENIED.format(path=path)) + else: + path = os.path.dirname(path) + empty_check = False + if ip not in servers_port: servers_port[ip] = {} ports = servers_port[ip] @@ -87,15 +173,26 @@ def critical(*arg, **kwargs): for key in ['port']: port = int(server_config[key]) if port in ports: - critical(EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'], - key=ports[port]['key'])) + critical( + 'port', + err.EC_CONFIG_CONFLICT_PORT.format(server1=server, port=port, server2=ports[port]['server'],key=ports[port]['key']), + [err.SUG_PORT_CONFLICTS.format()] + ) continue ports[port] = { 'server': server, 'key': key } if get_port_socket_inode(client, port): - critical('%s:%s port is already used' % (ip, port)) + critical( + 'port', + err.EC_CONFLICT_PORT.format(server=ip, port=port), + [err.SUG_USE_OTHER_PORT.format()] + ) + + for server in cluster_config.servers: + wait_2_pass() + if success: stdio.stop_loading('succeed') plugin_context.return_true() diff --git a/plugins/prometheus/2.37.1/stop.py b/plugins/prometheus/2.37.1/stop.py index cb75c84..d8a4f7d 100644 --- a/plugins/prometheus/2.37.1/stop.py +++ b/plugins/prometheus/2.37.1/stop.py @@ -26,7 +26,7 @@ def get_port_socket_inode(client, port, stdio): port = hex(port)[2:].zfill(4).upper() - cmd = "bash -c 'cat /proc/net/{tcp,udp}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port + cmd = "bash -c 'cat /proc/net/{tcp*,udp*}' | awk -F' ' '{print $2,$10}' | grep '00000000:%s' | awk -F' ' '{print $2}' | uniq" % port res = client.execute_command(cmd) inode = res.stdout.strip() if not res or not inode: @@ -72,7 +72,7 @@ def stop(plugin_context, *args, **kwargs): } else: stdio.verbose('failed to stop prometheus[pid:{}] in {}, permission deny'.format(prometheus_pid, server)) - success = True + success = False else: stdio.verbose('{} prometheus is not running'.format(server)) if not success: diff --git a/plugins/sysbench/3.1.0/pre_test.py b/plugins/sysbench/3.1.0/pre_test.py index a7a7832..3098562 100644 --- a/plugins/sysbench/3.1.0/pre_test.py +++ b/plugins/sysbench/3.1.0/pre_test.py @@ -65,18 +65,6 @@ def get_option(key, default=''): value = default return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - # stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - global stdio stdio = plugin_context.stdio cluster_config = plugin_context.cluster_config @@ -120,20 +108,19 @@ def execute(cursor, query, args=None): return sql = "select * from oceanbase.gv$tenant where tenant_name = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant_meta = cursor.fetchone() - if not tenant_meta: - stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) - return - sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['tenant_id'] - pool = execute(cursor, sql) - sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] - max_cpu = execute(cursor, sql)['max_cpu'] - except: - stdio.exception('') + tenant_meta = cursor.fetchone(sql, [tenant_name]) + if not tenant_meta: + stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) + return + sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['tenant_id'] + pool = cursor.fetchone(sql) + if pool is False: return + sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] + max_cpu = cursor.fetchone(sql) + if max_cpu is False: + return + max_cpu = max_cpu['max_cpu'] exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A -e" % ( obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '') @@ -144,7 +131,9 @@ def execute(cursor, query, args=None): return server_num = len(cluster_config.servers) sql = "select count(1) server_num from oceanbase.__all_server where status = 'active'" - ret = execute(cursor, sql) + ret = cursor.fetchone(sql) + if ret is False: + return if ret: server_num = ret.get("server_num", server_num) return plugin_context.return_true( diff --git a/plugins/sysbench/4.0.0.0/pre_test.py b/plugins/sysbench/4.0.0.0/pre_test.py index 5f03528..6c1a52f 100644 --- a/plugins/sysbench/4.0.0.0/pre_test.py +++ b/plugins/sysbench/4.0.0.0/pre_test.py @@ -65,18 +65,6 @@ def get_option(key, default=''): value = default return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - # stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - global stdio cluster_config = plugin_context.cluster_config stdio = plugin_context.stdio @@ -122,20 +110,20 @@ def execute(cursor, query, args=None): sql = "select * from oceanbase.DBA_OB_TENANTS where TENANT_NAME = %s" max_cpu = 2 tenant_meta = None - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant_meta = cursor.fetchone() - if not tenant_meta: - stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) - return - sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['TENANT_ID'] - pool = execute(cursor, sql) - sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] - max_cpu = execute(cursor, sql)['max_cpu'] - except: - stdio.exception('') + stdio.verbose('execute sql: %s' % (sql % tenant_name)) + tenant_meta = cursor.fetchone(sql, [tenant_name]) + if not tenant_meta: + stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) + return + sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['TENANT_ID'] + pool = cursor.fetchone(sql) + if pool is False: return + sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] + max_cpu = cursor.fetchone(sql) + if max_cpu is False: + return + max_cpu = max_cpu['max_cpu'] exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A -e" % ( obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '') @@ -147,7 +135,9 @@ def execute(cursor, query, args=None): server_num = len(cluster_config.servers) sql = "select count(1) server_num from oceanbase.__all_server where status = 'active'" - ret = execute(cursor, sql) + ret = cursor.fetchone(sql) + if ret is False: + return if ret: server_num = ret.get("server_num", server_num) return plugin_context.return_true( diff --git a/plugins/tpcc/3.1.0/build.py b/plugins/tpcc/3.1.0/build.py index 8544347..dfc02d2 100644 --- a/plugins/tpcc/3.1.0/build.py +++ b/plugins/tpcc/3.1.0/build.py @@ -43,18 +43,6 @@ def get_option(key, default=''): def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def run_sql(sql_file, force=False): sql_cmd = "{obclient} -h{host} -P{port} -u{user}@{tenant} {password_arg} -A {db} {force_flag} < {sql_file}".format( obclient=obclient_bin, host=host, port=port, user=user, tenant=tenant_name, @@ -112,57 +100,64 @@ def get_table_rows(table_name): stdio.exception('') return stdio.start_loading('Server check') - try: - # check for observer + # check for observer + while True: + sql = "select * from oceanbase.__all_server where status != 'active' or stop_time > 0 or start_service_time = 0" + ret = cursor.fetchone(sql) + if ret is False: + stdio.stop_loading('fail') + return + if ret is None: + break + time.sleep(3) + # check for obproxy + if odp_cursor: while True: - sql = "select * from oceanbase.__all_server where status != 'active' or stop_time > 0 or start_service_time = 0" - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - ret = cursor.fetchone() - if ret is None: - break - time.sleep(3) - # check for obproxy - if odp_cursor: - while True: - sql = "show proxycongestion all" - stdio.verbose('execute obproxy sql: %s' % sql) - odp_cursor.execute(sql) - proxy_congestions = odp_cursor.fetchall() - passed = True - for proxy_congestion in proxy_congestions: - if proxy_congestion.get('dead_congested') != 0 or proxy_congestion.get('server_state') != 'ACTIVE': - passed = False - break - if passed: + sql = "show proxycongestion all" + proxy_congestions = odp_cursor.fetchall(sql) + if proxy_congestions is False: + stdio.stop_loading('fail') + return + passed = True + for proxy_congestion in proxy_congestions: + if proxy_congestion.get('dead_congested') != 0 or proxy_congestion.get('server_state') != 'ACTIVE': + passed = False break - else: - time.sleep(3) - except: - stdio.stop_loading('fail') - stdio.exception('') - return + if passed: + break + else: + time.sleep(3) stdio.stop_loading('succeed') # drop old tables bmsql_sql_path = kwargs.get('bmsql_sql_path', '') run_sql(sql_file=os.path.join(bmsql_sql_path, 'tableDrops.sql'), force=True) - merge_version = execute(cursor, "select value from oceanbase.__all_zone where name='frozen_version'")['value'] + merge_version = cursor.fetchone("select value from oceanbase.__all_zone where name='frozen_version'") + if merge_version is False: + return + merge_version = merge_version['value'] stdio.start_loading('Merge') - execute(cursor, 'alter system major freeze') + if cursor.fetchone('alter system major freeze') is False: + return sql = "select value from oceanbase.__all_zone where name='frozen_version' and value != %s" % merge_version while True: - if execute(cursor, sql): + res = cursor.fetchone(sql) + if res is False: + return + if res: break time.sleep(1) while True: - if not execute(cursor, """select * from oceanbase.__all_zone - where name='last_merged_version' - and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) - and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') - """): + res = cursor.fetchone("""select * from oceanbase.__all_zone + where name='last_merged_version' + and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) + and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') + """) + if res is False: + return + if not res: break time.sleep(5) stdio.stop_loading('succeed') diff --git a/plugins/tpcc/3.1.0/pre_test.py b/plugins/tpcc/3.1.0/pre_test.py index cee1c0c..30ac968 100644 --- a/plugins/tpcc/3.1.0/pre_test.py +++ b/plugins/tpcc/3.1.0/pre_test.py @@ -59,18 +59,6 @@ def get_option(key, default=''): stdio.verbose('get option: {} value {}'.format(key, value)) return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -164,9 +152,7 @@ def local_execute_command(command, env=None, timeout=None): sql = "select a.id , b.cpu_total from oceanbase.__all_server a " \ "join oceanbase.__all_virtual_server_stat b on a.id=b.id " \ "where a.status = 'active' and a.stop_time = 0 and a.start_service_time > 0;" - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - all_services = cursor.fetchall() + all_services = cursor.fetchall(sql) if not all_services: stdio.error('No active server available.') return @@ -189,23 +175,20 @@ def local_execute_command(command, env=None, timeout=None): local_execute_command("sed -i 's/{{partition_num}}/%d/g' %s" % (cpu_total, create_table_sql)) sql = "select * from oceanbase.gv$tenant where tenant_name = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant_meta = cursor.fetchone() - if not tenant_meta: - stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) - return - sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['tenant_id'] - pool = execute(cursor, sql) - sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] - tenant_unit = execute(cursor, sql) - max_memory = tenant_unit['max_memory'] - max_cpu = int(tenant_unit['max_cpu']) - except Exception as e: - stdio.verbose(e) - stdio.error('Fail to get tenant info') + tenant_meta = cursor.fetchone(sql, [tenant_name]) + if not tenant_meta: + stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) + return + sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['tenant_id'] + pool = cursor.fetchone(sql) + if pool is False: return + sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] + tenant_unit = cursor.fetchone(sql) + if tenant_unit is False: + return + max_memory = tenant_unit['max_memory'] + max_cpu = int(tenant_unit['max_cpu']) host = get_option('host', '127.0.0.1') port = get_option('port', 2881) @@ -219,9 +202,14 @@ def local_execute_command(command, env=None, timeout=None): test_only = get_option('test_only') stdio.verbose('Check connect ready') - exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A %s -e" % ( - obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '', db_name) - ret = local_execute_command('%s "%s" -E' % (exec_sql_cmd, 'select version();')) + if not test_only: + exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A -e" % ( + obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '') + ret = local_execute_command('%s "%s" -E' % (exec_sql_cmd, 'create database if not exists %s' % db_name)) + else: + exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A %s -e" % ( + obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '', db_name) + ret = local_execute_command('%s "%s" -E' % (exec_sql_cmd, 'select version();')) if not ret: stdio.error('Connect to tenant %s failed' % tenant_name) return diff --git a/plugins/tpcc/3.1.0/run_test.py b/plugins/tpcc/3.1.0/run_test.py index 1a58ba8..e6526fc 100644 --- a/plugins/tpcc/3.1.0/run_test.py +++ b/plugins/tpcc/3.1.0/run_test.py @@ -43,18 +43,6 @@ def get_option(key, default=''): value = default return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -97,21 +85,31 @@ def local_execute_command(command, env=None, timeout=None): stdio.exception('') return - merge_version = execute(cursor, "select value from oceanbase.__all_zone where name='frozen_version'")['value'] + merge_version = cursor.fetchone("select value from oceanbase.__all_zone where name='frozen_version'") + if merge_version is False: + return + merge_version = merge_version['value'] stdio.start_loading('Merge') - execute(cursor, 'alter system major freeze') + if cursor.fetchone('alter system major freeze') is False: + return sql = "select value from oceanbase.__all_zone where name='frozen_version' and value != %s" % merge_version while True: - if execute(cursor, sql): + res = cursor.fetchone(sql) + if res is False: + return + if res: break time.sleep(1) while True: - if not execute(cursor, """select * from oceanbase.__all_zone - where name='last_merged_version' - and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) - and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') - """): + res = cursor.fetchone("""select * from oceanbase.__all_zone + where name='last_merged_version' + and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) + and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') + """) + if res is False: + return + if not res: break time.sleep(5) stdio.stop_loading('succeed') diff --git a/plugins/tpcc/4.0.0.0/analyze.sql b/plugins/tpcc/4.0.0.0/analyze.sql index e180f5b..d75e11a 100644 --- a/plugins/tpcc/4.0.0.0/analyze.sql +++ b/plugins/tpcc/4.0.0.0/analyze.sql @@ -1,10 +1,10 @@ -call dbms_stats.gather_table_stats('test', 'bmsql_warehouse', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_district', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_customer', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_new_order', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_oorder', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_order_line', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_stock', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_history', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_config', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'bmsql_item', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); \ No newline at end of file +call dbms_stats.gather_table_stats('{database}', 'bmsql_warehouse', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_district', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_customer', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_new_order', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_oorder', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_order_line', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_stock', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_history', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_config', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'bmsql_item', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); \ No newline at end of file diff --git a/plugins/tpcc/4.0.0.0/build.py b/plugins/tpcc/4.0.0.0/build.py index ef80663..38214df 100644 --- a/plugins/tpcc/4.0.0.0/build.py +++ b/plugins/tpcc/4.0.0.0/build.py @@ -42,18 +42,6 @@ def get_option(key, default=''): def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def run_sql(sql_file, force=False): sql_cmd = "{obclient} -h{host} -P{port} -u{user}@{tenant} {password_arg} -A {db} {force_flag} < {sql_file}".format( obclient=obclient_bin, host=host, port=port, user=user, tenant=tenant_name, @@ -111,36 +99,33 @@ def get_table_rows(table_name): stdio.exception('') return stdio.start_loading('Server check') - try: - # check for observer + # check for observer + while True: + sql = "select * from oceanbase.DBA_OB_SERVERS where STATUS != 'ACTIVE' or STOP_TIME is not NULL or START_SERVICE_TIME is NULL" + ret = cursor.fetchone(sql) + if ret is False: + stdio.stop_loading('fail') + return + if ret is None: + break + time.sleep(3) + # check for obproxy + if odp_cursor: while True: - sql = "select * from oceanbase.DBA_OB_SERVERS where STATUS != 'ACTIVE' or STOP_TIME is not NULL or START_SERVICE_TIME is NULL" - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - ret = cursor.fetchone() - if ret is None: - break - time.sleep(3) - # check for obproxy - if odp_cursor: - while True: - sql = "show proxycongestion all" - stdio.verbose('execute obproxy sql: %s' % sql) - odp_cursor.execute(sql) - proxy_congestions = odp_cursor.fetchall() - passed = True - for proxy_congestion in proxy_congestions: - if proxy_congestion.get('dead_congested') != 0 or proxy_congestion.get('server_state') != 'ACTIVE': - passed = False - break - if passed: + sql = "show proxycongestion all" + proxy_congestions = odp_cursor.fetchall(sql) + if proxy_congestions is False: + stdio.stop_loading('fail') + return + passed = True + for proxy_congestion in proxy_congestions: + if proxy_congestion.get('dead_congested') != 0 or proxy_congestion.get('server_state') not in ['DETECT_ALIVE', 'ACTIVE']: + passed = False break - else: - time.sleep(3) - except: - stdio.stop_loading('fail') - stdio.exception('') - return + if passed: + break + else: + time.sleep(3) stdio.stop_loading('succeed') # drop old tables @@ -152,25 +137,32 @@ def get_table_rows(table_name): # Major freeze stdio.start_loading('Merge') sql_frozen_scn = "select FROZEN_SCN, LAST_SCN from oceanbase.CDB_OB_MAJOR_COMPACTION where tenant_id = %s" % tenant_id - merge_version = execute(cursor, sql_frozen_scn)['FROZEN_SCN'] - stdio.verbose('merge version is: %s' % merge_version) - execute(cursor, "alter system major freeze tenant = %s" % tenant_name) + merge_version = cursor.fetchone(sql_frozen_scn) + if merge_version is False: + return + merge_version = merge_version['FROZEN_SCN'] + if cursor.fetchone("alter system major freeze tenant = %s" % tenant_name) is False: + return # merge version changed while True: - current_version = execute(cursor, sql_frozen_scn).get("FROZEN_SCN") + current_version = cursor.fetchone(sql_frozen_scn) + if current_version is False: + return + current_version = current_version['FROZEN_SCN'] if int(current_version) > int(merge_version): break time.sleep(5) stdio.verbose('current merge version is: %s' % current_version) # version updated while True: - ret = execute(cursor, sql_frozen_scn) + ret = cursor.fetchone(sql_frozen_scn) + if ret is False: + return if int(ret.get("FROZEN_SCN", 0)) / 1000 == int(ret.get("LAST_SCN", 0)) / 1000: break time.sleep(5) stdio.stop_loading('succeed') - # create new tables if not run_sql(sql_file=os.path.join(bmsql_sql_path, 'tableCreates.sql')): stdio.error('create tables failed') diff --git a/plugins/tpcc/4.0.0.0/pre_test.py b/plugins/tpcc/4.0.0.0/pre_test.py index 85848a1..6ea230b 100644 --- a/plugins/tpcc/4.0.0.0/pre_test.py +++ b/plugins/tpcc/4.0.0.0/pre_test.py @@ -59,18 +59,6 @@ def get_option(key, default=''): stdio.verbose('get option: {} value {}'.format(key, value)) return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -162,9 +150,7 @@ def local_execute_command(command, env=None, timeout=None): min_cpu = None try: sql = "select b.CPU_CAPACITY from oceanbase.DBA_OB_SERVERS a join oceanbase.GV$OB_SERVERS b on a.SVR_IP=b.SVR_IP and a.SVR_PORT = b.SVR_PORT where a.STATUS = 'ACTIVE' and a.STOP_TIME is NULL and a.START_SERVICE_TIME > 0" - stdio.verbose('execute sql: %s' % sql) - cursor.execute(sql) - all_services = cursor.fetchall() + all_services = cursor.fetchall(sql) if not all_services: stdio.error('No active server available.') return @@ -187,23 +173,20 @@ def local_execute_command(command, env=None, timeout=None): local_execute_command("sed -i 's/{{partition_num}}/%d/g' %s" % (cpu_total, create_table_sql)) sql = "select * from oceanbase.DBA_OB_TENANTS where TENANT_NAME = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant_meta = cursor.fetchone() - if not tenant_meta: - stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) - return - sql = "select * from oceanbase.DBA_OB_RESOURCE_POOLS where TENANT_ID = %d" % tenant_meta['TENANT_ID'] - pool = execute(cursor, sql) - sql = "select * from oceanbase.DBA_OB_UNIT_CONFIGS where UNIT_CONFIG_ID = %d" % pool['UNIT_CONFIG_ID'] - tenant_unit = execute(cursor, sql) - max_memory = tenant_unit['MEMORY_SIZE'] - max_cpu = int(tenant_unit['MAX_CPU']) - except Exception as e: - stdio.verbose(e) - stdio.error('fail to get tenant info') + tenant_meta = cursor.fetchone(sql, [tenant_name]) + if not tenant_meta: + stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) + return + sql = "select * from oceanbase.DBA_OB_RESOURCE_POOLS where TENANT_ID = %d" % tenant_meta['TENANT_ID'] + pool = cursor.fetchone(sql) + if pool is False: return + sql = "select * from oceanbase.DBA_OB_UNIT_CONFIGS where UNIT_CONFIG_ID = %d" % pool['UNIT_CONFIG_ID'] + tenant_unit = cursor.fetchone(sql) + if tenant_unit is False: + return + max_memory = tenant_unit['MEMORY_SIZE'] + max_cpu = int(tenant_unit['MAX_CPU']) host = get_option('host', '127.0.0.1') port = get_option('port', 2881) @@ -217,9 +200,14 @@ def local_execute_command(command, env=None, timeout=None): test_only = get_option('test_only') stdio.verbose('Check connect ready') - exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A %s -e" % ( - obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '', db_name) - ret = local_execute_command('%s "%s" -E' % (exec_sql_cmd, 'select version();')) + if not test_only: + exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A -e" % ( + obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '') + ret = local_execute_command('%s "%s" -E' % (exec_sql_cmd, 'create database if not exists %s' % db_name)) + else: + exec_sql_cmd = "%s -h%s -P%s -u%s@%s %s -A %s -e" % ( + obclient_bin, host, port, user, tenant_name, ("-p'%s'" % password) if password else '', db_name) + ret = local_execute_command('%s "%s" -E' % (exec_sql_cmd, 'select version();')) if not ret: stdio.error('Connect to tenant %s failed' % tenant_name) return diff --git a/plugins/tpcc/4.0.0.0/run_test.py b/plugins/tpcc/4.0.0.0/run_test.py index 290f60c..57f4696 100644 --- a/plugins/tpcc/4.0.0.0/run_test.py +++ b/plugins/tpcc/4.0.0.0/run_test.py @@ -43,18 +43,6 @@ def get_option(key, default=''): value = default return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -106,15 +94,24 @@ def local_execute_command(command, env=None, timeout=None): # Major freeze stdio.start_loading('Merge') sql_frozen_scn = "select FROZEN_SCN, LAST_SCN from oceanbase.CDB_OB_MAJOR_COMPACTION where tenant_id = %s" % tenant_id - merge_version = execute(cursor, sql_frozen_scn)['FROZEN_SCN'] - execute(cursor, "alter system major freeze tenant = %s" % tenant_name) + merge_version = cursor.fetchone(sql_frozen_scn) + if merge_version is False: + return + merge_version = merge_version['FROZEN_SCN'] + if cursor.fetchone("alter system major freeze tenant = %s" % tenant_name) is False: + return while True: - current_version = execute(cursor, sql_frozen_scn).get("FROZEN_SCN") + current_version = cursor.fetchone(sql_frozen_scn) + if current_version is False: + return + current_version = current_version['FROZEN_SCN'] if int(current_version) > int(merge_version): break time.sleep(5) while True: - ret = execute(cursor, sql_frozen_scn) + ret = cursor.fetchone(sql_frozen_scn) + if ret is False: + return if int(ret.get("FROZEN_SCN", 0)) / 1000 == int(ret.get("LAST_SCN", 0)) / 1000: break time.sleep(5) @@ -130,7 +127,7 @@ def local_execute_command(command, env=None, timeout=None): analyze_path = os.path.join(local_dir, 'analyze.sql') with FileUtil.open(analyze_path, stdio=stdio) as f: content = f.read() - analyze_content = content.format(cpu_total=cpu_total) + analyze_content = content.format(cpu_total=cpu_total, database=db_name) ret = LocalClient.execute_command('%s """%s"""' % (exec_sql_cmd, analyze_content), stdio=stdio) if not ret: stdio.error('failed to analyze table: {}'.format(ret.stderr)) diff --git a/plugins/tpch/3.1.0/pre_test.py b/plugins/tpch/3.1.0/pre_test.py index f6707b4..def7161 100644 --- a/plugins/tpch/3.1.0/pre_test.py +++ b/plugins/tpch/3.1.0/pre_test.py @@ -68,18 +68,6 @@ def get_path(key, default): stdio.verbose('get %s_path: %s' % (key, path)) return path if path else default - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -128,28 +116,27 @@ def local_execute_command(command, env=None, timeout=None): setattr(options, 'tmp_dir', tmp_dir) sql = "select * from oceanbase.gv$tenant where tenant_name = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant_meta = cursor.fetchone() - if not tenant_meta: - stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) - return - sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['tenant_id'] - pool = execute(cursor, sql) - sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] - tenant_unit = execute(cursor, sql) - max_cpu = tenant_unit['max_cpu'] - min_memory = tenant_unit['min_memory'] - unit_count = pool['unit_count'] - except: - stdio.error('fail to get tenant info') + tenant_meta = cursor.fetchone(sql, [tenant_name]) + if not tenant_meta: + stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) + return + sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['tenant_id'] + pool = cursor.fetchone(sql) + if pool is False: return + sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] + tenant_unit = cursor.fetchone(sql) + if tenant_unit is False: + return + max_cpu = tenant_unit['max_cpu'] + min_memory = tenant_unit['min_memory'] + unit_count = pool['unit_count'] server_num = len(cluster_config.servers) sql = "select count(1) server_num from oceanbase.__all_server where status = 'active'" - ret = execute(cursor, sql) - if ret: - server_num = ret.get("server_num", server_num) + ret = cursor.fetchone(sql) + if ret is False: + return + server_num = ret.get("server_num", server_num) if get_option('test_only'): return plugin_context.return_true( diff --git a/plugins/tpch/3.1.0/run_test.py b/plugins/tpch/3.1.0/run_test.py index 5dcfcbd..bc50c5b 100644 --- a/plugins/tpch/3.1.0/run_test.py +++ b/plugins/tpch/3.1.0/run_test.py @@ -62,17 +62,6 @@ def get_option(key, default=''): if value is None: value = default return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -129,7 +118,10 @@ def local_execute_command(command, env=None, timeout=None): cpu_total += int(server_config.get('cpu_count', 0)) try: sql = "select value from oceanbase.__all_virtual_sys_variable where tenant_id = %d and name = 'secure_file_priv'" % tenant_id - ret = execute(cursor, sql)['value'] + ret = cursor.fetchone(sql) + if ret is False: + return + ret = ret['value'] if ret is None: stdio.error('Access denied. Please set `secure_file_priv` to "".') return @@ -173,21 +165,30 @@ def local_execute_command(command, env=None, timeout=None): raise Exception(ret.stderr) stdio.stop_loading('succeed') - merge_version = execute(cursor, "select value from oceanbase.__all_zone where name='frozen_version'")['value'] + merge_version = cursor.fetchone("select value from oceanbase.__all_zone where name='frozen_version'") + if merge_version is False: + return + merge_version = merge_version['value'] stdio.start_loading('Merge') - execute(cursor, 'alter system major freeze') + if cursor.fetchone('alter system major freeze') is False: + return sql = "select value from oceanbase.__all_zone where name='frozen_version' and value != %s" % merge_version while True: - if execute(cursor, sql): + res = cursor.fetchone(sql) + if res is False: + return + if res: break - time.sleep(1) while True: - if not execute(cursor, """select * from oceanbase.__all_zone - where name='last_merged_version' - and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) - and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') - """): + res = cursor.fetchone("""select * from oceanbase.__all_zone + where name='last_merged_version' + and value != (select value from oceanbase.__all_zone where name='frozen_version' limit 1) + and zone in (select zone from oceanbase.__all_zone where name='status' and info = 'ACTIVE') + """) + if res is False: + return + if not res: break time.sleep(5) stdio.stop_loading('succeed') diff --git a/plugins/tpch/4.0.0.0/analyze.sql b/plugins/tpch/4.0.0.0/analyze.sql index 99946b3..1e2c758 100644 --- a/plugins/tpch/4.0.0.0/analyze.sql +++ b/plugins/tpch/4.0.0.0/analyze.sql @@ -1,8 +1,8 @@ -call dbms_stats.gather_table_stats('test', 'lineitem', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'orders', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'partsupp', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'part', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'customer', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'supplier', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'nation', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); -call dbms_stats.gather_table_stats('test', 'region', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); \ No newline at end of file +call dbms_stats.gather_table_stats('{database}', 'lineitem', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'orders', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'partsupp', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'part', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'customer', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'supplier', degree=>{cpu_total}, granularity=>'GLOBAL', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'nation', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); +call dbms_stats.gather_table_stats('{database}', 'region', degree=>{cpu_total}, granularity=>'AUTO', method_opt=>'FOR ALL COLUMNS SIZE AUTO'); \ No newline at end of file diff --git a/plugins/tpch/4.0.0.0/pre_test.py b/plugins/tpch/4.0.0.0/pre_test.py index 2fa8a5b..f42457a 100644 --- a/plugins/tpch/4.0.0.0/pre_test.py +++ b/plugins/tpch/4.0.0.0/pre_test.py @@ -58,18 +58,6 @@ def get_option(key, default=''): stdio.verbose('get option: %s value %s' % (key, value)) return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) - def get_path(key, default): path = get_option('%s_path' % key) if path and os.path.exists(path): @@ -131,29 +119,28 @@ def local_execute_command(command, env=None, timeout=None): server_num = len(cluster_config.servers) sql = "select * from oceanbase.DBA_OB_TENANTS where TENANT_NAME = %s" - try: - stdio.verbose('execute sql: %s' % (sql % tenant_name)) - cursor.execute(sql, [tenant_name]) - tenant_meta = cursor.fetchone() - if not tenant_meta: - stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) - return - sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['TENANT_ID'] - pool = execute(cursor, sql) - sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] - tenant_unit = execute(cursor, sql) - max_cpu = tenant_unit['max_cpu'] - min_memory = MIN_MEMORY - unit_count = pool['unit_count'] - except: - stdio.exception('') - stdio.error('fail to get tenant info') + stdio.verbose('execute sql: %s' % (sql % tenant_name)) + tenant_meta = cursor.fetchone(sql, [tenant_name]) + if not tenant_meta: + stdio.error('Tenant %s not exists. Use `obd cluster tenant create` to create tenant.' % tenant_name) + return + sql = "select * from oceanbase.__all_resource_pool where tenant_id = %d" % tenant_meta['TENANT_ID'] + pool = cursor.fetchone(sql) + if pool is False: return + sql = "select * from oceanbase.__all_unit_config where unit_config_id = %d" % pool['unit_config_id'] + tenant_unit = cursor.fetchone(sql) + if tenant_unit is False: + return + max_cpu = tenant_unit['max_cpu'] + min_memory = MIN_MEMORY + unit_count = pool['unit_count'] server_num = len(cluster_config.servers) sql = "select count(1) server_num from oceanbase.__all_server where status = 'active'" - ret = execute(cursor, sql) - if ret: - server_num = ret.get("server_num", server_num) + ret = cursor.fetchone(sql) + if ret is False: + return + server_num = ret.get("server_num", server_num) if get_option('test_only'): return plugin_context.return_true( diff --git a/plugins/tpch/4.0.0.0/run_test.py b/plugins/tpch/4.0.0.0/run_test.py index 64264fb..749253e 100644 --- a/plugins/tpch/4.0.0.0/run_test.py +++ b/plugins/tpch/4.0.0.0/run_test.py @@ -63,17 +63,6 @@ def get_option(key, default=''): if value is None: value = default return value - def execute(cursor, query, args=None): - msg = query % tuple(args) if args is not None else query - stdio.verbose('execute sql: %s' % msg) - stdio.verbose("query: %s. args: %s" % (query, args)) - try: - cursor.execute(query, args) - return cursor.fetchone() - except: - msg = 'execute sql exception: %s' % msg - stdio.exception(msg) - raise Exception(msg) def local_execute_command(command, env=None, timeout=None): return LocalClient.execute_command(command, env, timeout, stdio) @@ -132,7 +121,10 @@ def local_execute_command(command, env=None, timeout=None): try: sql = "select value from oceanbase.__all_virtual_sys_variable where tenant_id = %d and name = 'secure_file_priv'" % tenant_id - ret = execute(cursor, sql)['value'] + ret = cursor.fetchone(sql) + if ret is False: + return + ret = ret['value'] if ret is None: stdio.error('Access denied. Please set `secure_file_priv` to "".') return @@ -179,15 +171,24 @@ def local_execute_command(command, env=None, timeout=None): # Major freeze stdio.start_loading('Merge') sql_frozen_scn = "select FROZEN_SCN, LAST_SCN from oceanbase.CDB_OB_MAJOR_COMPACTION where tenant_id = %s" % tenant_id - merge_version = execute(cursor, sql_frozen_scn)['FROZEN_SCN'] - execute(cursor, "alter system major freeze tenant = %s" % tenant_name) + merge_version = cursor.fetchone(sql_frozen_scn) + if merge_version is False: + return + merge_version = merge_version['FROZEN_SCN'] + if cursor.fetchone("alter system major freeze tenant = %s" % tenant_name) is False: + return while True: - current_version = execute(cursor, sql_frozen_scn).get("FROZEN_SCN") + current_version = cursor.fetchone(sql_frozen_scn) + if current_version is False: + return + current_version = current_version['FROZEN_SCN'] if int(current_version) > int(merge_version): break time.sleep(5) while True: - ret = execute(cursor, sql_frozen_scn) + ret = cursor.fetchone(sql_frozen_scn) + if ret is False: + return if int(ret.get("FROZEN_SCN", 0)) / 1000 == int(ret.get("LAST_SCN", 0)) / 1000: break time.sleep(5) @@ -203,7 +204,7 @@ def local_execute_command(command, env=None, timeout=None): analyze_path = os.path.join(local_dir, 'analyze.sql') with FileUtil.open(analyze_path, stdio=stdio) as f: content = f.read() - analyze_content = content.format(cpu_total=cpu_total) + analyze_content = content.format(cpu_total=cpu_total, database=mysql_db) ret = LocalClient.execute_command('%s -e """%s"""' % (sql_cmd_prefix, analyze_content), stdio=stdio) if not ret: raise Exception(ret.stderr) diff --git a/profile/obd.sh b/profile/obd.sh index 1aa5028..24cdb09 100644 --- a/profile/obd.sh +++ b/profile/obd.sh @@ -52,10 +52,10 @@ function _obd_complete_func cur="${COMP_WORDS[COMP_CWORD]}" prev="${COMP_WORDS[COMP_CWORD-1]}" - all_cmds["obd"]="mirror cluster test update repo demo" + all_cmds["obd"]="mirror cluster test update repo demo web" all_cmds["obd cluster"]="autodeploy tenant start deploy redeploy restart reload destroy stop edit-config list display upgrade chst check4ocp reinstall" all_cmds["obd cluster *"]="_obd_reply_deploy_names" - all_cmds["obd cluster tenant"]="create drop" + all_cmds["obd cluster tenant"]="create drop show" all_cmds["obd cluster tenant *"]="_obd_reply_deploy_names" all_cmds["obd mirror"]="clone create list update enable disable" all_cmds["obd mirror clone"]="_obd_reply_current_files" @@ -63,7 +63,7 @@ function _obd_complete_func all_cmds["obd test"]="mysqltest sysbench tpch tpcc" all_cmds["obd test *"]="_obd_reply_deploy_names" - if [ -f "$env_file" ] && [ "$(grep '"OBD_DEV_MODE": "1"' "$env_file")" != "" ]; then + # if [ -f "$env_file" ] && [ "$(grep '"OBD_DEV_MODE": "1"' "$env_file")" != "" ]; then all_cmds["obd"]="${all_cmds[obd]} devmode env tool" all_cmds["obd devmode"]="enable disable" all_cmds["obd tool"]="command db_connect dooba" @@ -72,7 +72,7 @@ function _obd_complete_func all_cmds["obd tool command"]="_obd_reply_deploy_names" all_cmds["obd tool command *"]="_obd_reply_tool_commands" all_cmds["obd env"]="set unset show clear" - fi + # fi case $prev in list) return 0 diff --git a/rpm/build.sh b/rpm/build.sh index 71ef694..3216a2a 100755 --- a/rpm/build.sh +++ b/rpm/build.sh @@ -72,6 +72,15 @@ function pacakge_obd() rm -fr rpmbuild } +function package_web() +{ + cd2workdir + DIR=`pwd`/../web + cd $DIR + yarn + yarn build +} + function get_python() { if [ `id -u` != 0 ] ; then @@ -167,4 +176,7 @@ case "x$1" in get_python build ;; -esac \ No newline at end of file + xweb) + package_web + ;; +esac diff --git a/rpm/ob-deploy-build.sh b/rpm/ob-deploy-build.sh index 4fb1551..99be94e 100755 --- a/rpm/ob-deploy-build.sh +++ b/rpm/ob-deploy-build.sh @@ -7,8 +7,8 @@ RELEASE=$4 PYTHON3_SWITCH=$5 if [[ x"$PYTHON3_SWITCH" == x"" ]]; then - echo "No switch command is provided, so use the default switch command: 'source /environments/python3_env/bin/activate'" - PYTHON3_SWITCH="source /environments/python3_env/bin/activate" + echo "No switch command is provided, so use the default switch command: 'source py-env-activate py38'" + PYTHON3_SWITCH="source py-env-activate py38" fi CURDIR=$PWD diff --git a/rpm/ob-deploy.spec b/rpm/ob-deploy.spec index 46fd4e2..1bd2c18 100644 --- a/rpm/ob-deploy.spec +++ b/rpm/ob-deploy.spec @@ -54,16 +54,24 @@ VERSION="$RPM_PACKAGE_VERSION" if [ "$OBD_DUBUG" ]; then VERSION=$VERSION".`date +%s`" fi +cd $SRC_DIR/web +yarn +yarn build +cd $SRC_DIR cat _cmd.py | sed "s//$CID/" | sed "s//$BRANCH/" | sed "s//$DATE/" | sed "s//$OBD_DUBUG/" | sed "s//$VERSION/" > obd.py sed -i "s||$OBD_DOC_LINK|" _errno.py mkdir -p $BUILD_DIR/SOURCES ${RPM_BUILD_ROOT} mkdir -p $BUILD_DIR/SOURCES/{site-packages} mkdir -p ${RPM_BUILD_ROOT}/usr/bin mkdir -p ${RPM_BUILD_ROOT}/usr/obd -pip install -r plugins-requirements3.txt --target=$BUILD_DIR/SOURCES/site-packages -pyinstaller --hidden-import=decimal --hidden-import=configparser -F obd.py +pip install -r plugins-requirements3.txt --target=$BUILD_DIR/SOURCES/site-packages -i http://mirrors.aliyun.com/pypi/simple/ --trusted-host mirrors.aliyun.com +pip install -r service/service-requirements.txt --target=$BUILD_DIR/SOURCES/site-packages -i http://mirrors.aliyun.com/pypi/simple/ --trusted-host mirrors.aliyun.com +# pyinstaller -y --clean -n obd-web -p $BUILD_DIR/SOURCES/site-packages -F service/app.py +pyinstaller --hidden-import=decimal -p $BUILD_DIR/SOURCES/site-packages --hidden-import service/app.py --hidden-import=configparser -F obd.py rm -f obd.py obd.spec +\mkdir -p $BUILD_DIR/SOURCES/web \cp -rf $SRC_DIR/dist/obd ${RPM_BUILD_ROOT}/usr/bin/obd +\cp -rf $SRC_DIR/web/dist $BUILD_DIR/SOURCES/web \cp -rf $SRC_DIR/plugins $BUILD_DIR/SOURCES/plugins \cp -rf $SRC_DIR/optimize $BUILD_DIR/SOURCES/optimize \cp -rf $SRC_DIR/example $BUILD_DIR/SOURCES/example @@ -72,6 +80,7 @@ rm -f obd.py obd.spec \rm -fr $BUILD_DIR/SOURCES/config_parser/oceanbase-ce \cp -rf $SRC_DIR/profile/ $BUILD_DIR/SOURCES/ \cp -rf $SRC_DIR/mirror/ $BUILD_DIR/SOURCES/ +\cp -rf $BUILD_DIR/SOURCES/web ${RPM_BUILD_ROOT}/usr/obd/ \cp -rf $BUILD_DIR/SOURCES/plugins ${RPM_BUILD_ROOT}/usr/obd/ \cp -rf $BUILD_DIR/SOURCES/optimize ${RPM_BUILD_ROOT}/usr/obd/ \cp -rf $BUILD_DIR/SOURCES/config_parser ${RPM_BUILD_ROOT}/usr/obd/ @@ -120,6 +129,15 @@ echo -e 'Installation of obd finished successfully\nPlease source /etc/profile.d #/sbin/chkconfig obd on %changelog +* Wed Dec 14 2022 obd 1.6.2 + - new features: support OceanBaseCE BP upgrade + - fix bug: grafana init failed when remote deploy +* Thu Nov 24 2022 obd 1.6.1 + - new features: minimum startup resource check + - fix bug: grafana dashboard title + - fix bug: autodeploy maybe failed in the case of large memory and small disk + - fix bug: obproxy frequent core dump in demo + - fix bug: remote install rsync transmission does not use the user.port * Mon Oct 31 2022 obd 1.6.0 - new features: support oceanbase 4.0 - new features: support Prometheus diff --git a/service/__init__.py b/service/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/service/api/__init__.py b/service/api/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/service/api/response.py b/service/api/response.py new file mode 100644 index 0000000..8447d76 --- /dev/null +++ b/service/api/response.py @@ -0,0 +1,35 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from typing import TypeVar, Generic, Optional, List +from pydantic.generics import GenericModel + +Data = TypeVar('Data') + + +class OBResponse(GenericModel, Generic[Data]): + code: int = 200 + data: Optional[Data] = None + msg: str = '' + success: bool = True + + +class DataList(GenericModel, Generic[Data]): + total: int = 0 + items: List[Data] = [] diff --git a/service/api/response_utils.py b/service/api/response_utils.py new file mode 100644 index 0000000..629bf41 --- /dev/null +++ b/service/api/response_utils.py @@ -0,0 +1,64 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import traceback +from service.api.response import OBResponse, DataList +from fastapi import HTTPException +from http import HTTPStatus +from service.common import log + + +def new_ok_response(data): + response = OBResponse() + response.code = HTTPStatus.OK + response.msg = "successful" + response.success = True + if isinstance(data, list): + data_list = DataList() + data_list.total = len(data) + data_list.items = data + response.data = data_list + else: + response.data = data + return response + + +def new_bad_request_exception(ex): + log.get_logger().error("got bad request exception: {0}".format(traceback.format_exc())) + raise HTTPException(HTTPStatus.BAD_REQUEST, detail="bad request, exception: {0}".format(ex)) + + +def new_not_found_exception(ex): + log.get_logger().error("got not found exception: {0}".format(traceback.format_exc())) + raise HTTPException(HTTPStatus.NOT_FOUND, detail="resource not found, exception: {0}".format(ex)) + + +def new_internal_server_error_exception(ex): + log.get_logger().error("got internal server error exception: {0}".format(traceback.format_exc())) + raise HTTPException(HTTPStatus.INTERNAL_SERVER_ERROR, detail="internal server error, exception: {0}".format(ex)) + + +def new_service_unavailable_exception(ex): + log.get_logger().error("got service unavailable exception: {0}".format(traceback.format_exc())) + raise HTTPException(HTTPStatus.SERVICE_UNAVAILABLE, detail="service unavailable, exception: {0}".format(ex)) + + +def new_not_implemented_exception(ex): + log.get_logger().error("got not implemented exception: {0}".format(traceback.format_exc())) + raise HTTPException(HTTPStatus.NOT_IMPLEMENTED, detail="not implemented, exception: {0}".format(ex)) diff --git a/service/api/v1/__init__.py b/service/api/v1/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/service/api/v1/components.py b/service/api/v1/components.py new file mode 100644 index 0000000..a47d325 --- /dev/null +++ b/service/api/v1/components.py @@ -0,0 +1,76 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from typing import List + +from fastapi import APIRouter, Path + +from service.api.response import OBResponse, DataList + +from service.model.components import Component, ComponentInfo, ParameterMeta, ParameterRequest, ParameterFilter + +import service.api.response_utils as response_utils +import service.handler.handler_utils as handler_utils +from service.common import log + +router = APIRouter() + + +@router.post("/components/parameters", + response_model=OBResponse[DataList[ParameterMeta]], + description='query component parameters', + operation_id='queryComponentParameters', + tags=['Components']) +async def list_component_parameters(parameter_request: ParameterRequest = ...): + handler = handler_utils.new_component_handler() + parameters = handler.list_component_parameters(parameter_request) + return response_utils.new_ok_response(parameters) + + +@router.get("/components/{component}", + response_model=OBResponse[Component], + description='query component by component name', + tags=['Components'], + operation_id='queryComponentByComponentName') +async def get_component(component: str = Path(description='component name')): + handler = handler_utils.new_component_handler() + try: + ret = handler.get_component(component) + if ret is None: + return response_utils.new_not_found_exception(Exception("component {0} not found".format(component))) + else: + return response_utils.new_ok_response(ret) + except Exception as ex: + return response_utils.new_service_unavailable_exception(ex) + + +@router.get("/components", + response_model=OBResponse[DataList[Component]], + description='query all component versions', + operation_id='queryAllComponentVersions', + tags=['Components']) +async def list_components(): + handler = handler_utils.new_component_handler() + try: + components = handler.list_components() + return response_utils.new_ok_response(components) + except Exception as ex: + return response_utils.new_service_unavailable_exception(ex) + + diff --git a/service/api/v1/deployments.py b/service/api/v1/deployments.py new file mode 100644 index 0000000..2194d21 --- /dev/null +++ b/service/api/v1/deployments.py @@ -0,0 +1,214 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from fastapi import APIRouter, Path, Query, BackgroundTasks + +from service.api import response_utils +from service.api.response import OBResponse, DataList +from service.handler import handler_utils +from service.model.deployments import DeploymentConfig, PreCheckResult, RecoverChangeParameter, TaskInfo, \ + ConnectionInfo, InstallLog, Deployment, DeploymentInfo, DeploymentReport, DeploymentStatus + +router = APIRouter() + + +@router.post("/deployments/{name}", + response_model=OBResponse, + description='create deployment config', + operation_id='createDeploymentConfig', + tags=['Deployments']) +async def create_deployment(name: str = Path(description='name'), + config: DeploymentConfig = ...): + handler = handler_utils.new_deployment_handler() + cluster = None + try: + oceanbase_config_path = handler.generate_deployment_config(name, config) + cluster = handler.create_deployment(name, oceanbase_config_path) + except Exception as ex: + return response_utils.new_internal_server_error_exception(ex) + if cluster: + return response_utils.new_ok_response(cluster) + else: + return response_utils.new_bad_request_exception(Exception('deployment {0} already exists'.format(name))) + + +@router.post("/deployments/{name}/precheck", + response_model=OBResponse, + description='pre-check, asynchronous process', + operation_id='pre-check', + tags=['Deployments']) +async def pre_check(name: str, background_tasks: BackgroundTasks): + handler = handler_utils.new_deployment_handler() + try: + handler.precheck(name, background_tasks) + except Exception as ex: + return response_utils.new_internal_server_error_exception(ex) + return response_utils.new_ok_response("precheck for {0}".format(name)) + + +@router.get("/deployments/{name}/precheck", + response_model=OBResponse[PreCheckResult], + description='select pre-check status by pre deployment name', + operation_id='preCheckStatus', + tags=['Deployments']) +async def get_pre_check_status(name: str = Path(description='deployment name')): + handler = handler_utils.new_deployment_handler() + precheck_result = handler.get_precheck_result(name) + return response_utils.new_ok_response(precheck_result) + + +@router.post("/deployments/{name}/recover", + response_model=OBResponse[DataList[RecoverChangeParameter]], + description='recover', + operation_id='recover', + tags=['Deployments']) +async def recover(name: str = Path(description='deployment name')): + handler = handler_utils.new_deployment_handler() + try: + recover_result = handler.recover(name) + return response_utils.new_ok_response(recover_result) + except Exception as ex: + return response_utils.new_internal_server_error_exception(ex) + + +@router.post("/deployments/{name}/install", + response_model=OBResponse, + description='deploy and start a deployment', + operation_id='deployAndStartADeployment', + tags=['Deployments']) +async def install(name: str, background_tasks: BackgroundTasks): + handler = handler_utils.new_deployment_handler() + try: + handler.install(name, background_tasks) + except Exception as ex: + return response_utils.new_internal_server_error_exception(ex) + return response_utils.new_ok_response("") + + +@router.get("/deployments/{name}/install", + response_model=OBResponse[TaskInfo], + description='query install status', + operation_id='queryInstallStatus', + tags=['Deployments']) +async def get_install_status(name: str = Path(description='deployment name')): + handler = handler_utils.new_deployment_handler() + task_info = handler.get_install_task_info(name) + if task_info is None: + return response_utils.new_not_found_exception("task {0} not found".format(name)) + return response_utils.new_ok_response(task_info) + + +@router.get("/deployments/{name}/connection", + response_model=OBResponse[DataList[ConnectionInfo]], + description='query connect info', + operation_id='queryConnectionInfo', + tags=['Deployments']) +async def get_connect_info(name: str = Path(description='deployment name')): + handler = handler_utils.new_deployment_handler() + connection_info_list = handler.list_connection_info(name) + if connection_info_list is None: + return response_utils.new_not_found_exception(Exception("deployment {0} not found".format(name))) + else: + return response_utils.new_ok_response(connection_info_list) + + +@router.get("/deployments/{name}/install/log", + response_model=OBResponse[InstallLog], + description='query install log', + operation_id='queryInstallLog', + tags=['Deployments']) +async def get_install_log(name: str = Path(description='deployment name'), + offset: int = Query(None, description='log offset')): + handler = handler_utils.new_deployment_handler() + task_info = handler.get_install_task_info(name) + if task_info is None: + return response_utils.new_not_found_exception("task {0} not found".format(name)) + log_content = handler.buffer.read() + log_info = InstallLog(log=log_content[offset:], offset=len(log_content)) + return response_utils.new_ok_response(log_info) + + +@router.get("/deployments", + response_model=OBResponse[DataList[Deployment]], + description='get deployment', + operation_id='getDeployment', + tags=['Deployments']) +async def get_deployments(task_status: DeploymentStatus = Query(..., description='task status,ex:INSTALLING,DRAFT')): + handler = handler_utils.new_deployment_handler() + deployments = handler.list_deployments_by_status(task_status) + return response_utils.new_ok_response(deployments) + + +@router.get("/deployments/{name}", + response_model=OBResponse[DeploymentInfo], + description='query deployment config', + operation_id='queryDeploymentConfig', + tags=['Deployments']) +async def get_deployment(name: str = Path(description='deployment name')): + handler = handler_utils.new_deployment_handler() + deployment = handler.get_deployment_by_name(name) + if deployment is None: + return response_utils.new_not_found_exception(Exception('deployment {} not found'.format(name))) + return response_utils.new_ok_response(deployment) + + +@router.get("/deployments/{name}/report", + response_model=OBResponse[DataList[DeploymentReport]], + description='query deployment report', + operation_id='queryDeploymentReport', + tags=['Deployments']) +async def get_deployment_report(name: str = Path(description='deployment name')): + handler = handler_utils.new_deployment_handler() + try: + report_list = handler.get_deployment_report(name) + except Exception as ex: + raise response_utils.new_bad_request_exception(ex) + return response_utils.new_ok_response(report_list) + + +@router.delete("/deployments/{name}", + response_model=OBResponse, + description='destroy deployment ', + operation_id='destroyDeployment ', + tags=['Deployments']) +async def destroy_deployment(name: str, background_tasks: BackgroundTasks): + handler = handler_utils.new_deployment_handler() + background_tasks.add_task(handler.destroy_cluster, name) + return response_utils.new_ok_response("") + + +@router.get("/deployments/{name}/destroy", + response_model=OBResponse[TaskInfo], + description='get destroy task info', + operation_id='getDestroyTaskInfo', + tags=['Deployments']) +async def get_destroy_task_info(name: str): + handler = handler_utils.new_deployment_handler() + info = handler.get_destroy_task_info(name) + return response_utils.new_ok_response(info) +@router.get("/deployments_test", + response_model=OBResponse, + description='get destroy task info', + operation_id='getDestroyTaskInfo', + tags=['Deployments']) +async def get_destroy_task_info(): + + return response_utils.new_ok_response('inknnsdlafasd') + + diff --git a/service/api/v1/mirror.py b/service/api/v1/mirror.py new file mode 100644 index 0000000..043ae7d --- /dev/null +++ b/service/api/v1/mirror.py @@ -0,0 +1,42 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from fastapi import APIRouter + + +from service.api import response_utils +from service.api.response import OBResponse, DataList +from service.handler import handler_utils +from service.model.mirror import Mirror + +router = APIRouter() + + +@router.get("/mirrors", + response_model=OBResponse[DataList[Mirror]], + description='list remote mirrors', + operation_id='listRemoteMirrors', + tags=['Mirror']) +async def get_effective_mirror(): + handler = handler_utils.new_mirror_handler() + try: + mirrors = handler.list_mirrors() + except Exception as e: + return response_utils.new_service_unavailable_exception(e) + return response_utils.new_ok_response(mirrors) diff --git a/service/api/v1/process.py b/service/api/v1/process.py new file mode 100644 index 0000000..a10d2f3 --- /dev/null +++ b/service/api/v1/process.py @@ -0,0 +1,38 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from fastapi import APIRouter + +from fastapi import BackgroundTasks +from service.api.response import OBResponse +from service.api import response_utils +from service.handler import handler_utils + +router = APIRouter() + + +@router.post("/processes/suicide", + response_model=OBResponse, + description='exit process', + operation_id='exitProcess', + tags=['Processes']) +async def suicide(backgroundtasks: BackgroundTasks): + handler = handler_utils.new_process_handler() + backgroundtasks.add_task(handler.suicide) + return response_utils.new_ok_response("suicide") diff --git a/service/api/v1/service_info.py b/service/api/v1/service_info.py new file mode 100644 index 0000000..0ffe200 --- /dev/null +++ b/service/api/v1/service_info.py @@ -0,0 +1,38 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from fastapi import APIRouter + +from service.api import response_utils +from service.api.response import OBResponse +from service.handler import handler_utils +from service.model.service_info import ServiceInfo + +router = APIRouter() + + +@router.get("/info", + response_model=OBResponse[ServiceInfo], + description='get obd info', + operation_id='getObdInfo', + tags=['Info']) +async def get_info(): + handler = handler_utils.new_service_info_handler() + service_info = handler.get_service_info() + return response_utils.new_ok_response(service_info) diff --git a/service/app.py b/service/app.py new file mode 100644 index 0000000..bd9f0d3 --- /dev/null +++ b/service/app.py @@ -0,0 +1,70 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import asyncio + +import uvicorn +from fastapi import FastAPI +from fastapi.middleware.gzip import GZipMiddleware +from starlette.staticfiles import StaticFiles +from starlette_prometheus import metrics, PrometheusMiddleware + +from asgi_correlation_id import CorrelationIdMiddleware + +from service.common import log +from service.common.core import CoreManager +from service.api.v1 import components, deployments, process, service_info, mirror +from service.middleware.request_response_log import RequestResponseLogMiddleware +from service.middleware.process_time import ProcessTimeMiddleware +from service.handler import handler_utils +app = FastAPI() + + +class OBDWeb(object): + + def __init__(self, obd, resource_path): + CoreManager.INSTANCE = obd + self.app = app + self.app.add_route("/metrics", metrics) + self.app.include_router(components.router, prefix='/api/v1') + self.app.include_router(deployments.router, prefix='/api/v1') + self.app.include_router(process.router, prefix='/api/v1') + self.app.include_router(service_info.router, prefix='/api/v1') + self.app.include_router(mirror.router, prefix='/api/v1') + self.app.add_middleware(ProcessTimeMiddleware) + self.app.add_middleware(RequestResponseLogMiddleware, logger=log.get_logger()) + self.app.add_middleware(PrometheusMiddleware) + self.app.add_middleware(CorrelationIdMiddleware) + self.app.add_middleware(GZipMiddleware, minimum_size=1024) + self.app.mount("/", StaticFiles(directory="{0}/web/dist".format(resource_path), html=True), name="dist") + + + @staticmethod + async def init_mirrors(): + handler = handler_utils.new_mirror_handler() + await handler.init_mirrors_info() + + @staticmethod + @app.on_event("startup") + async def startup_event() -> None: + asyncio.create_task(OBDWeb.init_mirrors()) + + def start(self, port=8680): + uvicorn.run(self.app, host='0.0.0.0', port=port, log_level="debug", reload=False, log_config=log.get_logger_config(file_name="{0}/{1}".format(CoreManager.INSTANCE.home_path, "app.log"))) + diff --git a/service/common/__init__.py b/service/common/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/service/common/const.py b/service/common/const.py new file mode 100644 index 0000000..f47c334 --- /dev/null +++ b/service/common/const.py @@ -0,0 +1,60 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from collections import defaultdict + +MINIMAL_CONFIG = ''' +{0}: + global: + home_path: /root/oceanbase/oceanbase +''' + +PKG_ESTIMATED_SIZE = defaultdict(lambda:0) +PKG_ESTIMATED_SIZE.update({"oceanbase-ce":314142720, "obproxy-ce":45424640, "obagent": 25124864}) + + +OCEANBASE_CE = 'oceanbase-ce' +OCEANBASE = 'oceanbase' + +CE = "ce" +BUSINESS = "business" + +OBPROXY_CE = 'obproxy-ce' +OBPROXY = 'obproxy' + +OCP_EXPRESS = 'ocpexpress' + +OBAGENT = 'obagent' + +DESTROY_PLUGIN = "destroy" +INIT_PLUGINS = ("init",) +START_PLUGINS = ("start_check", "start", "connect", "bootstrap", "display") +# filter component of oceanbase and obproxy version above 4.0 +VERSION_FILTER = { + OCEANBASE: "4.0.0.0", + OCEANBASE_CE: "4.0.0.0", + OBPROXY: "4.0.0", + OBPROXY_CE: "4.0.0" +} + +RUNNING = 'running' +FINISHED = 'finished' + +GRACEFUL_TIMEOUT = 5 + diff --git a/service/common/core.py b/service/common/core.py new file mode 100644 index 0000000..e94f8c9 --- /dev/null +++ b/service/common/core.py @@ -0,0 +1,49 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . +from collections import defaultdict + +from singleton_decorator import singleton + +from _stdio import BufferIO + + +@singleton +class CoreManager(object): + + INSTANCE = None + + def __init__(self): + if CoreManager.INSTANCE is None: + raise Exception('CoreManager Uninitialized') + self._buffer = BufferIO(False) + CoreManager.INSTANCE.stdio.set_output_stream(self._buffer) + CoreManager.INSTANCE.stdio.set_input_stream(BufferIO(False)) + self._obd = CoreManager.INSTANCE + self._context = defaultdict(lambda: defaultdict(lambda: None)) + + def get_obd(self): + return self._obd + + def get_buffer(self): + return self._buffer + + def get_context(self): + return self._context + + diff --git a/service/common/log.py b/service/common/log.py new file mode 100644 index 0000000..a2fd67b --- /dev/null +++ b/service/common/log.py @@ -0,0 +1,66 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import logging.config + + +DEFAULT_LOGGER="DefaultLogger" + + +def get_logger_config(file_name="app.log", level="INFO"): + logger_config = { + 'version': 1, + 'disable_existing_loggers': False, + 'filters': { + 'correlation_id': { + '()': 'asgi_correlation_id.CorrelationIdFilter', + 'uuid_length': 32, + }, + }, + 'formatters': { + 'simple': { + 'class': 'logging.Formatter', + 'format': '%(asctime)s %(levelname)s %(funcName)s (%(filename)s:%(lineno)d) [%(correlation_id)s] %(message)s', + }, + }, + 'handlers': { + 'console': { + 'class': 'logging.StreamHandler', + 'filters': ['correlation_id'], + 'formatter': 'simple' + }, + 'file': { + 'class': 'logging.FileHandler', + 'filters': ['correlation_id'], + 'filename': file_name, + 'formatter': 'simple' + } + }, + 'loggers': { + DEFAULT_LOGGER: { + 'handlers': ['console', 'file'], + 'level': level + } + } + } + return logger_config + +def get_logger(): + return logging.getLogger(DEFAULT_LOGGER) + diff --git a/service/common/task.py b/service/common/task.py new file mode 100644 index 0000000..a22968f --- /dev/null +++ b/service/common/task.py @@ -0,0 +1,146 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import time +import functools +from threading import Lock +from collections import defaultdict +from singleton_decorator import singleton + +from enum import auto +from fastapi_utils.enums import StrEnum +from service.common import log + +DEFAULT_TASK_TYPE="undefined" + +def get_task_manager(): + return TaskManager() + + +class TaskStatus(StrEnum): + PENDING = auto() + RUNNING = auto() + FINISHED = auto() + + +class TaskResult(StrEnum): + SUCCESSFUL = auto() + FAILED = auto() + # running means task not finished, maybe define another name + RUNNING = auto() + + +class TaskInfo(object): + def __init__(self): + self.start_time = None + self.status = TaskStatus.PENDING + self.end_time = None + self.result = TaskResult.RUNNING + self.ret = None + self.exception = None + + def run(self): + self.status = TaskStatus.RUNNING + self.start_time = time.time() + + def finish(self): + self.status = TaskStatus.FINISHED + self.end_time = time.time() + + def success(self): + self.result = TaskResult.SUCCESSFUL + self.finish() + + def fail(self): + self.result = TaskResult.FAILED + self.finish() + + +@singleton +class TaskManager(object): + def __init__(self): + self.all_tasks = defaultdict(dict) + self.lock = Lock() + + def get_task_info(self, name, task_type=DEFAULT_TASK_TYPE): + ret = None + self.lock.acquire() + if name in self.all_tasks[task_type].keys(): + ret = self.all_tasks[task_type][name] + self.lock.release() + return ret + + def del_task_info(self, name, task_type=DEFAULT_TASK_TYPE): + ret = None + self.lock.acquire() + if name in self.all_tasks[task_type].keys(): + del(self.all_tasks[task_type][name]) + self.lock.release() + + def register_task(self, name, task_info, task_type=DEFAULT_TASK_TYPE): + self.lock.acquire() + log.get_logger().info("register task %s", name) + self.all_tasks[task_type][name] = task_info + self.lock.release() + + +class AutoRegister(object): + def __init__(self, task_type=DEFAULT_TASK_TYPE): + self._task_type = task_type + + def __call__(self, func): + @functools.wraps(func) + def wrapper(*args, **kwargs): + if len(args) < 2: + raise Exception("lack of parameter task_name") + name = args[1] + task_manager = get_task_manager() + task_info = TaskInfo() + task_manager.register_task(name, task_info, task_type=self._task_type) + try: + log.get_logger().info("start run task %s", name) + task_info.run() + task_info.ret = func(*args, **kwargs) + log.get_logger().info("task %s run finished", name) + task_info.success() + log.get_logger().info("task %s finished successful", name) + except BaseException as ex: + msg = "task {0} got exception".format(name) + log.get_logger().exception(msg) + task_info.exception = ex + task_info.fail() + log.get_logger().info("task %s finished failed", name) + return wrapper + + +class Serial(object): + def __init__(self, task_type=DEFAULT_TASK_TYPE): + self._task_type = task_type + self.lock = Lock() + + def __call__(self, func): + @functools.wraps(func) + def wrapper(*args, **kwargs): + self.lock.acquire() + try: + func(*args, **kwargs) + finally: + self.lock.release() + return wrapper + diff --git a/service/common/util.py b/service/common/util.py new file mode 100644 index 0000000..9b9151b --- /dev/null +++ b/service/common/util.py @@ -0,0 +1,26 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +def recursive_update_dict(a, b): + for key in b: + if isinstance(b[key], dict) and isinstance(a.get(key), dict): + a[key] = recursive_update_dict(a[key], b[key]) + else: + a[key] = b[key] + return a diff --git a/service/handler/__init__.py b/service/handler/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/service/handler/base_handler.py b/service/handler/base_handler.py new file mode 100644 index 0000000..56993a5 --- /dev/null +++ b/service/handler/base_handler.py @@ -0,0 +1,42 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from collections import defaultdict + +from _plugin import PluginContextNamespace +from service.common import core + +SPACENAME = "API" +class BaseHandler(object): + def __init__(self): + self._obd = core.CoreManager().get_obd() + self._buffer = core.CoreManager().get_buffer() + self._context = core.CoreManager().get_context() + + @property + def obd(self): + return self._obd + + @property + def buffer(self): + return self._buffer + + @property + def context(self): + return self._context diff --git a/service/handler/component_handler.py b/service/handler/component_handler.py new file mode 100644 index 0000000..1d55e6a --- /dev/null +++ b/service/handler/component_handler.py @@ -0,0 +1,181 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import uuid +import tempfile +from service.handler.base_handler import BaseHandler +from service.model.components import Component, ComponentInfo, ConfigParameter, ParameterMeta +from service.common import log +from _mirror import MirrorRepositoryType +from _plugin import PluginType +from _repository import Repository +from singleton_decorator import singleton +from collections import defaultdict +from _rpm import Version +from service.common import const + +def map_to_config_parameter(param): + log.get_logger().info("param {0} type: {1}".format(param.name, param._param_type.__name__)) + config_parameter = ConfigParameter() + config_parameter.auto = False + config_parameter.name = param.name + config_parameter.is_essential = param.essential + config_parameter.require = param.require + config_parameter.type = param._param_type.__name__ + config_parameter.default = str(param.default) if param.default is not None else "" + config_parameter.min_value = str(param.min_value) if param.min_value is not None else "" + config_parameter.max_value = str(param.max_value) if param.max_value is not None else "" + config_parameter.modify_limit = param.modify_limit.__name__ + config_parameter.need_restart = param.need_restart + config_parameter.need_redeploy = param.need_redeploy + config_parameter.need_reload = param.need_reload + config_parameter.section = param.section + config_parameter.description = param.description_local if param.description_local else param.description_en + return config_parameter + +@singleton +class ComponentHandler(BaseHandler): + + + def __get_all_components(self, component_filter=const.VERSION_FILTER): + local_packages = self.obd.mirror_manager.local_mirror.get_all_pkg_info() + remote_packages = list() + remote_mirrors = self.obd.mirror_manager.get_remote_mirrors() + for mirror in remote_mirrors: + remote_packages.extend(mirror.get_all_pkg_info()) + local_packages.sort() + remote_packages.sort() + local_pkg_idx = len(local_packages) - 1 + remote_pkg_idx = len(remote_packages) - 1 + component_dict = defaultdict(list) + while local_pkg_idx >= 0 and remote_pkg_idx >= 0: + local_pkg = local_packages[local_pkg_idx] + remote_pkg = remote_packages[remote_pkg_idx] + if local_pkg >= remote_pkg: + component_dict[local_pkg.name].append( + ComponentInfo(version=local_pkg.version, md5=local_pkg.md5, release=local_pkg.release, + arch=local_pkg.arch, type=MirrorRepositoryType.LOCAL.value, estimated_size=const.PKG_ESTIMATED_SIZE[local_pkg.name])) + local_pkg_idx -= 1 + else: + if len(component_dict[remote_pkg.name]) > 0 and component_dict[remote_pkg.name][-1].md5 == remote_pkg.md5: + log.get_logger().debug("already found local package %s", remote_pkg) + else: + component_dict[remote_pkg.name].append( + ComponentInfo(version=remote_pkg.version, md5=remote_pkg.md5, release=remote_pkg.release, + arch=remote_pkg.arch, type=MirrorRepositoryType.REMOTE.value, estimated_size=const.PKG_ESTIMATED_SIZE[remote_pkg.name])) + remote_pkg_idx -= 1 + if local_pkg_idx >= 0: + for pkg in local_packages[local_pkg_idx::-1]: + component_dict[pkg.name].append( + ComponentInfo(version=pkg.version, md5=pkg.md5, release=pkg.release, arch=pkg.arch, type=MirrorRepositoryType.LOCAL.value, estimated_size=const.PKG_ESTIMATED_SIZE[pkg.name])) + if remote_pkg_idx >= 0: + for pkg in remote_packages[remote_pkg_idx::-1]: + component_dict[pkg.name].append( + ComponentInfo(version=pkg.version, md5=pkg.md5, release=pkg.release, arch=pkg.arch, type=MirrorRepositoryType.REMOTE.value, estimated_size=const.PKG_ESTIMATED_SIZE[pkg.name])) + for component, version in component_filter.items(): + if component in component_dict.keys(): + log.get_logger().info("filter component: {0} above version: {1}".format(component, version)) + log.get_logger().info("original components: {0}".format(component_dict[component])) + component_dict[component] = list(filter(lambda c: Version(c.version) >= Version(version), component_dict[component])) + log.get_logger().info("filtered components: {0}".format(component_dict[component])) + return component_dict + + def list_components(self): + if self.context['mirror']['remote_mirror_info_status'] != const.FINISHED: + raise Exception("startup event mirror update still not finished") + component_list = list() + component_dict = self.__get_all_components() + for componentInfo in component_dict[const.OCEANBASE_CE]: + componentInfo.version_type = const.CE + for componentInfo in component_dict[const.OCEANBASE]: + componentInfo.version_type = const.BUSINESS + for componentInfo in component_dict[const.OBPROXY_CE]: + componentInfo.version_type = const.CE + for componentInfo in component_dict[const.OBPROXY]: + componentInfo.version_type = const.BUSINESS + + if const.OCEANBASE in component_dict.keys() and const.OCEANBASE_CE in component_dict.keys(): + component_dict[const.OCEANBASE].extend(component_dict[const.OCEANBASE_CE]) + component_dict.pop(const.OCEANBASE_CE) + component_dict[const.OCEANBASE].sort(key=lambda x: x.version, reverse=True) + elif const.OCEANBASE_CE in component_dict.keys(): + component_dict[const.OCEANBASE] = component_dict[const.OCEANBASE_CE] + component_dict.pop(const.OCEANBASE_CE) + if const.OBPROXY in component_dict.keys() and const.OBPROXY_CE in component_dict.keys(): + component_dict[const.OBPROXY].extend(component_dict[const.OBPROXY_CE]) + component_dict.pop(const.OBPROXY_CE) + component_dict[const.OBPROXY].sort(key=lambda x: x.version, reverse=True) + elif const.OBPROXY_CE in component_dict.keys(): + component_dict[const.OBPROXY] = component_dict[const.OBPROXY_CE] + component_dict.pop(const.OBPROXY_CE) + for name, info in component_dict.items(): + component_list.append(Component(name=name, info=info)) + return component_list + + def get_component(self, component_name): + if self.context['mirror']['remote_mirror_info_status'] != const.FINISHED: + raise Exception("startup event mirror update still not finished") + component = None + component_dict = self.__get_all_components() + if component_name in component_dict.keys(): + component = Component(name=component_name, info=component_dict[component_name]) + return component + + + def list_component_parameters(self, parameter_request): + parameter_metas = list() + for parameter_filter in parameter_request.filters: + name=uuid.uuid4().hex + # generate minimal deploy + config_path = '' + log.get_logger().info('dump config') + with tempfile.NamedTemporaryFile(prefix="obd", suffix="yaml", mode="w", encoding="utf-8") as f: + f.write(const.MINIMAL_CONFIG.format(parameter_filter.component)) + f.flush() + config_path = f.name + deploy = self.obd.deploy_manager.create_deploy_config(name, config_path) + if deploy is None: + raise Exception("create temp deployment failed") + self.obd.set_deploy(deploy) + + spacename = "{0}_parameter".format(parameter_filter.component) + gen_config_plugin = self.obd.plugin_manager.get_best_py_script_plugin("generate_config", parameter_filter.component, parameter_filter.version) + repository = Repository(parameter_filter.component, "") + self.obd.set_repositories([repository]) + ret = self.obd.call_plugin(gen_config_plugin, repository, return_generate_keys=True, generate_consistent_config=True, spacename=spacename, clients={}) + del(self.obd.namespaces[spacename]) + if not ret: + self.obd.deploy_manager.remove_deploy_config(name) + raise Exception("genconfig failed for compoennt: {0}".format(parameter_filter.component)) + else: + auto_keys = ret.get_return("generate_keys") + log.get_logger().info("auto keys for comopnent %s are %s", parameter_filter.component, auto_keys) + + parameter_plugin = self.obd.plugin_manager.get_best_plugin(PluginType.PARAM, parameter_filter.component, parameter_filter.version) + ## use plugin.params to generate parameter meta + config_parameters = list() + for param in parameter_plugin.params.values(): + config_parameter = map_to_config_parameter(param) + if config_parameter.name in auto_keys: + config_parameter.auto = True + if config_parameter.is_essential or not parameter_filter.is_essential_only: + config_parameters.append(config_parameter) + parameter_metas.append(ParameterMeta(component=parameter_filter.component, version=parameter_filter.version, config_parameters=config_parameters)) + self.obd.deploy_manager.remove_deploy_config(name) + return parameter_metas diff --git a/service/handler/deployment_handler.py b/service/handler/deployment_handler.py new file mode 100644 index 0000000..93e6e6a --- /dev/null +++ b/service/handler/deployment_handler.py @@ -0,0 +1,705 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import json +import tempfile +from collections import defaultdict + +from optparse import Values +from singleton_decorator import singleton +import yaml +from _deploy import DeployStatus, DeployConfigStatus +from _errno import CheckStatus, FixEval +from service.api.v1.deployments import DeploymentInfo +from service.handler.base_handler import BaseHandler +from service.model.deployments import DeploymentConfig, PreCheckResult, RecoverChangeParameter, TaskInfo, \ + ComponentInfo, PrecheckTaskResult, \ + DeployMode, ConnectionInfo, PreCheckInfo, RecoverAdvisement, DeploymentReport, Deployment, Auth, DeployConfig, \ + DeploymentStatus, Parameter + +from service.common import log, task, util, const +from service.common.task import TaskStatus, TaskResult +from service.common.task import Serial as serial +from service.common.task import AutoRegister as auto_register + + +@singleton +class DeploymentHandler(BaseHandler): + def get_deployment_by_name(self, name): + deployment = self.obd.deploy_manager.get_deploy_config(name) + if deployment is None: + return None + deployment_info = DeploymentInfo() + deployment_info.name = deployment.name + deployment_info.config_path = deployment.config_dir + deployment_info.status = deployment.deploy_info.status.value.upper() + deployment_info.config = self.context['deployment'][deployment.name] if self.context[ + 'deployment'] is not None else None + return deployment_info + + def generate_deployment_config(self, name: str, config: DeploymentConfig): + log.get_logger().debug('generate cluster config') + cluster_config = {} + if config.auth is not None: + self.generate_auth_config(cluster_config, config.auth) + if config.components.oceanbase is not None: + self.generate_oceanbase_config(cluster_config, config, name, config.components.oceanbase) + if config.components.obproxy is not None: + cluster_config[config.components.obproxy.component] = self.generate_component_config(config, const.OBPROXY, ['cluster_name', 'prometheus_listen_port', 'listen_port']) + if config.components.obagent is not None: + cluster_config[config.components.obagent.component] = self.generate_component_config(config, const.OBAGENT, ['monagent_http_port', 'mgragent_http_port']) + if config.components.ocpexpress is not None: + cluster_config[config.components.ocpexpress.component] = self.generate_component_config(config, const.OCP_EXPRESS, ['port']) + cluster_config_yaml_path = '' + log.get_logger().info('dump config from path: %s' % cluster_config_yaml_path) + with tempfile.NamedTemporaryFile(delete=False, prefix="obd", suffix="yaml", mode="w", encoding="utf-8") as f: + f.write(yaml.dump(cluster_config, sort_keys=False)) + cluster_config_yaml_path = f.name + self.context['deployment'][name] = config + return cluster_config_yaml_path + + def generate_component_config(self, config, component_name, ext_keys=[]): + comp_config = dict() + input_comp_config = getattr(config.components, component_name) + config_dict = input_comp_config.dict() + for key in config_dict: + if config_dict[key] and key in {'servers', 'version', 'package_hash', 'release'}: + comp_config[key] = config_dict[key] + + if 'global' not in comp_config.keys(): + comp_config['global'] = dict() + + ext_keys.insert(0, 'home_path') + for key in ext_keys: + if config_dict[key]: + comp_config['global'][key] = config_dict[key] + + if input_comp_config.home_path == '': + comp_config['global']['home_path'] = config.home_path + '/' + component_name + + for parameter in input_comp_config.parameters: + if not parameter.adaptive: + comp_config['global'][parameter.key] = parameter.value + return comp_config + + def generate_oceanbase_config(self, cluster_config, config, name, oceanbase): + oceanbase_config = dict() + config_dict = oceanbase.dict() + for key in config_dict: + if config_dict[key] and key in {'version', 'release', 'package_hash'}: + oceanbase_config[key] = config_dict[key] + servers = [] + if oceanbase.topology: + for zone in oceanbase.topology: + root_service = zone.rootservice + servers.append(root_service) + for zone in oceanbase.topology: + root_service = zone.rootservice + if root_service not in oceanbase_config.keys(): + oceanbase_config[root_service] = {} + oceanbase_config[root_service]['zone'] = zone.name + for server in zone.servers: + ip = server.ip + if ip not in oceanbase_config.keys(): + oceanbase_config[ip] = {} + if ip != root_service: + servers.append(server.ip) + oceanbase_config[ip]['zone'] = zone.name + if server.parameters: + for parameter in server.parameters: + for key, value in parameter: + oceanbase_config[ip][key] = value + oceanbase_config['servers'] = servers + if 'global' not in oceanbase_config.keys(): + oceanbase_config['global'] = {} + + for key in config_dict: + if config_dict[key] and key in {'mysql_port', 'rpc_port', 'home_path', 'data_dir', 'redo_dir', 'appname', + 'root_password'}: + oceanbase_config['global'][key] = config_dict[key] + + if oceanbase.home_path == '': + oceanbase_config['global']['home_path'] = config.home_path + '/oceanbase' + + if oceanbase.parameters: + for parameter in oceanbase.parameters: + if not parameter.adaptive: + oceanbase_config['global'][parameter.key] = parameter.value + if oceanbase.component == const.OCEANBASE_CE: + cluster_config[const.OCEANBASE_CE] = oceanbase_config + elif oceanbase.component == const.OCEANBASE: + cluster_config[const.OCEANBASE] = oceanbase_config + else: + log.get_logger().error('oceanbase component : %s not exist' % oceanbase.component) + raise Exception('oceanbase component : %s not exist' % oceanbase.component) + + def generate_auth_config(self, cluster_config, auth): + if 'user' not in cluster_config.keys(): + cluster_config['user'] = {} + cluster_config['user']['username'] = auth.user + if auth.password: + cluster_config['user']['password'] = auth.password + cluster_config['user']['port'] = auth.port + + def create_deployment(self, name: str, config_path: str): + log.get_logger().debug('deploy cluster') + deploy = self.obd.deploy_manager.get_deploy_config(name) + if deploy: + deploy_info = deploy.deploy_info + if deploy_info.status not in [DeployStatus.STATUS_CONFIGURED, DeployStatus.STATUS_DESTROYED]: + log.get_logger().error('Deploy "%s" is %s. You could not deploy an %s cluster.' % ( + name, deploy_info.status.value, deploy_info.status.value)) + raise Exception('Deploy "%s" is %s. You could not deploy an %s cluster.' % ( + name, deploy_info.status.value, deploy_info.status.value)) + if deploy_info.config_status != DeployConfigStatus.UNCHNAGE: + log.get_logger().debug('Apply temp deploy configuration') + if not deploy.apply_temp_deploy_config(): + log.get_logger().error('Failed to apply new deploy configuration') + raise Exception('Failed to apply new deploy configuration') + + deploy = self.obd.deploy_manager.create_deploy_config(name, config_path) + if not deploy: + log.get_logger().error('Failed to create deploy: %s. please check you configuration file' % name) + raise Exception('Failed to create deploy: %s. please check you configuration file' % name) + self.obd.set_deploy(deploy) + log.get_logger().info('cluster config path: %s ' % config_path) + return config_path + + def get_precheck_result(self, name): + precheck_result = PreCheckResult() + deploy = self.obd.deploy + if not deploy: + deploy = self.obd.deploy_manager.get_deploy_config(name) + self.obd.set_deploy(deploy) + components = deploy.deploy_config.components + info = [] + total = 0 + finished = 0 + all_passed = True + param_check_status = None + connect_check_status = None + if 'deployment' in self.context.keys(): + param_check_status = self.context['deployment']['param_check_status'] + connect_check_status = self.context['deployment']['connect_check_status'] + connect_check_status_flag = True + for component in components: + namespace_union = {} + namespace = self.obd.get_namespace(component) + if namespace: + variables = namespace.variables + if 'start_check_status' in variables.keys(): + namespace_union = util.recursive_update_dict(namespace_union, variables.get('start_check_status')) + if param_check_status is not None: + namespace_union = util.recursive_update_dict(namespace_union, param_check_status[component]) + if connect_check_status is not None and connect_check_status_flag and 'ssh' in connect_check_status.keys(): + namespace_union = util.recursive_update_dict(namespace_union, connect_check_status['ssh']) + connect_check_status_flag = False + + if namespace_union: + for server, result in namespace_union.items(): + if result is None: + log.get_logger().warn("precheck for server: {} is None".format(server.ip)) + continue + all_passed, finished, total = self.parse_precheck_result(all_passed, component, finished, info, server, total, result) + info.sort(key=lambda p: p.status) + + task_info = task.get_task_manager().get_task_info(name, task_type="precheck") + if task_info is not None: + if task_info.status == TaskStatus.FINISHED: + precheck_result.status = task_info.result + if task_info.result == TaskResult.FAILED: + precheck_result.message = '{}'.format(task_info.exception) + else: + precheck_result.status = TaskResult.RUNNING + precheck_result.info = info + precheck_result.total = total + if total == 0: + all_passed = False + precheck_result.all_passed = all_passed + precheck_result.finished = total if precheck_result.status == TaskResult.SUCCESSFUL else finished + return precheck_result + + def parse_precheck_result(self, all_passed, component, finished, info, server, total, result): + for k, v in result.items(): + total += 1 + check_info = PreCheckInfo(name='{}:{}'.format(component, k), server=server.ip) + if v.status == v.PASS: + check_info.result = PrecheckTaskResult.PASSED + check_info.status = TaskStatus.FINISHED + finished += 1 + elif v.status == v.FAIL: + check_info.result = PrecheckTaskResult.FAILED + check_info.status = TaskStatus.FINISHED + all_passed = False + + check_info.code = v.error.code + check_info.description = v.error.msg + check_info.recoverable = len(v.suggests) > 0 and v.suggests[0].auto_fix + msg = v.suggests[0].msg if len(v.suggests) > 0 and v.suggests[0].msg is not None else '' + advisement = RecoverAdvisement(description=msg) + check_info.advisement = advisement + + finished += 1 + elif v.status == v.WAIT: + check_info.status = TaskStatus.PENDING + all_passed = False + info.append(check_info) + return all_passed, finished, total + + @serial("install") + def install(self, name, background_tasks): + task_manager = task.get_task_manager() + task_info = task_manager.get_task_info(name, task_type="install") + if task_info is not None and task_info.status != TaskStatus.FINISHED: + raise Exception("task {0} exists and not finished".format(name)) + task_manager.del_task_info(name, task_type="install") + background_tasks.add_task(self._do_install, name) + + @auto_register("install") + def _do_install(self, name): + log.get_logger().info("clean io buffer before start install") + self.buffer.clear() + log.get_logger().info("clean namespace for init") + for c in self.obd.deploy.deploy_config.components: + for plugin in const.INIT_PLUGINS: + self.obd.namespaces[c].set_return(plugin, None) + log.get_logger().info("clean namespace for start") + for component in self.obd.deploy.deploy_config.components: + for plugin in const.START_PLUGINS: + self.obd.namespaces[component].set_return(plugin, None) + + log.get_logger().info("start do deploy %s", name) + self.obd.set_options(Values()) + deploy_success = self.obd.deploy_cluster(name) + if not deploy_success: + log.get_logger().warn("deploy %s failed", name) + log.get_logger().info("finish do deploy %s", name) + log.get_logger().info("start do start %s", name) + + repositories = self.obd.load_local_repositories(self.obd.deploy.deploy_info, False) + repositories = self.obd.sort_repository_by_depend(repositories, self.obd.deploy.deploy_config) + start_success = True + connection_info_list = list() + for repository in repositories: + opt = Values() + setattr(opt, "components", repository.name) + self.obd.set_options(opt) + ret = self.obd._start_cluster(self.obd.deploy, repositories) + if not ret: + log.get_logger().warn("failed to start component: %s", repository.name) + start_success = False + else: + display_ret = self.obd.namespaces[repository.name].get_return("display") + connection_info = self.__build_connection_info(repository.name, display_ret.get_return("info")) + if connection_info is not None: + connection_info_list.append(connection_info) + self.obd.set_options(Values) + if not deploy_success: + raise Exception("task {0} deploy failed".format(name)) + if not start_success: + raise Exception("task {0} start failed".format(name)) + self.obd.deploy.update_deploy_status(DeployStatus.STATUS_RUNNING) + log.get_logger().info("finish do start %s", name) + self.context["connection_info"][name] = connection_info_list + deployment_report = self.get_deployment_report(name) + self.context["deployment_report"][name] = deployment_report + + def get_install_task_info(self, name): + task_info = task.get_task_manager().get_task_info(name, task_type="install") + if task_info is None: + raise Exception("task {0} not found".format(name)) + components = self.obd.deploy.deploy_config.components + total_count = (len(const.START_PLUGINS) + len(const.INIT_PLUGINS)) * len(components) + finished_count = 0 + current = "" + task_result = TaskResult.RUNNING + info_dict = dict() + + for component in self.obd.deploy.deploy_config.components: + info_dict[component] = ComponentInfo(component=component, status=TaskStatus.PENDING, + result=TaskResult.RUNNING) + if component in self.obd.namespaces: + for plugin in const.INIT_PLUGINS: + if self.obd.namespaces[component].get_return(plugin) is not None: + info_dict[component].status = TaskStatus.RUNNING + finished_count += 1 + current = "{0}: {1} finished".format(component, plugin) + if not self.obd.namespaces[component].get_return(plugin): + info_dict[component].result = TaskResult.FAILED + + for component in self.obd.deploy.deploy_config.components: + for plugin in const.START_PLUGINS: + if component not in self.obd.namespaces: + break + if self.obd.namespaces[component].get_return(plugin) is not None: + info_dict[component].status = TaskStatus.RUNNING + finished_count += 1 + current = "{0}: {1} finished".format(component, plugin) + if not self.obd.namespaces[component].get_return(plugin): + info_dict[component].result = TaskResult.FAILED + else: + if plugin == const.START_PLUGINS[-1]: + info_dict[component].result = TaskResult.SUCCESSFUL + + if task_info.status == TaskStatus.FINISHED: + task_result = task_info.result + for v in info_dict.values(): + v.status = TaskStatus.FINISHED + if v.result != TaskResult.SUCCESSFUL: + v.result = TaskResult.FAILED + info_list = list() + for info in info_dict.values(): + info_list.append(info) + msg = "" if task_info.result == TaskResult.SUCCESSFUL else '{0}'.format(task_info.exception) + return TaskInfo(total=total_count, finished=finished_count if task_result != TaskResult.SUCCESSFUL else total_count, current=current, status=task_result, info=info_list, + msg=msg) + + def __build_connection_info(self, component, info): + if info is None: + log.get_logger().warn("component {0} info from display is None".format(component)) + return None + return ConnectionInfo(component=component, + access_url="{0}:{1}".format(info['ip'], info['port']), + user=info['user'], password=info['password'], + connect_url=info['cmd'] if info['type'] == 'db' else info['url']) + + def list_connection_info(self, name): + if self.context["connection_info"][name] is not None: + log.get_logger().info("get deployment {0} connection info from context".format(name)) + return self.context["connection_info"][name] + if name != self.obd.deploy.name: + raise Exception("deployment name not match, current: {0}, from param: {1}".format(self.obd.deploy.name, name)) + deploy = self.obd.deploy_manager.get_deploy_config(name) + connection_info_list = list() + task_info = self.get_install_task_info(name) + component_info = task_info.info + for component, config in deploy.deploy_config.components.items(): + connection_info = None + start_ok = False + for c in component_info: + if c.component == component and c.status == TaskStatus.FINISHED and c.result == TaskResult.SUCCESSFUL: + start_ok = True + if not start_ok: + log.get_logger().warn("component %s start failed", component) + continue + display_ret = self.obd.namespaces[component].get_return("display") + connection_info = self.__build_connection_info(component, display_ret.get_return("info")) + if connection_info is not None: + connection_info_list.append(connection_info) + else: + log.get_logger().warn("can not get connection info for component: {0}".format(component)) + return connection_info_list + + @serial("precheck") + def precheck(self, name, background_tasks): + task_manager = task.get_task_manager() + task_info = task_manager.get_task_info(name, task_type="precheck") + if task_info is not None and task_info.status != TaskStatus.FINISHED: + raise Exception("task {0} exists and not finished".format(name)) + deploy = self.obd.deploy + if not deploy: + raise Exception("no such deploy {0}".format(name)) + deploy_config = deploy.deploy_config + # Get the repository + pkgs, repositories, errors = self.obd.search_components_from_mirrors(deploy_config, only_info=True) + if errors: + raise Exception("{}".format('\n'.join(errors))) + repositories.extend(pkgs) + repositories = self.obd.sort_repository_by_depend(repositories, deploy_config) + for repository in repositories: + real_servers = set() + cluster_config = deploy_config.components[repository.name] + for server in cluster_config.servers: + if server.ip in real_servers: + raise Exception( + "Deploying multiple {} instances on the same server is not supported.'".format( + repository.name)) + return False + real_servers.add(server.ip) + self.obd.search_param_plugin_and_apply(repositories, deploy_config) + self.obd.set_repositories(repositories) + + if 'deployment' in self.context.keys() and self.context['deployment'][name] is not None and self.context['deployment'][name].components.oceanbase is not None and self.context['deployment'][name].components.oceanbase.mode == DeployMode.DEMO.value: + for repository in repositories: + self.obd.get_namespace(repository.name).set_variable('generate_config_mini', True) + + start_check_plugins = self.obd.search_py_script_plugin(repositories, 'start_check', no_found_act='warn') + + self._precheck(name, repositories, start_check_plugins, init_check_status=True) + info = task_manager.get_task_info(name, task_type="precheck") + if info is not None and info.exception is not None: + raise info.exception + task_manager.del_task_info(name, task_type="precheck") + background_tasks.add_task(self._precheck, name, repositories, start_check_plugins, init_check_status=False) + + def _init_check_status(self, check_key, servers, check_result={}): + check_status = defaultdict(lambda: defaultdict(lambda: None)) + for server in servers: + if server in check_result: + status = check_result[server] + else: + status = CheckStatus() + check_status[server] = {check_key: status} + return check_status + + @auto_register('precheck') + def _precheck(self, name, repositories, start_check_plugins, init_check_status=False): + if init_check_status: + self._init_precheck(repositories, start_check_plugins) + else: + self._do_precheck(repositories, start_check_plugins) + + def _init_precheck(self, repositories, start_check_plugins): + param_check_status = {} + servers_set = set() + for repository in repositories: + if repository not in start_check_plugins: + continue + repository_status = {} + res = self.obd.call_plugin(start_check_plugins[repository], repository, + init_check_status=True, work_dir_check=True, clients={}) + if not res and res.get_return("exception"): + raise res.get_return("exception") + servers = self.obd.deploy.deploy_config.components.get(repository.name).servers + for server in servers: + repository_status[server] = {'param': CheckStatus()} + servers_set.add(server) + param_check_status[repository.name] = repository_status + + self.context['deployment']['param_check_status'] = param_check_status + server_connect_status = {} + for server in servers_set: + server_connect_status[server] = {'ssh': CheckStatus()} + self.context['deployment']['connect_check_status'] = {'ssh': server_connect_status} + self.context['deployment']['servers_set'] = servers_set + + def _do_precheck(self, repositories, start_check_plugins): + ssh_clients, connect_status = self.obd.get_clients_with_connect_status(self.obd.deploy.deploy_config, + repositories, fail_exit=False) + check_status = self._init_check_status('ssh', self.context['deployment']['servers_set'], connect_status) + self.context['deployment']['connect_check_status'] = {'ssh': check_status} + for k, v in connect_status.items(): + if v.status == v.FAIL: + return + gen_config_plugins = self.obd.search_py_script_plugin(repositories, 'generate_config') + if len(repositories) != len(gen_config_plugins): + raise Exception("param_check: config error, check stop!") + + param_check_status, check_pass = self.obd.deploy_param_check_return_check_status(repositories, self.obd.deploy.deploy_config, gen_config_plugins=gen_config_plugins) + param_check_status_result = {} + for comp_name in param_check_status: + status_res = param_check_status[comp_name] + param_check_status_result[comp_name] = self._init_check_status('param', status_res.keys(), status_res) + self.context['deployment']['param_check_status'] = param_check_status_result + + if not check_pass: + return + + for repository in repositories: + ret = self.obd.call_plugin(gen_config_plugins[repository], repository, generate_check=False, + generate_consistent_config=True, auto_depend=True) + if ret is None: + raise Exception("generate config error") + elif not ret and ret.get_return("exception"): + raise ret.get_return("exception") + if not self.obd.deploy.deploy_config.dump(): + raise Exception('generate config dump error,place check disk space!') + + for repository in repositories: + res = self.obd.call_plugin(start_check_plugins[repository], repository, init_check_status=False, work_dir_check=True, precheck=True) + if not res and res.get_return("exception"): + raise res.get_return("exception") + + + def get_deployment_report(self, name): + if self.context["deployment_report"][name] is not None: + log.get_logger().info("get deployment {0} report from context".format(name)) + return self.context["deployment_report"][name] + if name != self.obd.deploy.name: + raise Exception("deployment name not match, current: {0}, from param: {1}".format(self.obd.deploy.name, name)) + report_list = list() + for component, config in self.obd.deploy.deploy_config.components.items(): + status = TaskResult.FAILED + if self.obd.namespaces[component].get_return("display"): + status = TaskResult.SUCCESSFUL + report_list.append( + DeploymentReport(name=component, version=config.version, servers=[s.ip for s in config.servers], + status=status)) + return report_list + + def list_deployments_by_status(self, deployment_status): + deployments = self.obd.deploy_manager.get_deploy_configs() + deploys = [] + if deployment_status == DeploymentStatus.INSTALLING: + # query installing task + for deployment in deployments: + task_info = task.get_task_manager().get_task_info(deployment.name, task_type="install") + if task_info is not None and task_info.status == TaskStatus.RUNNING: + deploy = Deployment(name=deployment.name, status=deployment.deploy_info.status.value.upper()) + deploys.append(deploy) + elif deployment_status == DeploymentStatus.DRAFT: + # query draft task + obd_deploy_status = ['configured', 'deployed', 'destroyed'] + for deployment in deployments: + if deployment.deploy_info.status.value in obd_deploy_status: + config = self.context['deployment'][deployment.name] if self.context['deployment'] is not None else None + if config is not None: + deploy = Deployment(name=deployment.name, status=deployment.deploy_info.status.value.upper()) + deploys.append(deploy) + return deploys + + @auto_register("destroy") + def destroy_cluster(self, name): + deploy = self.obd.deploy_manager.get_deploy_config(name) + if not deploy: + raise Exception("no such deploy {0}".format(name)) + self.obd.set_deploy(deploy) + repositories = self.obd.load_local_repositories(deploy.deploy_info) + self.obd.set_repositories(repositories) + self.obd.set_options(Values({'force_kill': True})) + self.obd.search_param_plugin_and_apply(repositories, deploy.deploy_config) + # set namespace return value to none before do destroy + for component in self.obd.deploy.deploy_config.components: + if component in self.obd.namespaces: + self.obd.namespaces[component].set_return(const.DESTROY_PLUGIN, None) + + ret = self.obd._destroy_cluster(deploy, repositories) + if not ret: + raise Exception("destroy cluster {0} failed".format(name)) + deploy.update_deploy_status(DeployStatus.STATUS_CONFIGURED) + self.obd.set_options(Values()) + return ret + + def get_destroy_task_info(self, name): + task_info = task.get_task_manager().get_task_info(name, task_type="destroy") + if task_info is None: + raise Exception("task {0} not found".format(name)) + components = self.obd.deploy.deploy_config.components + total_count = len(components) + finished_count = 0 + current = "" + task_result = TaskResult.RUNNING + info_dict = dict() + for c in self.obd.deploy.deploy_config.components: + info_dict[c] = ComponentInfo(component=c, status=TaskStatus.PENDING, result=TaskResult.RUNNING) + if c in self.obd.namespaces: + if self.obd.namespaces[c].get_return(const.DESTROY_PLUGIN) is not None: + info_dict[c].status = TaskStatus.FINISHED + finished_count += 1 + current = "{0}: {1} finished".format(c, const.DESTROY_PLUGIN) + if not self.obd.namespaces[c].get_return(const.DESTROY_PLUGIN): + info_dict[c].result = TaskResult.FAILED + else: + info_dict[c].result = TaskResult.SUCCESSFUL + if task_info.status == TaskStatus.FINISHED: + task_result = task_info.result + for v in info_dict.values(): + if v.status != TaskStatus.FINISHED: + v.status = TaskStatus.FINISHED + finished_count += 1 + if v.result != TaskResult.SUCCESSFUL: + v.result = TaskResult.FAILED + info_list = list() + for info in info_dict.values(): + info_list.append(info) + msg = "" if task_info.result == TaskResult.SUCCESSFUL else '{0}'.format(task_info.exception) + return TaskInfo(total=total_count, finished=finished_count, current=current, status=task_result, info=info_list, + msg=msg) + + def recover(self, name): + deploy = self.obd.deploy + if not deploy: + deploy = self.obd.deploy_manager.get_deploy_config(name) + self.obd.set_deploy(deploy) + components = deploy.deploy_config.components + param_check_status = None + if 'deployment' in self.context.keys(): + param_check_status = self.context['deployment']['param_check_status'] + recover_change_parameter_list = [] + for component in components: + namespace_union = {} + if component in self.obd.namespaces: + namespace = self.obd.get_namespace(component) + if namespace: + util.recursive_update_dict(namespace_union, namespace.variables.get('start_check_status', {})) + util.recursive_update_dict(namespace_union, param_check_status.get('component', {})) + + for server, precheck_result in namespace_union.items(): + if precheck_result is None: + log.get_logger().warn('component : {},precheck_result is None'.format(component)) + continue + for k, v in precheck_result.items(): + if v.status == v.FAIL and v.suggests is not None and v.suggests[0].auto_fix and v.suggests[0].fix_eval: + for fix_eval in v.suggests[0].fix_eval: + if fix_eval.operation == FixEval.SET: + config_json = None + old_value = None + if fix_eval.is_global: + deploy.deploy_config.update_component_global_conf(name, fix_eval.key, fix_eval.value, save=False) + else: + deploy.deploy_config.update_component_server_conf(name, server, fix_eval.key, fix_eval.value, save=False) + else: + config_json, old_value = self.modify_config(component, name, fix_eval) + + if config_json is None: + log.get_logger().warn('config json is None') + continue + recover_change_parameter = RecoverChangeParameter(name=fix_eval.key, old_value=old_value, new_value=fix_eval.value) + recover_change_parameter_list.append(recover_change_parameter) + self.context['deployment'][name] = DeploymentConfig(**json.loads(json.dumps(config_json))) + deploy.deploy_config.dump() + self.recreate_deployment(name) + + return recover_change_parameter_list + + def recreate_deployment(self, name): + config = self.context['deployment'][name] if self.context['deployment'] is not None else None + if config is not None: + cluster_config_yaml_path = self.generate_deployment_config(name, config) + self.create_deployment(name, cluster_config_yaml_path) + + def modify_config(self, component, name, fix_eval): + if fix_eval.key == "parameters": + raise Exception("try to change parameters") + config = self.context['deployment'][name] if self.context['deployment'] is not None else None + if config is None: + log.get_logger().warn("config is none, no need to modify") + raise Exception('config is none') + config_dict = config.dict() + if config_dict['components'] is None: + log.get_logger().warn("component is none, no need to modify") + raise Exception('component is none') + old_value = None + for value in config_dict['components'].values(): + if value is not None and 'component' in value.keys() and value['component'] == component: + if fix_eval.key in value.keys(): + old_value = value[fix_eval.key] + value[fix_eval.key] = fix_eval.value + elif "parameters" in value.keys() and value["parameters"] is not None: + for parameter_dict in value["parameters"]: + parameter = Parameter(**parameter_dict) + if parameter.key == fix_eval.key: + if fix_eval.operation == FixEval.DEL: + old_value = parameter.value + value["parameters"].remove(parameter_dict) + else: + parameter_dict[fix_eval.key] = fix_eval.value + return config_dict, old_value + return None, None + diff --git a/service/handler/handler_utils.py b/service/handler/handler_utils.py new file mode 100644 index 0000000..b95fa3f --- /dev/null +++ b/service/handler/handler_utils.py @@ -0,0 +1,44 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from service.handler.component_handler import ComponentHandler +from service.handler.deployment_handler import DeploymentHandler +from service.handler.service_info_handler import ServiceInfoHandler +from service.handler.process_handler import ProcessHandler +from service.handler.mirror_handler import MirrorHandler + + +def new_component_handler(): + return ComponentHandler() + + +def new_deployment_handler(): + return DeploymentHandler() + + +def new_process_handler(): + return ProcessHandler() + + +def new_service_info_handler(): + return ServiceInfoHandler() + + +def new_mirror_handler(): + return MirrorHandler() diff --git a/service/handler/mirror_handler.py b/service/handler/mirror_handler.py new file mode 100644 index 0000000..ab8a0ff --- /dev/null +++ b/service/handler/mirror_handler.py @@ -0,0 +1,56 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from _errno import LockError +from service.common import log +from service.common import const +from service.handler.base_handler import BaseHandler +from singleton_decorator import singleton +from service.model.mirror import Mirror + +@singleton +class MirrorHandler(BaseHandler): + + def list_mirrors(self): + if self.context['mirror']['remote_mirror_info_status'] != const.FINISHED: + raise Exception('update mirror not finished') + remote_mirror_info = self.context['mirror']['remote_mirror_info'] + return remote_mirror_info + + async def init_mirrors_info(self): + self.context['mirror']['remote_mirror_info_status'] = const.RUNNING + try: + mirror_list = [] + mirrors = self.obd.mirror_manager.get_remote_mirrors(is_enabled=True) + mirrors_disabled = self.obd.mirror_manager.get_remote_mirrors(is_enabled=False) + mirrors.extend(mirrors_disabled) + for mirror in mirrors: + mirror_list.append( + Mirror(name=mirror.name, mirror_path=mirror.mirror_path, section_name=mirror.section_name, + baseurl=mirror.baseurl, + repomd_age=mirror.repomd_age, priority=mirror.priority, gpgcheck=mirror.gpgcheck, + enabled=mirror.enabled, available=mirror.available, repo_age=mirror.repo_age)) + self.context['mirror']['remote_mirror_info'] = mirror_list + except LockError: + log.get_logger().error('Another app is currently holding the obd lock.') + except Exception as ex: + log.get_logger().exception("got exception {} when init mirror".format(ex)) + finally: + self.context['mirror']['remote_mirror_info_status'] = const.FINISHED + diff --git a/service/handler/process_handler.py b/service/handler/process_handler.py new file mode 100644 index 0000000..784f708 --- /dev/null +++ b/service/handler/process_handler.py @@ -0,0 +1,36 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import os +import time + +from singleton_decorator import singleton + +from service.common import log, const +from service.handler.base_handler import BaseHandler + +@singleton +class ProcessHandler(BaseHandler): + + def suicide(self): + pid = os.getpid() + log.get_logger().info("got suicide requrest, pid is %d", pid) + time.sleep(const.GRACEFUL_TIMEOUT) + log.get_logger().info("suicide") + os.kill(pid, 9) diff --git a/service/handler/service_info_handler.py b/service/handler/service_info_handler.py new file mode 100644 index 0000000..0dc0687 --- /dev/null +++ b/service/handler/service_info_handler.py @@ -0,0 +1,33 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from _deploy import UserConfig +from service.handler.base_handler import BaseHandler +from singleton_decorator import singleton + +from service.model.service_info import ServiceInfo + + +@singleton +class ServiceInfoHandler(BaseHandler): + + def get_service_info(self): + info = ServiceInfo(user=UserConfig.DEFAULT.get('username')) + return info + diff --git a/service/middleware/process_time.py b/service/middleware/process_time.py new file mode 100644 index 0000000..fcd662c --- /dev/null +++ b/service/middleware/process_time.py @@ -0,0 +1,36 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import time +from starlette.middleware.base import BaseHTTPMiddleware + +class ProcessTimeMiddleware(BaseHTTPMiddleware): + def __init__( + self, + app, + ): + super().__init__(app) + + async def dispatch(self, request, call_next): + start_time = time.time() + response = await call_next(request) + end_time = time.time() + process_time_str = "{0}ms".format((end_time - start_time) * 1000) + response.headers["X-Process-Time"] = process_time_str + return response diff --git a/service/middleware/request_response_log.py b/service/middleware/request_response_log.py new file mode 100644 index 0000000..2230ccd --- /dev/null +++ b/service/middleware/request_response_log.py @@ -0,0 +1,44 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +import json +from starlette.middleware.base import BaseHTTPMiddleware +from starlette.concurrency import iterate_in_threadpool + + +async def set_body(request, body): + async def receive(): + return {'type': 'http.request', 'body': body} + request._receive = receive + +class RequestResponseLogMiddleware(BaseHTTPMiddleware): + def __init__( + self, + app, + logger, ): + super().__init__(app) + self.logger = logger + + async def dispatch(self, request, call_next): + request_body = await request.body() + self.logger.info("app receive request, method: %s, url: %s, query_params: %s, body: %s, from: %s:%d", request.method, request.url, request.query_params, request_body.decode(), request.client.host, request.client.port) + await set_body(request, request_body) + response = await call_next(request) + self.logger.info("app send response, code: %d", response.status_code) + return response diff --git a/service/model/__init__.py b/service/model/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/service/model/components.py b/service/model/components.py new file mode 100644 index 0000000..4f5a702 --- /dev/null +++ b/service/model/components.py @@ -0,0 +1,72 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from typing import List + +from fastapi import Body +from pydantic import BaseModel + + +class ComponentInfo(BaseModel): + estimated_size: int = Body(0, description='estimated size after install') + version: str = Body('', description='component version') + type: str = Body('local', description='component type,ex:remote,local') + release: str = Body('', description='component release no') + arch: str = Body('', description='component package arch info') + md5: str = Body('', description='component package md5 info') + version_type: str = Body('', description=' version type,ex:ce,business') + + +class Component(BaseModel): + name: str = Body(..., description='component name') + info: List[ComponentInfo] = Body(None, description='info') + + +class ConfigParameter(BaseModel): + is_essential: bool = Body(False, description='is essential') + name: str = Body("", description='parameter name') + require: bool = Body(False, description='parameter is it required') + auto: bool = Body(False, description='parameter can be calculated automatically') + description: str = Body("", description='parameter description') + type: str = Body("", description='parameter type') + default: str = Body("", description='parameter default value') + min_value: str = Body("", description='parameter min value') + max_value: str = Body("", description='parameter max value') + need_redeploy: bool = Body(False, description='need redeploy') + modify_limit: str = Body("", description='modify limit') + need_reload: bool = Body(False, description='need reload') + need_restart: bool = Body(False, description='need restart') + section: str = Body("", description='section') + + +class ParameterMeta(BaseModel): + component: str = ... + version: str = ... + config_parameters: List[ConfigParameter] = ... + + +class ParameterFilter(BaseModel): + component: str = Body(..., description='component name') + version: str = Body(..., description='version name') + is_essential_only: bool = Body(False, description='essential parameter filter') + + +class ParameterRequest(BaseModel): + filters: List[ParameterFilter] = Body(..., description='parameter filters') + diff --git a/service/model/deployments.py b/service/model/deployments.py new file mode 100644 index 0000000..ee73333 --- /dev/null +++ b/service/model/deployments.py @@ -0,0 +1,235 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from enum import auto +from typing import List, Optional + +from fastapi import Body +from pydantic import BaseModel +from fastapi_utils.enums import StrEnum + +from service.common.task import TaskStatus, TaskResult + + +class Auth(BaseModel): + user: str = Body('', description='ssh user') + password: str = Body(None, description='ssh password') + port: int = Body(22, description='ssh port') + + +class PrecheckTaskResult(StrEnum): + PASSED = auto() + FAILED = auto() + RUNNING = auto() + + +class DeployMode(StrEnum): + DEMO = auto() + PRODUCTION = auto() + + +class DeploymentStatus(StrEnum): + INSTALLING = auto() + DRAFT = auto() + + +class Resource(BaseModel): + cpu: int = Body(None, description='cpu resource') + memory: str = Body(None, description='memory resource') + + +class OceanbaseServers(BaseModel): + ip: str = Body(..., description='server ip') + parameters: dict = None + +class Zone(BaseModel): + name: str = Body(..., description='zone name') + rootservice: str = Body(..., description='root service') + servers: List[OceanbaseServers] + + +class Parameter(BaseModel): + key: str = Body(..., description='parameter key') + value: str = Body(..., description='parameter value') + adaptive: bool = Body(None, description='parameter value is adaptive') + + +class OceanBase(BaseModel): + component: str = Body(..., description='oceanbase component name,ex:oceanbase-ce,oceanbase') + appname: str = Body(..., description='cluster name') + version: str = Body(..., description='version') + release: str = Body(..., description='oceanbase release no') + package_hash: str = Body('', description='oceanbase package md5') + mode: DeployMode = Body(..., description='deploy mode ex:DEMO,PRODUCTION') + root_password: str = Body(..., description='root password') + mysql_port: int = Body(..., description='sql port') + rpc_port: int = Body(..., description='rpc port') + home_path: str = Body('', description='install OceanBase home path') + data_dir: str = Body('', description='OceanBase data path') + redo_dir: str = Body('', description='clog path') + parameters: List[Parameter] = Body(None, description='config parameter') + topology: List[Zone] = Body(..., description='topology') + + +class ObProxy(BaseModel): + component: str = Body(..., description='obproxy component name, ex:obproxy-ce,obproxy') + version: str = Body(..., description='version') + package_hash: str = Body('', description='obproxy package md5') + release: str = Body(..., description='obproxy release no') + cluster_name: str = Body(None, description='obproxy name') + home_path: str = Body('', description='install obproxy home path') + prometheus_listen_port: int = Body(..., description='prometheus port') + listen_port: int = Body(..., description='sql port') + parameters: List[Parameter] = Body(None, description='config parameter') + servers: List[str] = Body(..., description="server ip, ex:[ '1.1.1.1','2.2.2.2']") + + +class OcpExpress(BaseModel): + component: str = Body('ocp-express', description='ocp-express component name') + version: str = Body(..., description='version') + package_hash: str = Body('', description='ocp-express package md5') + release: str = Body(..., description='ocp-express release no') + home_path: str = Body('', description='install ocp-express home path') + port: int = Body(..., description='server port') + parameters: List[Parameter] = Body(None, description='config parameter') + servers: List[str] = Body(..., description="server ip, ex:[ '1.1.1.1','2.2.2.2']") + + +class ObAgent(BaseModel): + component: str = Body('obagent', description='obagent component name,ex:obagent') + version: str = Body(..., description='version') + package_hash: str = Body('', description='obagent package md5') + release: str = Body(..., description='obagent release no') + home_path: str = Body('', description='install obagent home path') + monagent_http_port: int = Body(..., description='server port') + mgragent_http_port: int = Body(..., description='debug port') + parameters: List[Parameter] = Body(None, description='config parameter') + servers: List[str] = Body(..., description="server ip, ex:[ '1.1.1.1','2.2.2.2']") + + +class ObClient(BaseModel): + component: str = Body('obclient', description='obclient component name,ex:obclient') + version: str = Body(..., description='version') + release: str = Body(..., description='obclient release no') + parameters: List[Parameter] = Body(None, description='config parameter') + home_path: str = Body('', description='install obclient home path') + servers: List[str] = Body(..., description="server ip, ex:[ '1.1.1.1','2.2.2.2']") + + +class ComponentConfig(BaseModel): + oceanbase: OceanBase + obproxy: Optional[ObProxy] + ocpexpress: Optional[OcpExpress] + obagent: Optional[ObAgent] + obclient: Optional[ObClient] + + +class DeploymentConfig(BaseModel): + auth: Auth + components: ComponentConfig + home_path: str = Body('', description='global home path') + + +class DeploymentInfo(BaseModel): + name: str = Body('', description='deployment name') + config_path: str = Body('', description='config path') + status: str = Body('', + description='ex:CONFIGURING,CONFIGURED,DEPLOYING,DEPLOYED,RUNNING,STOPPING,STOPPED,DESTROYING,DESTROYED,UPGRADING') + config: DeploymentConfig = None + + +class RecoverAdvisement(BaseModel): + description: str = Body('', description='advisement description') + + +class PreCheckInfo(BaseModel): + name: str = Body(..., description='pre check item') + server: str = Body(..., description='server node') + status: TaskStatus = Body(TaskStatus.PENDING, description='status, ex:FINISHED, RUNNING, PENDING') + result: PrecheckTaskResult = Body(PrecheckTaskResult.RUNNING, description='result, ex:PASSED, FAILED') + recoverable: bool = Body(True, description='can be automatically repaired') + code: str = Body('', description='error code') + description: str = Body('', description='error description') + advisement: RecoverAdvisement = Body(None, description='repaired suggestion') + + +class PreCheckResult(BaseModel): + total: int = Body(0, description='total item for pre check') + finished: int = Body(0, description='finished item for pre check') + all_passed: bool = Body(False, description='is all passed') + status: TaskResult = Body(TaskResult.RUNNING, description='pre check task status,ex:RUNNING,SUCCESSFUL,FAILED') + message: str = Body('', description='pre check task message') + info: List[PreCheckInfo] = Body(None, description='pre check item info') + + +class RecoverChangeParameter(BaseModel): + name: str = Body(..., description='repaired item') + old_value: object = Body('', description='old value item') + new_value: object = Body('', description='new value item') + + +class ComponentInfo(BaseModel): + component: str = Body(..., description='install component name') + status: TaskStatus = Body(..., description='status, ex:FINISHED, RUNNING, PENDING') + result: TaskResult = Body(..., description='result, ex:SUCCESSFUL, FAILED') + + +class TaskInfo(BaseModel): + total: int = Body(0, description='total item for install') + finished: int = Body(0, description='finished item for install') + current: str = Body('', description='current item for install') + status: TaskResult = Body(..., description='status,ex:RUNNING,SUCCESSFUL,FAILED') + msg: str = Body('', description='task message') + info: List[ComponentInfo] = Body(None, description='install item info') + + +class ConnectionInfo(BaseModel): + component: str = Body(..., description='component name') + access_url: str = Body(..., description='access url') + user: str = Body(..., description='user') + password: str = Body(..., description='password') + connect_url: str = Body(..., description='connect url') + + +class InstallLog(BaseModel): + log: str = Body('', description='install log') + offset: int = Body(0, description='log offset') + + +class Deployment(BaseModel): + name: str = Body(..., description='deployment name') + status: str = Body(..., description='status, ex:CONFIGURED,DEPLOYED,STARTING,RUNNING,DESTROYED,UPGRADING') + + +class DeploymentReport(BaseModel): + name: str = Body(..., description='component name') + version: str = Body(..., description='component version') + servers: List[str] = Body(..., description='server ip') + status: TaskResult = Body(..., description='status, ex: RUNNING, SUCCESSFUL, FAILED') + + +class DeployConfig(BaseModel): + name: str + config: str + + class Config: + orm_mode = True + + + diff --git a/service/model/mirror.py b/service/model/mirror.py new file mode 100644 index 0000000..c9b711b --- /dev/null +++ b/service/model/mirror.py @@ -0,0 +1,35 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from fastapi import Body +from pydantic import BaseModel + + +class Mirror(BaseModel): + mirror_path: str = Body('', description='mirror path') + name: str = Body(..., description='mirror name') + section_name: str = Body('', description='section name') + baseurl: str = Body('', description='baseurl') + repomd_age: int = Body(None, description='repomd age') + repo_age: int = Body(None, description='repo age') + priority: int = Body(None, description='priority') + gpgcheck: str = Body('', description='gpgcheck') + enabled: bool = Body('', description='remote mirror is enabled') + available: bool = Body('', description='remote mirror is enabled') + diff --git a/service/model/service_info.py b/service/model/service_info.py new file mode 100644 index 0000000..327d98f --- /dev/null +++ b/service/model/service_info.py @@ -0,0 +1,25 @@ +# coding: utf-8 +# OceanBase Deploy. +# Copyright (C) 2021 OceanBase +# +# This file is part of OceanBase Deploy. +# +# OceanBase Deploy is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# OceanBase Deploy is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OceanBase Deploy. If not, see . + +from fastapi import Body +from pydantic import BaseModel + + +class ServiceInfo(BaseModel): + user: str = Body(..., description='user name') diff --git a/service/service-requirements.txt b/service/service-requirements.txt new file mode 100644 index 0000000..13b97c5 --- /dev/null +++ b/service/service-requirements.txt @@ -0,0 +1,9 @@ +argparse==1.4.0 +pyyaml==6.0 +pymongo==4.2.0 +fastapi==0.87.0 +uvicorn==0.20.0 +asgi-correlation-id==3.2.1 +starlette_prometheus==0.9.0 +singleton-decorator==1.0.0 +fastapi_utils==0.2.1 \ No newline at end of file diff --git a/service/tests/__init__.py b/service/tests/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/ssh.py b/ssh.py index a076b5b..5eacbce 100644 --- a/ssh.py +++ b/ssh.py @@ -42,6 +42,7 @@ from tool import COMMAND_ENV, DirectoryUtil, FileUtil from _stdio import SafeStdio +from _errno import EC_SSH_CONNECT from _environ import ENV_DISABLE_RSYNC @@ -53,10 +54,10 @@ class SshConfig(object): def __init__(self, host, username='root', password=None, key_filename=None, port=22, timeout=30): self.host = host self.username = username - self.password = password + self.password = password if password is None else str(password) self.key_filename = key_filename - self.port = port - self.timeout = timeout + self.port = int(port) + self.timeout = int(timeout) def __str__(self): return '%s@%s' % (self.username ,self.host) @@ -262,6 +263,7 @@ def is_localhost(self, stdio=None): def _login(self, stdio=None): if self.is_connected: return True + err = None try: self.ssh_client.set_missing_host_key_policy(AutoAddPolicy()) self.ssh_client.connect( @@ -275,13 +277,16 @@ def _login(self, stdio=None): self.is_connected = True except AuthenticationException: stdio.exception('') - stdio.critical('%s@%s username or password error' % (self.config.username, self.config.host)) + err = EC_SSH_CONNECT.format(user=self.config.username, ip=self.config.host, message='username or password error') except NoValidConnectionsError: stdio.exception('') - stdio.critical('%s@%s connect failed: time out' % (self.config.username, self.config.host)) - except Exception as e: + err = EC_SSH_CONNECT.format(user=self.config.username, ip=self.config.host, message='time out') + except BaseException as e: stdio.exception('') - stdio.critical('%s@%s connect failed: %s' % (self.config.username, self.config.host, e)) + err = EC_SSH_CONNECT.format(user=self.config.username, ip=self.config.host, message=e) + if err: + stdio.critical(err) + return err return self.is_connected def _open_sftp(self, stdio=None): @@ -330,7 +335,6 @@ def _execute_command(self, command, timeout=None, retry=3, stdio=None): if code: verbose_msg = 'exited code %s, error output:\n%s' % (code, error) stdio.verbose(verbose_msg) - return SshReturn(code, stdout, error) except SSHException as e: if retry: self.close() @@ -341,8 +345,10 @@ def _execute_command(self, command, timeout=None, retry=3, stdio=None): raise e except Exception as e: stdio.exception('') - stdio.critical('%s@%s connect failed: %s' % (self.config.username, self.config.host, e)) - raise e + code = 255 + stdout = '' + error = str(e) + return SshReturn(code, stdout, error) def execute_command(self, command, timeout=None, stdio=None): if timeout is None: diff --git a/tool.py b/tool.py index 15f9941..a09fa59 100644 --- a/tool.py +++ b/tool.py @@ -21,7 +21,7 @@ from __future__ import absolute_import, division, print_function -import os +import os import bz2 import sys import stat @@ -32,6 +32,7 @@ import re import json import hashlib +import socket from io import BytesIO from ruamel.yaml import YAML, YAMLContextManager, representer @@ -132,7 +133,7 @@ def __init__(self, module): def add_lib_path(lib): if lib not in DynamicLoading.LIBS_PATH: DynamicLoading.LIBS_PATH[lib] = 0 - if DynamicLoading.LIBS_PATH[lib] == 0: + if DynamicLoading.LIBS_PATH[lib] == 0: sys.path.insert(0, lib) DynamicLoading.LIBS_PATH[lib] += 1 @@ -140,7 +141,7 @@ def add_lib_path(lib): def add_libs_path(libs): for lib in libs: DynamicLoading.add_lib_path(lib) - + @staticmethod def remove_lib_path(lib): if lib not in DynamicLoading.LIBS_PATH: @@ -217,6 +218,10 @@ def get_list_from_dict(conf, key, transform_func=None): class DirectoryUtil(object): + @staticmethod + def get_owner(path): + return os.stat(path)[stat.ST_UID] + @staticmethod def list_dir(path, stdio=None): files = [] @@ -352,7 +357,7 @@ def copy(src, dst, stdio=None): return False else: raise IOError(info) - + try: if os.path.islink(src): FileUtil.symlink(os.readlink(src), dst) @@ -438,7 +443,7 @@ def rm(path, stdio=None): except: stdio and getattr(stdio, 'exception', print)('failed to remove %s' % path) return False - + @staticmethod def move(src, dst, stdio=None): return shutil.move(src, dst) @@ -477,7 +482,7 @@ def __init__(self, stdio=None, typ=None, pure=False, output=None, plug_ins=None) self.stdio = stdio if not self.Representer.yaml_multi_representers and self.Representer.yaml_representers: self.Representer.yaml_multi_representers = self.Representer.yaml_representers - + def load(self, stream): try: return super(YamlLoader, self).load(stream) @@ -631,4 +636,11 @@ def show_env(self): return self._cmd_env -COMMAND_ENV = CommandEnv() +class NetUtil(object): + @staticmethod + def get_host_ip(): + hostname = socket.gethostname() + ip = socket.gethostbyname(hostname) + return ip + +COMMAND_ENV=CommandEnv() diff --git a/web/.editorconfig b/web/.editorconfig new file mode 100755 index 0000000..7e3649a --- /dev/null +++ b/web/.editorconfig @@ -0,0 +1,16 @@ +# http://editorconfig.org +root = true + +[*] +indent_style = space +indent_size = 2 +end_of_line = lf +charset = utf-8 +trim_trailing_whitespace = true +insert_final_newline = true + +[*.md] +trim_trailing_whitespace = false + +[Makefile] +indent_style = tab diff --git a/web/.eslintrc.js b/web/.eslintrc.js new file mode 100644 index 0000000..4594b11 --- /dev/null +++ b/web/.eslintrc.js @@ -0,0 +1,3 @@ +module.exports = { + extends: [require.resolve('@umijs/fabric/dist/eslint')], +}; diff --git a/web/.prettierignore b/web/.prettierignore new file mode 100644 index 0000000..25b3bfc --- /dev/null +++ b/web/.prettierignore @@ -0,0 +1,9 @@ +**/*.md +**/*.svg +**/*.ejs +**/*.html +package.json +.umi +.umi-production +.umi-test +mock/*.mock.ts diff --git a/web/.prettierrc b/web/.prettierrc new file mode 100644 index 0000000..94beb14 --- /dev/null +++ b/web/.prettierrc @@ -0,0 +1,11 @@ +{ + "singleQuote": true, + "trailingComma": "all", + "printWidth": 80, + "overrides": [ + { + "files": ".prettierrc", + "options": { "parser": "json" } + } + ] +} diff --git a/web/README.md b/web/README.md new file mode 100644 index 0000000..07afeb7 --- /dev/null +++ b/web/README.md @@ -0,0 +1,15 @@ +# umi project + +## Getting Started + +Install dependencies, + +```bash +$ yarn +``` + +Start the dev server, + +```bash +$ yarn start +``` diff --git a/web/config/config.ts b/web/config/config.ts new file mode 100644 index 0000000..4447dfc --- /dev/null +++ b/web/config/config.ts @@ -0,0 +1,31 @@ +import { defineConfig } from 'umi'; + +export default defineConfig({ + nodeModulesTransform: { + type: 'none', + }, + routes: [{ path: '/', component: 'index' }], + title: 'OceanBase 部署', + fastRefresh: {}, + mfsu: {}, + favicon: '/assets/logo.png', + metas: [ + { + 'http-equiv': 'Cache-Control', + content: 'no-cache, must-revalidate', + }, + { + name: 'data-bizType', + content: 'common', + }, + { + name: 'data-aspm', + content: 'a3696', + }, + ], + headScripts: [ + `!function(modules){function __webpack_require__(moduleId){if(installedModules[moduleId])return installedModules[moduleId].exports;var module=installedModules[moduleId]={exports:{},id:moduleId,loaded:!1};return modules[moduleId].call(module.exports,module,module.exports,__webpack_require__),module.loaded=!0,module.exports}var installedModules={};return __webpack_require__.m=modules,__webpack_require__.c=installedModules,__webpack_require__.p="",__webpack_require__(0)}([function(module,exports){"use strict";!function(){if(!window.Tracert){for(var Tracert={_isInit:!0,_readyToRun:[],_guid:function(){return"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g,function(c){var r=16*Math.random()|0,v="x"===c?r:3&r|8;return v.toString(16)})},get:function(key){if("pageId"===key){if(window._tracert_loader_cfg=window._tracert_loader_cfg||{},window._tracert_loader_cfg.pageId)return window._tracert_loader_cfg.pageId;var metaa=document.querySelectorAll("meta[name=data-aspm]"),spma=metaa&&metaa[0].getAttribute("content"),spmb=document.body&&document.body.getAttribute("data-aspm"),pageId=spma&&spmb?spma+"."+spmb+"_"+Tracert._guid()+"_"+Date.now():"-_"+Tracert._guid()+"_"+Date.now();return window._tracert_loader_cfg.pageId=pageId,pageId}return this[key]},call:function(){var argsList,args=arguments;try{argsList=[].slice.call(args,0)}catch(ex){var argsLen=args.length;argsList=[];for(var i=0;i { + api.modifyHTML(($) => { + // 设置 b 位 + $('body').attr('data-aspm', 'b57206'); + return $; + }); +}; diff --git a/web/mock/createDeploymentConfig.mock.ts b/web/mock/createDeploymentConfig.mock.ts new file mode 100644 index 0000000..83b53b8 --- /dev/null +++ b/web/mock/createDeploymentConfig.mock.ts @@ -0,0 +1,15 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'POST /api/v1/deployments/:name': (req: Request, res: Response) => { + res + .status(200) + .send({ + code: 92, + data: null, + msg: '半质他运步己很自却力效头西效。', + success: true, + }); + }, +}; diff --git a/web/mock/deleteDeployment .mock.ts b/web/mock/deleteDeployment .mock.ts new file mode 100644 index 0000000..ee15b71 --- /dev/null +++ b/web/mock/deleteDeployment .mock.ts @@ -0,0 +1,15 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'DELETE /api/v1/deployments/:name': (req: Request, res: Response) => { + res + .status(200) + .send({ + code: 70, + data: null, + msg: '被或队他少而置面置般类立严无也最。', + success: true, + }); + }, +}; diff --git a/web/mock/finishInstallAndKillProcess.mock.ts b/web/mock/finishInstallAndKillProcess.mock.ts new file mode 100644 index 0000000..41b8198 --- /dev/null +++ b/web/mock/finishInstallAndKillProcess.mock.ts @@ -0,0 +1,13 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'POST /api/v1/processes/suicide': (req: Request, res: Response) => { + res.status(200).send({ + code: 83, + data: null, + msg: '想具心率期头达研产正确转维题。', + success: true, + }); + }, +}; diff --git a/web/mock/getDestroyTaskInfo.mock.ts b/web/mock/getDestroyTaskInfo.mock.ts new file mode 100644 index 0000000..6f38273 --- /dev/null +++ b/web/mock/getDestroyTaskInfo.mock.ts @@ -0,0 +1,31 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/destroy': (req: Request, res: Response) => { + res.status(200).send({ + code: 98, + data: { + total: 100, + finished: 88, + current: '农商江持连无马年布属果下划响问参。', + status: 'processing', + msg: '战整青它指强容张太矿速维种着在按始广。', + info: [ + { + component: '技办边山思边济反动务完由。', + status: 'processing', + result: '造就基资心节美志消路原天放业重清。', + }, + { + component: '济政见为给般动我强人价化白委值法等。', + status: 'error', + result: '增军红说展着连一率别标山五同人度。', + }, + ], + }, + msg: '利党办们南小交查组连法难空。', + success: false, + }); + }, +}; diff --git a/web/mock/getObdInfo.mock.ts b/web/mock/getObdInfo.mock.ts new file mode 100644 index 0000000..b643699 --- /dev/null +++ b/web/mock/getObdInfo.mock.ts @@ -0,0 +1,13 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/info': (req: Request, res: Response) => { + res.status(200).send({ + code: 74, + data: { user: '没点价种但想军约张界委气建张。' }, + msg: '老老县说局建东通面水市论面月就命八光。', + success: false, + }); + }, +}; diff --git a/web/mock/installDeployment.mock.ts b/web/mock/installDeployment.mock.ts new file mode 100644 index 0000000..181a339 --- /dev/null +++ b/web/mock/installDeployment.mock.ts @@ -0,0 +1,15 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'POST /api/v1/deployments/:name/install': (req: Request, res: Response) => { + res + .status(200) + .send({ + code: 67, + data: null, + msg: '根么高力林厂争由公就识非车。', + success: true, + }); + }, +}; diff --git a/web/mock/preCheckStatus.mock.ts b/web/mock/preCheckStatus.mock.ts new file mode 100644 index 0000000..a220a96 --- /dev/null +++ b/web/mock/preCheckStatus.mock.ts @@ -0,0 +1,91 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/precheck': (req: Request, res: Response) => { + res.status(200).send({ + code: 75, + data: { + total: 61, + finished: 98, + all_passed: false, + status: 'default', + message: '温千着除电处每价花这题为持按回采。', + info: [ + { + name: '金丽', + server: '转压队规他难层却只从着无铁往。', + status: 'success', + result: '改没到称深很了流先低五各好反才。', + recoverable: false, + code: '然响空图被收定教她在争工易道。', + description: '年争深除意题样人油很技几变只规天布。', + advisement: null, + }, + { + name: '叶平', + server: '还者代日例场由族则广今建。', + status: 'processing', + result: '儿红己多我步技工即正子万据。', + recoverable: false, + code: '单断然共设打根应地眼面四金族更根。', + description: '红二员步条生少做山极全备。', + advisement: null, + }, + { + name: '秦杰', + server: '路查达地南最外属着小儿参区。', + status: 'processing', + result: '长按委物调身任律后写领断海白。', + recoverable: true, + code: '象细济月种色区权状话花又整增制条。', + description: '进般记形对育内有则信府最生心角。', + advisement: null, + }, + { + name: '袁霞', + server: '变律证必角水群片按系新料等育。', + status: 'success', + result: '提本认路变导议意二们共参规声收叫代。', + recoverable: true, + code: '素写干难者没位带达名火形部七。', + description: '由响立料现见质达产治品济打带生。', + advisement: null, + }, + { + name: '陆伟', + server: '群别党需元质相多么声要系报准。', + status: 'error', + result: '条在风置立效其较京实国半名一。', + recoverable: false, + code: '步状规好交们带应难江内花能组。', + description: '结理合几般学比受率是走年头。', + advisement: null, + }, + { + name: '田杰', + server: '组济当回史适量主因内广去报值然。', + status: 'error', + result: '王明老圆相长展工长条器图快主达都问。', + recoverable: true, + code: '走条面在住极般龙复参料程积今科圆同。', + description: '山石细色写酸气也却些米即四局构前管。', + advisement: null, + }, + { + name: '朱秀英', + server: '太那置拉土现光风五会立天果影设。', + status: 'default', + result: '才而很治海现业家照交写精商。', + recoverable: false, + code: '分生矿划规门准列业则路从格群经根须。', + description: '劳部新保入光正方空马九界千。', + advisement: null, + }, + ], + }, + msg: '车值二清平打经值见型查龙划边示江质。', + success: true, + }); + }, +}; diff --git a/web/mock/precheck.mock.ts b/web/mock/precheck.mock.ts new file mode 100644 index 0000000..02e0733 --- /dev/null +++ b/web/mock/precheck.mock.ts @@ -0,0 +1,15 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'POST /api/v1/deployments/:name/precheck': (req: Request, res: Response) => { + res + .status(200) + .send({ + code: 88, + data: null, + msg: '主四路复离些收素志标算才价具。', + success: true, + }); + }, +}; diff --git a/web/mock/queryAllComponentVersions.mock.ts b/web/mock/queryAllComponentVersions.mock.ts new file mode 100644 index 0000000..2d42b1f --- /dev/null +++ b/web/mock/queryAllComponentVersions.mock.ts @@ -0,0 +1,602 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/components': (req: Request, res: Response) => { + res.status(200).send({ + code: 74, + data: { + total: 87, + items: [ + { + name: '唐勇', + info: [ + { + estimated_size: 94, + version: '化际示见其总研料管加即今红。', + type: 52, + release: '族西被加见于号放影江干地持月。', + arch: '准流石和路联起直制型内场统连电。', + md5: '应步太层极治市九律素例到从别低外品。', + }, + { + estimated_size: 79, + version: '术压林重满始术化声九局求造思目解前。', + type: 53, + release: '便经斗将合什量划加型极领还想量。', + arch: '影音复千华相工感地门亲九斗。', + md5: '者亲形四新理米斗传场小子体形这。', + }, + { + estimated_size: 90, + version: '始指断发表接争变管立研规新因条样周。', + type: 54, + release: '七取每除导称白万出后眼资速。', + arch: '而速中层道具科眼利声龙给圆研问深。', + md5: '月形处所真政革情强应界却安主动向清全。', + }, + { + estimated_size: 61, + version: '量口易广以代史习维物张得当也。', + type: 55, + release: '传查目列变非原开还快族么四至容日。', + arch: '水能外受走民样西接江观候。', + md5: '必候意和关必历志五广学压积总众路。', + }, + { + estimated_size: 98, + version: '系处建和说厂学最提体世己切直一。', + type: 56, + release: '六看才速拉保西始意次者指办。', + arch: '化连片金记术于商天难根节百。', + md5: '位品向白克些工代用对展适所。', + }, + { + estimated_size: 89, + version: '看论价五状气象维再便影过型张电农日。', + type: 57, + release: '但响住系接小联战叫温器京变。', + arch: '阶解根果受方往边合白只压时马点委安带。', + md5: '主各体满度克从青口议今路都满省报支。', + }, + { + estimated_size: 61, + version: '存十极除龙治平形路群党动。', + type: 58, + release: '石选快办便划分法候特群习为员声且科济。', + arch: '由几共外红己易争断论十况引有容解。', + md5: '院空外劳八他值复使度条造红分统级。', + }, + { + estimated_size: 76, + version: '物叫头山质发定三然老这他工。', + type: 59, + release: '备党间时但正任切间按明影被。', + arch: '准油单展结受合五共队划正前离关。', + md5: '飞革出术写育平如况是构她验除先思次。', + }, + { + estimated_size: 96, + version: '圆部花做权人更出今件开而严拉他花次养。', + type: 60, + release: '商建被二去毛再角列学边王强算史。', + arch: '中场时才线元果如原法利如问日品县。', + md5: '回铁东过强心商专至复亲红分山。', + }, + { + estimated_size: 68, + version: '们府层验造为治较适已么然展。', + type: 61, + release: '认眼海务团马造会能道六世便放打油。', + arch: '向七育听照只九全你称动新七能十月五。', + md5: '片十深市持样确人求济处众火造得际心想。', + }, + { + estimated_size: 73, + version: '族年效土团海感快质流算式的西。', + type: 62, + release: '确参子华通组备头红即流切也平须。', + arch: '至红难厂入务支大生你作了时高。', + md5: '快也算克石别研劳导料正住员动复。', + }, + { + estimated_size: 95, + version: '最江发信设米最来难空们总使区而同得队。', + type: 63, + release: '样导林阶准步长候反再先性将平。', + arch: '油基达间从会性白没般称转教。', + md5: '身权又战细任石白同眼样接统知矿结。', + }, + { + estimated_size: 69, + version: '正程她克石具须思其文就发半际出三没。', + type: 64, + release: '织军外反角养快加验时持程热七商认。', + arch: '路老有年其后广所始省立采。', + md5: '族那往问区资什各百确了员而入育日再。', + }, + { + estimated_size: 71, + version: '白构表展山算并铁照山级必龙车长利铁教。', + type: 65, + release: '张要律号人众分目好手样类再专片。', + arch: '实消学则文率上产题先示变化。', + md5: '因权军为身儿调治五去族空者。', + }, + ], + }, + { + name: '尹丽', + info: [ + { + estimated_size: 73, + version: '有发学间周书带用从统就改物确水新关。', + type: 66, + release: '报但转系取口间经数每酸大。', + arch: '置身标取机一有济心做最正价构界市同打。', + md5: '带建个较系深看门教天便在利级段。', + }, + { + estimated_size: 79, + version: '术引火该放程写观位所安别民做。', + type: 67, + release: '强拉种上半半第圆快见收叫你把。', + arch: '去近厂志两结最形位连对力常记完定。', + md5: '但王比会如断头现率价思厂车数与千目也。', + }, + { + estimated_size: 78, + version: '接适名复海行族立者变立年叫格张什商教。', + type: 68, + release: '况海市花反火文间去维劳进形。', + arch: '算争器分际向查与别看回之带规持期专。', + md5: '又机规然主导我适种去八斗或知果容。', + }, + { + estimated_size: 81, + version: '速极素期党县反难算做着马以老规火老快。', + type: 69, + release: '她状王受空书林划取值离放是连白用政。', + arch: '完问老国的北面工定学才东海空导。', + md5: '红该着府对定次一任组把任标种引。', + }, + { + estimated_size: 92, + version: '先片他思此军历格数起示当办且再门产。', + type: 70, + release: '把手心西反路近深时线体战十。', + arch: '连常合住入军位利始音达候。', + md5: '到六我半济界议果美计里包须老。', + }, + { + estimated_size: 93, + version: '满求内每办单且多对选术铁也自分流。', + type: 71, + release: '金应增斯共所光打时角三此军张则质。', + arch: '正万比空候中九全见造务维社。', + md5: '不资接义方边然复便音用圆果。', + }, + { + estimated_size: 96, + version: '千情少直空天东已全知运干业四己热西路。', + type: 72, + release: '对总对养然儿法写太调即音。', + arch: '段代今体产响整利中因经史九术。', + md5: '马处料想她准影分历学省深传题。', + }, + { + estimated_size: 78, + version: '何常积命达转集观往铁变这天产观起。', + type: 73, + release: '备些具西传作日从克一构管消。', + arch: '把通严联院最先就一可被格发作最。', + md5: '龙增许体使花级经劳土济史造造。', + }, + { + estimated_size: 73, + version: '地证生例低他话好进里受步。', + type: 74, + release: '数第员况最党活头对连作众米者照京发。', + arch: '研今离来办并须传走料算很影每论亲。', + md5: '八劳美号矿空口建小体龙音京同关再。', + }, + { + estimated_size: 66, + version: '人名话向接力世速形数易由受增几。', + type: 75, + release: '单放场到多美文线科传术重调认指斯。', + arch: '族研体拉布话发做满并价现回么市头月认。', + md5: '正他话史清术层外期美带己须毛系金。', + }, + { + estimated_size: 79, + version: '中算越京极儿只由该去无个平照。', + type: 76, + release: '存真织力教反少主热马确西压通王整。', + arch: '导前历严的达决合听太周几况易群。', + md5: '局前老受标每以只我与流年现明速组。', + }, + { + estimated_size: 80, + version: '识识世什一海走学件认意往。', + type: 77, + release: '石斯类列林己广变提石月花。', + arch: '进据交立而员所十太解治量上群。', + md5: '住家老毛再业世打且小放历社容得四据。', + }, + { + estimated_size: 81, + version: '界它向关方生体快正王二区往多。', + type: 78, + release: '热音较据济高易什上号识油对。', + arch: '安方先总结细在局而者第年由情即样须。', + md5: '题又低就观现收圆些装以性。', + }, + { + estimated_size: 79, + version: '或较都成产米为从济存响八量光式。', + type: 79, + release: '根习生照义家干圆性七广下际活须才青。', + arch: '都文海斯老值看空民先又做信属据角。', + md5: '应义文认养权成空光去什处受。', + }, + ], + }, + { + name: '龚洋', + info: [ + { + estimated_size: 72, + version: '美间基过生日线之张经社然。', + type: 80, + release: '打根年即争物百数而记阶条关习件验无别。', + arch: '教办正则京派她反委存无及需快办。', + md5: '机产例情有斗边常你且管儿空求式基族。', + }, + { + estimated_size: 90, + version: '己达较无组报度增众成不土业状。', + type: 81, + release: '上效头据时养但用制到声广产意线百员世。', + arch: '候府写论得广没该安强置林称年步六导种。', + md5: '白市从世特现太什算用同思段所必于。', + }, + { + estimated_size: 67, + version: '被拉真安持以所手多对青便维。', + type: 82, + release: '于证集数音验整上导你型中满话大。', + arch: '色民越电把情料中定是气成位效我即情向。', + md5: '属构外几者话反拉第我确史实每做。', + }, + { + estimated_size: 97, + version: '层细而龙器度等安商西广口治极。', + type: 83, + release: '条更前美办起革局亲做便备上开。', + arch: '利素六七种传包议安受几角正资重研。', + md5: '同群华容等身斯明活做然道几住位。', + }, + { + estimated_size: 92, + version: '比好期百记战况受消全人程该建己义化收。', + type: 84, + release: '观式五件却真院往多构完单外接构。', + arch: '体难社力府法好度低西亲用确。', + md5: '史至放问眼设九商且次效真斗。', + }, + { + estimated_size: 91, + version: '得一连自前话列只查志导验你。', + type: 85, + release: '如电件并人世活海众边包重因提织东。', + arch: '界斯定建术上九比心使况已号。', + md5: '就影整必转标大建么小政头王次。', + }, + { + estimated_size: 98, + version: '自加地级生已具劳白商斗生果。', + type: 86, + release: '事报书适事天红已结音己每府原省。', + arch: '门方构是适义和立派作究者因八及看包受。', + md5: '感整并而少素报东非单气化西老。', + }, + { + estimated_size: 94, + version: '家取达会量等当术基四矿油到经前取须。', + type: 87, + release: '前见立八想位力据标方称常白象真。', + arch: '持周后积矿如族声外变千指严入。', + md5: '查等制果并建周图算只记际这之新。', + }, + { + estimated_size: 95, + version: '对情条华命几论国回调下至机快县长员重。', + type: 88, + release: '光现况程重那热马感志众产林就计口。', + arch: '重型照前因阶题车已存容干收着动提土。', + md5: '列天它里难品论电则开满说家。', + }, + { + estimated_size: 79, + version: '从小她因出指六带理见心象值空近土放。', + type: 89, + release: '但强将选王节音圆争任听价斯复。', + arch: '安族会好声构口学龙公史近数思去育。', + md5: '解向对角须江白来道须切就标意农。', + }, + { + estimated_size: 81, + version: '办始合和由行率必等命计她光造。', + type: 90, + release: '叫始响特取音性前强山清自于厂百办管而。', + arch: '入克带回联里高龙分高标积装。', + md5: '发此维问验员江日件求造再正今军什。', + }, + { + estimated_size: 71, + version: '而人切和无则各斯术心理传持。', + type: 91, + release: '准片来听声术安场于事实特南格华并。', + arch: '定万他连利面回光在林动水造着。', + md5: '速办式治低阶当导本平带历断斯何过头事。', + }, + { + estimated_size: 63, + version: '七品比片拉热特论合但低下省越动目。', + type: 92, + release: '亲些克转律计光造根当斗基究制。', + arch: '感于律养万几表使般道识可商老。', + md5: '会文学在技存级路许明准理求太史。', + }, + { + estimated_size: 90, + version: '且他使反什构即千律矿九通价时向着。', + type: 93, + release: '走和连改好好书车种存样百。', + arch: '又府美者道所交照传王使意会状至。', + md5: '行王别她最联现复天七路着料包易并。', + }, + ], + }, + { + name: '万洋', + info: [ + { + estimated_size: 68, + version: '种易结行马把到门叫中过劳段日马引写。', + type: 94, + release: '军式更结则体满布构压张量府往。', + arch: '每斯力确层题它所你加位身感党。', + md5: '如该然米题反声制用心造基支五声本工团。', + }, + { + estimated_size: 70, + version: '干业器太般原先通这条他说正体六期许气。', + type: 95, + release: '它质矿林安如步然数速万过性义。', + arch: '全日市外除题效非期又离处支。', + md5: '号近近造且关件向务土克南形后酸候。', + }, + { + estimated_size: 81, + version: '度需情石先议将单般际决度上非转的律程。', + type: 96, + release: '参一品都意群化连队着直广复并打。', + arch: '海今易为么选构级着越和知则构后程。', + md5: '色次参细技走深种北位全集步先斗。', + }, + { + estimated_size: 74, + version: '和离北质及例性应共该特平深经去。', + type: 97, + release: '格养资半与联化育化周基两资林油。', + arch: '属过气团么备器度加线观上入合情属支本。', + md5: '与便一志包济度使万变或众安。', + }, + { + estimated_size: 88, + version: '进九她精器石北线图低品在算石开须维加。', + type: 98, + release: '别养事斯具对按度海王名值做人米据。', + arch: '克是已方科月性十选示活酸连无山。', + md5: '例几国思农华火指发真非体志统此。', + }, + { + estimated_size: 80, + version: '样受因维基开那料布低米使易但命府之。', + type: 99, + release: '取受器办办二把消向知难难。', + arch: '太指般九要自地办口与上素派条万。', + md5: '说红写七白体文矿家造学作东。', + }, + { + estimated_size: 99, + version: '置元号亲毛意划务斗美严劳六织它长想干。', + type: 100, + release: '劳知局东拉就风群手求界火正些实容连己。', + arch: '容单问山空斯难争相该达世度听八。', + md5: '者完将眼思况期转前想心住头。', + }, + { + estimated_size: 72, + version: '是务者酸于也海相白主和石却关。', + type: 101, + release: '片种须与难但质建达适三向起也水原。', + arch: '单队查看元三而受标列器资量众叫。', + md5: '明造压真图全深头养机高图毛那。', + }, + { + estimated_size: 99, + version: '则细它决且加时化身大小八由革样。', + type: 102, + release: '次百二整气世火价才老明子口红思。', + arch: '什以战商证国联入素本已风持。', + md5: '共南术即造话把实化种子度更六。', + }, + { + estimated_size: 70, + version: '米处合明之会阶用眼话以易间。', + type: 103, + release: '元理月之利是入级华定西建速者习太。', + arch: '十也小太才层民是电织治指。', + md5: '国声龙证龙以国须通流适很般很。', + }, + { + estimated_size: 67, + version: '龙商情看装数通元响律放电白段照。', + type: 104, + release: '权圆度先精体个类江国员最别。', + arch: '价路处国里金管却将新少个支七角论议。', + md5: '此存音系际统县元现代两记劳商。', + }, + { + estimated_size: 83, + version: '而油备强团样口空南消点越小。', + type: 105, + release: '口开变通听太展等养容与石维。', + arch: '最状现油活头养公成收那级一。', + md5: '派但意关给总前特列起想走好。', + }, + { + estimated_size: 63, + version: '该土极直军张根了教题他和带极江实样。', + type: 106, + release: '党石极毛已生度织消真九走农积。', + arch: '则头商有关何结是则事并结达到局。', + md5: '年低带程气步实较面我变日内位调想。', + }, + { + estimated_size: 60, + version: '素样查工称去省行青多去知基公。', + type: 107, + release: '养收造真天义家同型场深他。', + arch: '示器政成手元维型面律阶通文国本。', + md5: '称群命例快资海济准文场过关来么住第。', + }, + ], + }, + { + name: '梁军', + info: [ + { + estimated_size: 65, + version: '从完体林青段行群间团市消无战也候便太。', + type: 108, + release: '由如存易消结业老个林群说使及铁划张。', + arch: '需非品音设起月须求相号物。', + md5: '分知为海身验三新增这照增问。', + }, + { + estimated_size: 86, + version: '火需九质直属级收例也流选强任。', + type: 109, + release: '好日性把县众意党色分拉完列。', + arch: '命非程万单广百应团果话然别内万型当特。', + md5: '指引次总证完干节不厂比的。', + }, + { + estimated_size: 97, + version: '始号万省温么需况第素道统示布越。', + type: 110, + release: '调应压队育着需育厂适山强准学。', + arch: '件识始达己代几维会但王保石。', + md5: '厂不改报深条证产矿我问委两么约月打。', + }, + { + estimated_size: 90, + version: '间集干便容变切料将关总家社入民必些小。', + type: 111, + release: '作理存手精人要流长断得发最真达照。', + arch: '生计组相资局九查位常由格门。', + md5: '没被议却况商先布此少管思元。', + }, + { + estimated_size: 74, + version: '住就划平全院水金增科业改。', + type: 112, + release: '极派华人后自本议原该么引和红题于支酸。', + arch: '权多构二所打问国值元比本入交和。', + md5: '引又值值铁些领满数数等级事面收。', + }, + { + estimated_size: 83, + version: '电选数已要专她会复务机对生十家究处。', + type: 113, + release: '又记代解从边构结只论世别自。', + arch: '候完段包强数府无以值达分按。', + md5: '几土打现花开把解着局准织新共。', + }, + { + estimated_size: 94, + version: '小基属改离除方四任说可只研气三只精。', + type: 114, + release: '信百带常建且低做委龙区导力。', + arch: '管此青话议解多流院件议拉。', + md5: '政他信量这电者该越克体少研想法。', + }, + { + estimated_size: 71, + version: '料长二列加式例律条才收主我济马。', + type: 115, + release: '存王斗劳金达名月带太行关府究称。', + arch: '开个往联切育照来技提金存求政过。', + md5: '油毛派将起指热定外相学查上易更。', + }, + { + estimated_size: 97, + version: '细本许名使一直山研身动观带。', + type: 116, + release: '土做区消始需中求所自体形前收非的。', + arch: '教程中此复很下相派日动总人。', + md5: '事因八易联县东程再所确数许织今温需。', + }, + { + estimated_size: 89, + version: '主力达组次题革江积将几两山规战天。', + type: 117, + release: '为写商交响开然花其来片造龙连长。', + arch: '况又口好头上影济处除建性观话去于严。', + md5: '可将把状斯信深线则斯带花意走段认需。', + }, + { + estimated_size: 83, + version: '建体消传教制什道存争据几要并它小。', + type: 118, + release: '达且之难多理法石步商也速百组。', + arch: '员车但来江们科里只己器我关压根今很数。', + md5: '马导声府而报自农知毛又是众议时精。', + }, + { + estimated_size: 70, + version: '已非机资十持信得历权只解。', + type: 119, + release: '装构因类再以门心米水出火话影上色。', + arch: '确计计二般证头正温去声根无今支回。', + md5: '党般大油治团花多京管教又革全实现。', + }, + { + estimated_size: 73, + version: '亲共且任老习有那系们细万示并道干书。', + type: 120, + release: '内列候斗青称和改道器光制将书上小求。', + arch: '员合济技断手他油西体叫回劳际受。', + md5: '就北反位造满现难七观为取该量。', + }, + { + estimated_size: 98, + version: '深实布复速行水部建界体量标。', + type: 121, + release: '又是回照料如飞正温济级个品业金今位。', + arch: '手东电收量圆运与王理向系没的接支。', + md5: '放质清员而包约声平书高为手组层将得。', + }, + ], + }, + ], + }, + msg: '思派律向解者路除基容压东义立真把时全。', + success: false, + }); + }, +}; diff --git a/web/mock/queryComponentByComponentName.mock.ts b/web/mock/queryComponentByComponentName.mock.ts new file mode 100644 index 0000000..e1c33a4 --- /dev/null +++ b/web/mock/queryComponentByComponentName.mock.ts @@ -0,0 +1,105 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/components/:component': (req: Request, res: Response) => { + res.status(200).send({ + code: 70, + data: { + name: '余霞', + info: [ + { + estimated_size: 93, + version: '济律分的上办还百节实一半公标。', + type: 41, + release: '二八广养成志那量历今节亲路。', + arch: '手造素说值空眼公者斗主天育方传。', + md5: '成无京联风表据华更新劳何多信况打数。', + }, + { + estimated_size: 84, + version: '作规出组着持所眼观京见于再根做张称。', + type: 42, + release: '完什应反必下关消基候青感离收半我并。', + arch: '队斯非存然运作往越该美共专。', + md5: '性人究铁安间直各行构动政专院又口。', + }, + { + estimated_size: 90, + version: '性发战集领力信全细合题又不转约。', + type: 43, + release: '相得观律引名治把身完形天见又教之基线。', + arch: '万率想任见单易改才对族本面是。', + md5: '建明深争山林长此细天口才名片论场。', + }, + { + estimated_size: 74, + version: '天连维众速转之工上将际按问到。', + type: 44, + release: '指与温周却府备便又人能时济家强查情。', + arch: '经做权们持最手族战运关程南交八。', + md5: '马来要备法严眼制思道行以共系干参克。', + }, + { + estimated_size: 61, + version: '目说流器品厂速报全个见之响时。', + type: 45, + release: '越究开亲单在然向眼六没业可门完或务。', + arch: '己千华元马象北易非置拉济使好真非消。', + md5: '日基意决石准色龙或思西产温第张市率越。', + }, + { + estimated_size: 71, + version: '西工面件动张道律越业感变斗海我精里。', + type: 46, + release: '过革划民观界式展周是小示意和一。', + arch: '声集与率北头林但干号通火办月有正机。', + md5: '东光儿除都京却第即然而其入江。', + }, + { + estimated_size: 62, + version: '调转音年上积重标理作族变党表京发完。', + type: 47, + release: '要听无她音示率者七张图量容节完。', + arch: '直实接派八完得目斗度何酸和。', + md5: '很量参打际为界住完进斯发需子指品然。', + }, + { + estimated_size: 98, + version: '子知给以给段作约积按管们思。', + type: 48, + release: '农团节机员除确制上少上则利期为里义话。', + arch: '党中维传心此真传领总民每务。', + md5: '任根保统多队需一好内想质精局。', + }, + { + estimated_size: 83, + version: '儿题须眼求还何线有电总整口适数。', + type: 49, + release: '意已入事委已图解了素容世外外型。', + arch: '目飞思关压六争车品现开许西。', + md5: '量速日般研织合要心需提了交只过办。', + }, + { + estimated_size: 98, + version: '史各毛当放关争时信选指科。', + type: 50, + release: '其史毛样马第认量还老科头信其群。', + arch: '务五明也立委水热头制性之即列型口是。', + md5: '听家几在直之新矿按北响花。', + }, + { + estimated_size: 79, + version: '反了持每青角低关研场广之设。', + type: 51, + release: '响身能观响机克以强水道段省。', + arch: '满合张形常类北红眼多书件真。', + md5: '组少确调正育至火多得必理好。', + }, + ], + }, + msg: '队车毛要点可族个度选品头务养近数很证。', + success: false, + }); + }, +}; diff --git a/web/mock/queryComponentParameters.mock.ts b/web/mock/queryComponentParameters.mock.ts new file mode 100644 index 0000000..16723dd --- /dev/null +++ b/web/mock/queryComponentParameters.mock.ts @@ -0,0 +1,681 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'POST /api/v1/components/parameters': (req: Request, res: Response) => { + res.status(200).send({ + code: 68, + data: { + total: 90, + items: [ + { + component: '心格易科什料半布低志作条。', + version: '名下展形员何化价无多必心解构类。', + config_parameters: [ + { + is_essential: false, + name: '易平', + require: false, + auto: true, + description: '准经表进长车器国工并造特定期。', + type: 1, + default: '并国马西难选资土断复称据毛般七。', + min_value: '半话他只上什通长情造运断。', + max_value: '容写月美体取变何听级但王你装阶次。', + need_redeploy: false, + modify_limit: '好权三果连石文格段南权来干很。', + need_reload: true, + need_restart: true, + section: '给使制张权育当将算己声构五取与非当。', + }, + { + is_essential: false, + name: '杜洋', + require: false, + auto: false, + description: '亲新标个车称形万可过六造理定表速写。', + type: 2, + default: '指即究和先论引难儿划反化整不国毛。', + min_value: '很马回共外人型达商每领十过快步器标。', + max_value: '目风何想格厂须安各把现应。', + need_redeploy: true, + modify_limit: '龙改队许步厂场由山即候明别最满运不。', + need_reload: true, + need_restart: false, + section: '识维及完清参且称九价头术王北论子头。', + }, + { + is_essential: true, + name: '戴平', + require: true, + auto: false, + description: '儿般效布始红般史斯气名听表者之何。', + type: 3, + default: '取到无拉条效适革派价史内次律。', + min_value: '当领集亲质性步面自两接第酸实由成。', + max_value: '间东常际住用形活务风千广许流。', + need_redeploy: true, + modify_limit: '专她率济国快参界义际江周关应且这光风。', + need_reload: true, + need_restart: true, + section: '口流可构分反许术并身放大济。', + }, + { + is_essential: false, + name: '于敏', + require: true, + auto: true, + description: '水火除来价求部身比论进进候号样气。', + type: 4, + default: '上同中级天来口明山已你律支报。', + min_value: '色果由把称去相验月京号市太八教图约。', + max_value: '期向下去活本也府易克结以取。', + need_redeploy: false, + modify_limit: '济府断她声阶建完安变对基金况作。', + need_reload: false, + need_restart: false, + section: '年引时电太低件相想九究想形划此技。', + }, + { + is_essential: true, + name: '叶艳', + require: true, + auto: true, + description: '问华效热统口么同面近带每问己斗。', + type: 5, + default: '识至电到据是效但思才放完现儿深派边。', + min_value: '转展因认说织规以处拉列理其会使。', + max_value: '己族素空形标不是却至经音成已世。', + need_redeploy: true, + modify_limit: '分验义论完步白入并公了再无设育单包。', + need_reload: true, + need_restart: true, + section: '京便和采周率花实认节型表半。', + }, + { + is_essential: false, + name: '金洋', + require: false, + auto: true, + description: '越义林值最包复主较委内观理地。', + type: 6, + default: '共它压工员战工别月走查石开着。', + min_value: '产世把亲除也更思太面机百史决。', + max_value: '光西长应设商型济张在也导群也。', + need_redeploy: false, + modify_limit: '众人革别种受么者马属机易信拉复名六。', + need_reload: false, + need_restart: true, + section: '相被路张可运它及求然所决新业长准治。', + }, + { + is_essential: true, + name: '阎磊', + require: false, + auto: false, + description: '选只新于京片流元用理图带己两它体律算。', + type: 7, + default: '来队基长白美是接叫断点非图增商去作。', + min_value: '每济间之认第花般众年方按。', + max_value: '而便年须党革阶数与类长二用己般。', + need_redeploy: true, + modify_limit: '物海求气除组么断金已明什开队三。', + need_reload: false, + need_restart: true, + section: '受花解清速龙看查复一结制机一。', + }, + { + is_essential: false, + name: '冯强', + require: true, + auto: false, + description: '好子并至管得造小际问志二解。', + type: 8, + default: '整展一万半上进变样必到领有开得。', + min_value: '治着近干以会今或性在族议他较院十。', + max_value: '着战精教么月技清思学历委度规听。', + need_redeploy: true, + modify_limit: '为价主品使十变层道今矿边所别。', + need_reload: false, + need_restart: true, + section: '六它受立义六无领社影需变。', + }, + { + is_essential: false, + name: '沈秀兰', + require: false, + auto: false, + description: '再角话米须作美置万龙别全农毛林。', + type: 9, + default: '改每队还必米问她近层委土步持。', + min_value: '马调龙她志组关工天解量油规合具时。', + max_value: '类建金例展厂厂实备通体求机样。', + need_redeploy: false, + modify_limit: '切代即做报看政导精化条时。', + need_reload: false, + need_restart: true, + section: '身红团通也图切立林总音队路区。', + }, + { + is_essential: true, + name: '许洋', + require: false, + auto: false, + description: '儿无认阶专林委外三明参例国民酸入海何。', + type: 10, + default: '里都个角她并矿亲出变决得受此适只变称。', + min_value: '却派下义即整分角率展命今料六。', + max_value: '间把军需决资许大看性族矿段正给。', + need_redeploy: false, + modify_limit: '间那同国更多各里统圆天消百再。', + need_reload: true, + need_restart: false, + section: '本中报片张北须江程市容真改。', + }, + ], + }, + { + component: '称车列清使约平清同属一色装确料。', + version: '器美建置里的员们何新只质家由需取。', + config_parameters: [ + { + is_essential: false, + name: '史磊', + require: false, + auto: false, + description: '完无少有变直将越气解向导铁百。', + type: 11, + default: '争将除场革技增油而位识马所委。', + min_value: '表很组几维于才数组六我物小发还文。', + max_value: '最真员些权或备由克红专土十率形传油。', + need_redeploy: true, + modify_limit: '起及始很合三们速达识我关素理历话标。', + need_reload: true, + need_restart: false, + section: '她易影动争第度转数马越则记。', + }, + { + is_essential: true, + name: '文娜', + require: true, + auto: false, + description: '快公院指量众务才备江给率志代己越。', + type: 12, + default: '难经相石关并特义地科发共三。', + min_value: '压己后书值说照场律听米日由在眼算。', + max_value: '求经团照作细马外性济能回至全收。', + need_redeploy: true, + modify_limit: '天调满世关电至上地行得张连带无做维。', + need_reload: false, + need_restart: false, + section: '意验但以生划观王物始文教。', + }, + { + is_essential: true, + name: '郑霞', + require: false, + auto: true, + description: '程共该石界般实处一再斯建品儿在取。', + type: 13, + default: '他自达亲类动别问场族在很提生性。', + min_value: '整以前同候他果书于收看最二东走。', + max_value: '特际同广也指义文地收传东没空空委样。', + need_redeploy: true, + modify_limit: '立生达表方越张龙手历图共点铁能不每。', + need_reload: true, + need_restart: false, + section: '只且分和数业马权大花复中议压单身。', + }, + { + is_essential: false, + name: '陆超', + require: false, + auto: false, + description: '老转说计参共北并对我达象最毛适。', + type: 14, + default: '土矿难解两或海温基建支关面。', + min_value: '是区色面除持必毛米去教当件商。', + max_value: '称习对影何北东给共争动带。', + need_redeploy: false, + modify_limit: '些写之作争量身完流身什号却周管定子达。', + need_reload: true, + need_restart: false, + section: '才格也声定地约单酸用变委对长。', + }, + { + is_essential: false, + name: '杜平', + require: true, + auto: true, + description: '四成道队北基月亲然次议斯米文解。', + type: 15, + default: '志主花性片想市是速低型斯难领话。', + min_value: '称界当共史很调事对林持实不。', + max_value: '布交省度从那其术号等已己见农认体。', + need_redeploy: true, + modify_limit: '则安体与其王增重走过文应管建业段。', + need_reload: false, + need_restart: false, + section: '动公往建五确一于报快从斯元强。', + }, + { + is_essential: true, + name: '龚洋', + require: false, + auto: false, + description: '众般商种速油了包美即备如始响。', + type: 16, + default: '间价支位按东六能根由写后速深基类统北。', + min_value: '学她节列选选文即心备报统果细。', + max_value: '成前说式党之精团广矿织感。', + need_redeploy: true, + modify_limit: '议引出许商做号世后克又光上因力证。', + need_reload: false, + need_restart: true, + section: '员自式东受外斗养节出住龙几部。', + }, + { + is_essential: true, + name: '尹秀英', + require: false, + auto: false, + description: '长效织新社委变张江面定你要。', + type: 17, + default: '间了还候装照证公土上专省省又影列事。', + min_value: '科证节民南据很不回天类发者此。', + max_value: '场调得积手其新重情矿这变十。', + need_redeploy: false, + modify_limit: '门关林派志全响两严展正是果而得律。', + need_reload: false, + need_restart: true, + section: '头党从存决子地统数她建间组。', + }, + { + is_essential: true, + name: '程霞', + require: false, + auto: false, + description: '理飞思除阶国道正处段千万没头收前。', + type: 18, + default: '院海相各四统须县把运称可越新便问土。', + min_value: '风中意度工大光子你也书百了。', + max_value: '军难导况空离石研示式成该政子年六。', + need_redeploy: true, + modify_limit: '团段该二果交种识第给采也员办满。', + need_reload: true, + need_restart: true, + section: '至己组响听系中所素该具酸江除身年到。', + }, + { + is_essential: true, + name: '赵杰', + require: true, + auto: false, + description: '速四世自空联清状安快基算构条革之名但。', + type: 19, + default: '步深据再至完在多白道加六年响。', + min_value: '前声军由亲各满深体习求影周意劳从。', + max_value: '斯织议马求看统平龙引革北适术。', + need_redeploy: false, + modify_limit: '使质任四深见形林西联教定西发军。', + need_reload: false, + need_restart: true, + section: '运好地般象才该入共易增流型持效国速。', + }, + { + is_essential: true, + name: '汪霞', + require: true, + auto: true, + description: '西大之话受火置利外问已细门准风。', + type: 20, + default: '带在同打严派并可律成于拉方。', + min_value: '参发县机新全命并青名成治。', + max_value: '阶存称目期品干下表给亲运山江。', + need_redeploy: true, + modify_limit: '难子育太增断包名华受事际积县青将走。', + need_reload: false, + need_restart: false, + section: '子经史行区们带标片因意做。', + }, + ], + }, + { + component: '二米命里委变年属于队太取。', + version: '求质器风解派况第战手九八证适。', + config_parameters: [ + { + is_essential: false, + name: '汪娜', + require: true, + auto: true, + description: '指清思中北较信包才先水带算。', + type: 21, + default: '直员数林严气九太党或质数身素又响参。', + min_value: '调我无面方先展打常电组记次。', + max_value: '热全步别日在设强局也情信各报标拉自比。', + need_redeploy: false, + modify_limit: '电世备快最先决感酸调难队门文族广。', + need_reload: true, + need_restart: false, + section: '装习却任马西维于商去重矿位。', + }, + { + is_essential: false, + name: '马磊', + require: false, + auto: true, + description: '标法们做细达级算较速只至个红。', + type: 22, + default: '权始这采段任须长型石天年后。', + min_value: '国此力我眼为要教江铁头军改还江。', + max_value: '公立称联米例表件便表子系名取边运打七。', + need_redeploy: true, + modify_limit: '深个年万华成应清各劳片平步完团体。', + need_reload: true, + need_restart: false, + section: '便受下在确证明完进百同酸。', + }, + { + is_essential: true, + name: '雷娟', + require: true, + auto: false, + description: '规际须如土斯人增过于保报。', + type: 23, + default: '产则品受列压可生少人器色家与派度思。', + min_value: '事技林是火前变边定统出能这下相思片论。', + max_value: '几象就四联话美好相西红低还叫定起今。', + need_redeploy: false, + modify_limit: '之入内果那学比具精往个场同军。', + need_reload: true, + need_restart: true, + section: '些满解量集设百斗计取离千入回必实干。', + }, + { + is_essential: true, + name: '白刚', + require: false, + auto: true, + description: '断图去斗毛展合消记人系五消第价应型。', + type: 24, + default: '直证度同产百界段据列议们用厂。', + min_value: '成真存需片周织具克正极马党合可铁。', + max_value: '气积查军四江王打领自济知几线新六。', + need_redeploy: true, + modify_limit: '离除口规展西众几好克毛除传一术力千界。', + need_reload: false, + need_restart: true, + section: '太科技思得族江不管连层决际。', + }, + { + is_essential: false, + name: '吴秀英', + require: false, + auto: false, + description: '做能红权意把政论半认行本温类当际气。', + type: 25, + default: '美情她林华铁院除六查社日日相。', + min_value: '年精些八话没建质式青酸面土。', + max_value: '写商空周二置东世立历除除五。', + need_redeploy: true, + modify_limit: '住阶有争但用根半厂形线军非确验自。', + need_reload: false, + need_restart: true, + section: '书很性型斯将质劳加属身布者务。', + }, + { + is_essential: false, + name: '龚艳', + require: false, + auto: false, + description: '金才基先教产些亲打例之光还代。', + type: 26, + default: '酸近正文究周已么最度增机热。', + min_value: '系业质增放收图办只只计习万局。', + max_value: '会根前报理了美则业然律已设。', + need_redeploy: true, + modify_limit: '斗真一路建流许志非子取明同话称经。', + need_reload: false, + need_restart: false, + section: '史器准太拉回商四电节半义整两。', + }, + { + is_essential: false, + name: '金敏', + require: false, + auto: true, + description: '其即西车情布用称物系表运何红身角眼。', + type: 27, + default: '图界却定商必养调酸土广提都议油。', + min_value: '导提商小院特指才决边响收问己。', + max_value: '成高许开列便道行型转都并热精。', + need_redeploy: false, + modify_limit: '收术也热发水区龙最也相力风变展指月。', + need_reload: false, + need_restart: true, + section: '原该石手见体权律今际建到部自。', + }, + { + is_essential: false, + name: '钱磊', + require: false, + auto: true, + description: '系认百信界必团算外号适受府政信力。', + type: 28, + default: '将为地权上划相事中支增力红装。', + min_value: '了半般变连达重就会花况型率外采从布点。', + max_value: '严技接需向办写料空展日对白报置它面。', + need_redeploy: true, + modify_limit: '着其联位联火布据为要器水展。', + need_reload: false, + need_restart: true, + section: '本般明过你制院专王机后情事火。', + }, + { + is_essential: true, + name: '卢娜', + require: false, + auto: false, + description: '高定反个六或干才二动理作世门可马。', + type: 29, + default: '里表构应命强受许水好但打就总车。', + min_value: '场就品构四飞器名劳走半社区。', + max_value: '得法去张角位元段高只合部来性按七每复。', + need_redeploy: false, + modify_limit: '委件该高观造们安们情院色四华他。', + need_reload: true, + need_restart: false, + section: '道信持标于连口们克道济人习。', + }, + { + is_essential: false, + name: '崔霞', + require: false, + auto: true, + description: '月半它月产响能周北战位东么容。', + type: 30, + default: '除南已来部素土种温油报点处。', + min_value: '美天只油组术据情九点出前飞为。', + max_value: '住题年此派知维支红战增她她切政算。', + need_redeploy: false, + modify_limit: '断发至联干按同建片本门过向车常。', + need_reload: false, + need_restart: false, + section: '干外代果克他色委证段生省领它里温处。', + }, + ], + }, + { + component: '或需外酸采月来程并持数果向化之取京。', + version: '八达往八报思们知而直取众。', + config_parameters: [ + { + is_essential: true, + name: '于静', + require: false, + auto: true, + description: '列四个真压石近存选部严立位着。', + type: 31, + default: '规目需可儿转是省种三市规体社认路。', + min_value: '方干三说育明空治维况保许论或还候。', + max_value: '省里便张线严圆结无文心者世务又织。', + need_redeploy: false, + modify_limit: '形强年她连此省选该用信华为又是战。', + need_reload: true, + need_restart: true, + section: '华己资保几局准习可收几近党总设照。', + }, + { + is_essential: true, + name: '程敏', + require: false, + auto: false, + description: '声重南北走府六或门类速大系高象。', + type: 32, + default: '部物由历去江群级导证小光学受。', + min_value: '导料千增有时己构角打众度料界精题。', + max_value: '毛位火万军率外带完国界持与。', + need_redeploy: false, + modify_limit: '上量这集关影精积历称压压种。', + need_reload: false, + need_restart: false, + section: '量条县相体共光象点心方提布低知。', + }, + { + is_essential: false, + name: '汪强', + require: false, + auto: false, + description: '样低种满却验象三头压道为难着产它八。', + type: 33, + default: '不经火近原商月龙多风石革于价。', + min_value: '日此身而样矿际想克亲越十决厂动。', + max_value: '头格须整易以复反学千京相八。', + need_redeploy: false, + modify_limit: '验小边比于得行然压员许民较自。', + need_reload: false, + need_restart: true, + section: '实王都矿水例白合存亲构样习量术何立。', + }, + { + is_essential: false, + name: '乔刚', + require: true, + auto: true, + description: '及共完月与消物能看题克十片好响加。', + type: 34, + default: '时名较式用位今压元界委族多影达复。', + min_value: '战果深但话格门处机新水总南团。', + max_value: '当热音当油义精选究包解就面。', + need_redeploy: true, + modify_limit: '干人政命点律按样般委全起心可。', + need_reload: true, + need_restart: true, + section: '厂天资记毛那于线究间满团方青前调体。', + }, + { + is_essential: true, + name: '许艳', + require: true, + auto: false, + description: '效该习金象什油总油用酸查具我上。', + type: 35, + default: '同值委建社法酸例名方因具那值海军张。', + min_value: '后严北身家严方真还气题道。', + max_value: '四手期出严第会即写米红高老价流者对。', + need_redeploy: false, + modify_limit: '生水的计能容示内个于效打片平老众。', + need_reload: true, + need_restart: true, + section: '龙队带照石容任上内北王保。', + }, + { + is_essential: false, + name: '田杰', + require: false, + auto: true, + description: '几切白商院资线而取情强半叫以技。', + type: 36, + default: '东想斗斯思位题复提般心角造人圆声干。', + min_value: '低单却文事积比他取须确及装快上。', + max_value: '因先根名相约再支传状积她表区取斯子解。', + need_redeploy: false, + modify_limit: '把至维而长生九一运矿速风万件权美总几。', + need_reload: true, + need_restart: true, + section: '手话参际作养员金清应总水权。', + }, + { + is_essential: false, + name: '苏强', + require: false, + auto: false, + description: '才风原象农老常行关产天造律。', + type: 37, + default: '着也每都低至此面权因他为提消细办。', + min_value: '主连又何快连难见任色总然领我。', + max_value: '于她亲调比任况况型参命无眼车适切。', + need_redeploy: true, + modify_limit: '节流人身重者性容维位五那严没算。', + need_reload: true, + need_restart: false, + section: '其外信却任国什照没花且却斗。', + }, + { + is_essential: false, + name: '毛敏', + require: false, + auto: true, + description: '步心老设特压因根子特照年应象道林。', + type: 38, + default: '解经两价叫度会联情铁一角转并并主眼。', + min_value: '将心列查准林史或参常公准数高工观技使。', + max_value: '以展科后响却复米圆再那但火头。', + need_redeploy: false, + modify_limit: '节边按张生起干向拉来机定己无。', + need_reload: false, + need_restart: true, + section: '近节林什金场周直同列具造公。', + }, + { + is_essential: true, + name: '程娟', + require: true, + auto: true, + description: '消受情关和与提集龙产果已满研族造。', + type: 39, + default: '分象最层任许毛毛增入或也点。', + min_value: '队界离矿运美叫管出土算本七还反战解社。', + max_value: '信下高何拉西期派然相论直严效然着阶。', + need_redeploy: true, + modify_limit: '照法毛龙单维千例学件离八论业。', + need_reload: true, + need_restart: false, + section: '单群条众商米制体存政每越作便身花。', + }, + { + is_essential: false, + name: '吴秀兰', + require: false, + auto: true, + description: '其题阶江且面题关结并组素记去此之期听。', + type: 40, + default: '平达作水军而九线通选最条质常压。', + min_value: '例层了常增理其南中了江因间。', + max_value: '米上政色业属外是省温问对位。', + need_redeploy: true, + modify_limit: '权温越多马办时强拉反他即学存平。', + need_reload: false, + need_restart: true, + section: '系事整中派对各今现或深风人起改属。', + }, + ], + }, + ], + }, + msg: '两立过了火有听团院场与包备员类。', + success: true, + }); + }, +}; diff --git a/web/mock/queryConnectInfo.mock.ts b/web/mock/queryConnectInfo.mock.ts new file mode 100644 index 0000000..28e29ba --- /dev/null +++ b/web/mock/queryConnectInfo.mock.ts @@ -0,0 +1,94 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/connection': (req: Request, res: Response) => { + res.status(200).send({ + code: 88, + data: { + total: 68, + items: [ + { + component: '委起么所只参广水看于始立称图。', + access_url: 'https://umijs.org/', + user: '越西发知可新现细收决广于张相。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '个听带我实量高回他五外热命基。', + access_url: 'https://github.com/umijs/dumi', + user: '位金连府养思点天称报观件究特。', + password: 'string(16)', + connect_url: 'https://ant.design', + }, + { + component: '而义风置任问资目标质路候会厂达气性金。', + access_url: 'https://procomponents.ant.design/', + user: '那油些这重交百相增正群及。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '走来第响长自证计今上准干走放包。', + access_url: 'https://umijs.org/', + user: '日王生等为技北观无始好证连动器相己。', + password: 'string(16)', + connect_url: 'https://github.com/umijs/dumi', + }, + { + component: '里口经红直类技许调离织如一。', + access_url: '', + user: '头反还明物备京天主指走住如亲。', + password: 'string(16)', + connect_url: 'https://github.com/umijs/dumi', + }, + { + component: '各志线子江六采议定管没华一里素自。', + access_url: 'https://umijs.org/', + user: '必阶而半还细而选队或中也共教。', + password: 'string(16)', + connect_url: '', + }, + { + component: '无采文向速层给月节常传住县山。', + access_url: 'https://github.com/umijs/dumi', + user: '先叫还矿线群高则除算按切加。', + password: 'string(16)', + connect_url: 'https://github.com/umijs/dumi', + }, + { + component: '件思更义研由严格按有往西事都面很电。', + access_url: 'https://ant.design', + user: '层书题按加百进声通求应团便。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '直它层件达整种义划生保亲天成府才成。', + access_url: 'https://github.com/umijs/dumi', + user: '种教住看同图运才力飞音将省真传。', + password: 'string(16)', + connect_url: 'https://umijs.org/', + }, + { + component: '统按有最素及样身先真可列算须约划。', + access_url: 'https://ant.design', + user: '内料则实民却极电使用商界接。', + password: 'string(16)', + connect_url: 'https://github.com/umijs/dumi', + }, + { + component: '前由连导问你务位酸办政较北色划。', + access_url: 'https://github.com/umijs/dumi', + user: '做快使劳感类类六即节林革把式年然象收。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + ], + }, + msg: '素于件具酸率道重识不战不动史立图。', + success: true, + }); + }, +}; diff --git a/web/mock/queryConnectionInfo.mock.ts b/web/mock/queryConnectionInfo.mock.ts new file mode 100644 index 0000000..d147803 --- /dev/null +++ b/web/mock/queryConnectionInfo.mock.ts @@ -0,0 +1,87 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/connection': (req: Request, res: Response) => { + res.status(200).send({ + code: 71, + data: { + total: 80, + items: [ + { + component: '复关量报文自对且铁京断外然严用。', + access_url: 'https://preview.pro.ant.design/dashboard/analysis', + user: '精转强去查难叫点为们多角开以响个。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '书热集全王政验下业间历们。', + access_url: '', + user: '济调标号其集类层家层求革京决况几。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '事高原火还形总群与各和关保场第六毛。', + access_url: 'https://umijs.org/', + user: '料维作性等好相发快况况究打于力论。', + password: 'string(16)', + connect_url: 'https://ant.design', + }, + { + component: '响马住二少基高压称听展眼。', + access_url: 'https://umijs.org/', + user: '率至直能属主定支体列深次它此即身世应。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '支采商里平导专提般展图工前动克史。', + access_url: '', + user: '水例此量队于全无义际观进领机。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '风真又验领这和只现装积世集精九展极元。', + access_url: 'https://umijs.org/', + user: '更由准不今八性增风林身层带总水给。', + password: 'string(16)', + connect_url: 'https://ant.design', + }, + { + component: '志活须毛重今然与圆市上理议。', + access_url: '', + user: '酸认决标华行需角育空水通劳去类。', + password: 'string(16)', + connect_url: 'https://umijs.org/', + }, + { + component: '越利积查例准议文太资却住样二证而。', + access_url: 'https://procomponents.ant.design/', + user: '热解近采准实改持断进存专义。', + password: 'string(16)', + connect_url: 'https://ant.design', + }, + { + component: '县资了整合话美大布据好先类型次存张。', + access_url: 'https://procomponents.ant.design/', + user: '组条形织率形难观观处场书每领边。', + password: 'string(16)', + connect_url: 'https://procomponents.ant.design/', + }, + { + component: '求了看按往果信因矿体例名给反低作斯温。', + access_url: 'https://umijs.org/', + user: '全组听影些主气示事给给科表次制化和劳。', + password: 'string(16)', + connect_url: 'https://github.com/umijs/dumi', + }, + ], + }, + msg: '样林里花外同见小算党参石石色日影队。', + success: true, + }); + }, +}; diff --git a/web/mock/queryDeploymentConfig.mock.ts b/web/mock/queryDeploymentConfig.mock.ts new file mode 100644 index 0000000..670f393 --- /dev/null +++ b/web/mock/queryDeploymentConfig.mock.ts @@ -0,0 +1,960 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name': (req: Request, res: Response) => { + res.status(200).send({ + code: 84, + data: { + name: '邓秀兰', + config_path: '何农精感治直农们理时结革发分温样。', + status: 'error', + config: { + auth: { + user: '放识己打越度比级其压准治斗家历。', + password: 'string(16)', + port: 91, + }, + components: { + oceanbase: { + component: '却心少阶体商解即面确走活什示东直素识。', + appname: '达快任走确例化就利农矿之军。', + version: '京制织但年江点发装认老社。', + release: '引广算光号求关以正离层会是众不。', + package_hash: '半属然究积称论至局第接很对量说。', + mode: '准太该林上开比目务民白省院想山应。', + root_password: '正着西委写已计其当热且条至热团安除。', + mysql_port: 77, + rpc_port: 77, + home_path: '这更眼状求级回引三院也通长。', + data_dir: '之之取然才白水往离布日使管时走算器。', + redo_dir: '主十色由从生记化始效二政作美千商到信。', + parameters: [ + { + key: 122, + value: '信表大共传连飞周技导造想。', + adaptive: false, + }, + { + key: 123, + value: '什派被布知四满及大专想美线商实着。', + adaptive: false, + }, + { + key: 124, + value: '十养你开等美验力文号场资办角米。', + adaptive: false, + }, + { + key: 125, + value: '书年制治决华式类社矿价系光。', + adaptive: true, + }, + { + key: 126, + value: '格市因华见但特信且式四平发对集。', + adaptive: true, + }, + { + key: 127, + value: '维质利快合干体说变自特成约连象。', + adaptive: false, + }, + { + key: 128, + value: '改定长群回比便结单集传和定只公学积。', + adaptive: true, + }, + { + key: 129, + value: '群约小该满东斯带市通中受数。', + adaptive: true, + }, + { + key: 130, + value: '务真解产斗清但约行八高带提府拉。', + adaptive: false, + }, + { + key: 131, + value: '识色但元次验划该族着非看。', + adaptive: false, + }, + { + key: 132, + value: '农件持立府队市入人清东劳比但米。', + adaptive: true, + }, + { + key: 133, + value: '者东六林和族时务较受京空思具通广战带。', + adaptive: false, + }, + { + key: 134, + value: '后专增也天任身示农界决算较不切属。', + adaptive: false, + }, + { + key: 135, + value: '这总更众按导加影布党始收成易。', + adaptive: false, + }, + ], + topology: [ + { + name: '马磊', + rootservice: '列系安原分场命术果格集布保。', + servers: [ + { ip: '关期研造同精花片今三备亲。', parameters: {} }, + { ip: '于公学先六日家利应度温东发值。', parameters: {} }, + { ip: '程做率做立个十法手位法题基品团。', parameters: {} }, + { ip: '近社目矿以各立思由布上战则。', parameters: {} }, + { ip: '想眼二青素况般设类体将片低般题。', parameters: {} }, + { ip: '儿种清其位动住人问将种却定。', parameters: {} }, + { ip: '转任光正国但社向代关科内求。', parameters: {} }, + { + ip: '给世政性产只山产第权比图个里量速指。', + parameters: {}, + }, + { ip: '热月由却合展近看里收际越信。', parameters: {} }, + { ip: '气局革就切况八去统今音族由经。', parameters: {} }, + { ip: '见例交组什近使条它给党设出结。', parameters: {} }, + { + ip: '况极导点选验装石则说列重再或分点。', + parameters: {}, + }, + { + ip: '专段论义心识加须产大受精三造调常采经。', + parameters: {}, + }, + ], + }, + { + name: '秦涛', + rootservice: '具此斗识清为出或白百会价济决同。', + servers: [ + { + ip: '运写格革体政院自马于展意级适电数周。', + parameters: {}, + }, + { ip: '便他办律定山万回儿素例价。', parameters: {} }, + { + ip: '际会织型红受受却制省该市济江海音器。', + parameters: {}, + }, + { ip: '农长但你实情各青场级提土地民具。', parameters: {} }, + { ip: '须劳造意我直管前由总却电完手。', parameters: {} }, + { + ip: '因两要来争工领深利放队线引状常重据。', + parameters: {}, + }, + { + ip: '取听直技话路象千教发示族为易音场色。', + parameters: {}, + }, + { ip: '手他并理众日状只因活活切层。', parameters: {} }, + { ip: '什被务政出也毛话局取京本力心现。', parameters: {} }, + { + ip: '也据大特却常引经重权厂存展报响局。', + parameters: {}, + }, + { ip: '干用方备角管上办七例制设转。', parameters: {} }, + { + ip: '你离时设采天便点气养主风属打打响。', + parameters: {}, + }, + { ip: '或果造天信起问发都在车然世快亲。', parameters: {} }, + ], + }, + { + name: '邱霞', + rootservice: '记团府京委业任造团道土准住问具。', + servers: [ + { ip: '及她构是第适明科儿意现油离业金。', parameters: {} }, + { + ip: '不权青报门精认马组级文具么影低军。', + parameters: {}, + }, + { ip: '切史验族代上高且接而门各究其。', parameters: {} }, + { ip: '叫律四真音府热个才别花提说。', parameters: {} }, + { + ip: '动空者与对活话人阶么象但花海华最党众。', + parameters: {}, + }, + { + ip: '根才应农的其具光从且且条江构织队。', + parameters: {}, + }, + { + ip: '性按指格办设段于号科并断飞管们史科。', + parameters: {}, + }, + { ip: '因用低圆金图立支府月需火。', parameters: {} }, + { + ip: '切边大何证无好每效教条意加劳就证。', + parameters: {}, + }, + { ip: '引细相也子些的效土文满内之角还。', parameters: {} }, + { + ip: '万已高然想究会例农土正支志一办低。', + parameters: {}, + }, + { + ip: '养型花里就支战之飞物样少千省求江业。', + parameters: {}, + }, + { ip: '者装证单通加些县严容阶八听。', parameters: {} }, + ], + }, + { + name: '魏娜', + rootservice: '把中酸通平工通律率斗小意。', + servers: [ + { + ip: '具北分周照能为面任而家为置值入会。', + parameters: {}, + }, + { + ip: '向想造究走快管构集先题听将两机真。', + parameters: {}, + }, + { ip: '力实始证说青代关必且长劳小。', parameters: {} }, + { ip: '七两体状门口劳活入体矿些精高。', parameters: {} }, + { ip: '多些土般什院进书天称精们。', parameters: {} }, + { + ip: '选速养车特者计须文万精实北真利条。', + parameters: {}, + }, + { ip: '适何行根或花了无产委称观者。', parameters: {} }, + { ip: '九有速京提运积论果约济业号场。', parameters: {} }, + { + ip: '整平律列论产形证许军海义火天收切。', + parameters: {}, + }, + { ip: '接水片义厂根品得思南体办应或。', parameters: {} }, + { ip: '其决向系通后受表会类志定张实些。', parameters: {} }, + { ip: '到完子者局增命人四己毛放眼。', parameters: {} }, + { ip: '间队节用万年上百族往况上时复。', parameters: {} }, + ], + }, + { + name: '戴敏', + rootservice: '如以开样亲什价积回面资八。', + servers: [ + { + ip: '全较以它据研两张位员局或机但细毛。', + parameters: {}, + }, + { ip: '水系据外话压采型样但规第。', parameters: {} }, + { ip: '体直养历取省北决山领集前深。', parameters: {} }, + { + ip: '当精立质段造品为报王得管商代其际。', + parameters: {}, + }, + { ip: '族住以消更等务京料队直装。', parameters: {} }, + { + ip: '前半设往理话半通知较京况了经广省。', + parameters: {}, + }, + { + ip: '安的电回持会却单性业在基资商就记。', + parameters: {}, + }, + { ip: '共织划问任矿这只型道可术。', parameters: {} }, + { ip: '联计观上离话细主风子容龙持。', parameters: {} }, + { + ip: '许一极展的青查三号于受响越价法写被后。', + parameters: {}, + }, + { ip: '议风了以革月文作题打华观三收六。', parameters: {} }, + { + ip: '习资世她北地政面九必体步制造张期。', + parameters: {}, + }, + { ip: '同造口术马原市斯法马法空值山即。', parameters: {} }, + ], + }, + { + name: '沈伟', + rootservice: '千精厂争往来来价不毛科门群亲组高。', + servers: [ + { ip: '体她音精最包快观边等者什易造。', parameters: {} }, + { + ip: '列能家参风领石达能与压属特如劳大低只。', + parameters: {}, + }, + { + ip: '外铁志油强是条区便入它候小族便音王。', + parameters: {}, + }, + { + ip: '发报通系易标标由十效么美没运省般它。', + parameters: {}, + }, + { + ip: '油圆装近级共内器术切式目农划从复。', + parameters: {}, + }, + { ip: '维京传议东史了等提市进米元装。', parameters: {} }, + { + ip: '收问质真众等很切清证她声四断门段。', + parameters: {}, + }, + { + ip: '布便许车放红节划可应题整受才验眼不需。', + parameters: {}, + }, + { ip: '派过知入称确定七高向布能快。', parameters: {} }, + { ip: '话况进问是便日华万交研动半东。', parameters: {} }, + { + ip: '过口东原且往国工统线术接应要反把复。', + parameters: {}, + }, + { + ip: '着性色期都正以算本生采其文复太群规线。', + parameters: {}, + }, + { ip: '品非节你加压红理整权按实音。', parameters: {} }, + ], + }, + { + name: '贺娜', + rootservice: '铁看化件办高便回中性前据列称参。', + servers: [ + { + ip: '教任列名铁办命加期整研难战深这直。', + parameters: {}, + }, + { ip: '争用且你代张织速导布需业。', parameters: {} }, + { + ip: '立际和太转由两成动年水改路立如习。', + parameters: {}, + }, + { ip: '日火又所北做老叫府主增文你。', parameters: {} }, + { ip: '何准压权满计际别除三农油问。', parameters: {} }, + { ip: '标红单力一来步南风在革立该带院。', parameters: {} }, + { + ip: '确些角二政造专制业文原断下加至红资。', + parameters: {}, + }, + { + ip: '单具几术什支业内飞完之得查院教毛。', + parameters: {}, + }, + { ip: '拉空全她题备由次即层山计提间。', parameters: {} }, + { + ip: '其有五容度圆查当主叫展点越复年号影。', + parameters: {}, + }, + { + ip: '率离以省儿展交太记素志被做两深件得细。', + parameters: {}, + }, + { + ip: '多图国这且争选于放别调如来厂白她。', + parameters: {}, + }, + { ip: '国意段十类矿习各为况住我。', parameters: {} }, + ], + }, + { + name: '袁洋', + rootservice: '管千眼表进更白深建出专月已。', + servers: [ + { + ip: '取区线整实标最力近什斗别片又日任低一。', + parameters: {}, + }, + { ip: '你代层必有家整具次东有研。', parameters: {} }, + { + ip: '教生场会律后完段则划求也广革局加治。', + parameters: {}, + }, + { ip: '口场照做县传世外重识相解志。', parameters: {} }, + { ip: '知正示约花被这速义议国该根。', parameters: {} }, + { + ip: '联装持通门众具真级很位养名它党写。', + parameters: {}, + }, + { + ip: '西众除在非决六然采写平志最片给给对点。', + parameters: {}, + }, + { ip: '决支过面九持新通空位对标门。', parameters: {} }, + { ip: '大员育样大他江花只世集放半因劳。', parameters: {} }, + { ip: '场图类明就又支光江管条需空。', parameters: {} }, + { + ip: '往实复九济作学后界派平等信造广心儿山。', + parameters: {}, + }, + { ip: '还重基学同体治生色新又划外眼。', parameters: {} }, + { ip: '量个世四运理安查花九放土府织就。', parameters: {} }, + ], + }, + { + name: '杜超', + rootservice: '给中能日商省学层名民志下速组技。', + servers: [ + { ip: '厂要二设见少积社料交起争人个。', parameters: {} }, + { ip: '族样品历照派金半品交接小。', parameters: {} }, + { ip: '思取法千政月易红身况制候前江规。', parameters: {} }, + { + ip: '动性意元示角机今部斯报候需划他政。', + parameters: {}, + }, + { ip: '过北者位当种整越被律把因全。', parameters: {} }, + { ip: '是儿分清过没常关上参建各。', parameters: {} }, + { ip: '万合五老术与七此第离导报接严相。', parameters: {} }, + { ip: '学值些适商建战正关作党整治会。', parameters: {} }, + { ip: '确响路其精是越制就于后见院金力。', parameters: {} }, + { ip: '列展酸金太压南电度当所世科办团。', parameters: {} }, + { ip: '去准斗给满五图联约不处论。', parameters: {} }, + { ip: '必状门边商安根例上千立毛严何。', parameters: {} }, + { ip: '验单基海白直情包极速专县育先花。', parameters: {} }, + ], + }, + { + name: '邓娜', + rootservice: '省处展作亲场展交县家五身看影团干位。', + servers: [ + { + ip: '品报毛然满克可她百约技运圆府备其中。', + parameters: {}, + }, + { + ip: '眼小高门必明与林土看后能分确使将复题。', + parameters: {}, + }, + { ip: '商由观务等联公毛可身马上话。', parameters: {} }, + { + ip: '到工状油特商体白二两群规层战化白。', + parameters: {}, + }, + { + ip: '是按活车由子石办二色十选总音厂期。', + parameters: {}, + }, + { ip: '规史动者变平见非构识设单传。', parameters: {} }, + { ip: '地事际接从立向广共小名社西备给。', parameters: {} }, + { ip: '之专识里号果连思料年每许头。', parameters: {} }, + { + ip: '活实节具众风省点将海口所日适内新天。', + parameters: {}, + }, + { ip: '东解商拉石率是五济毛利那温。', parameters: {} }, + { ip: '品心式周完种两济无片次大约所次。', parameters: {} }, + { ip: '适什知组建向步族内往级七千。', parameters: {} }, + { ip: '市头美律议内集质始约们部观。', parameters: {} }, + ], + }, + { + name: '林勇', + rootservice: '用价立局验证何林社处今明保素统入。', + servers: [ + { ip: '四律史权对型象产时日而界矿。', parameters: {} }, + { ip: '也用点众管集中行想口团极运。', parameters: {} }, + { ip: '身合调无千成记青体特程系关。', parameters: {} }, + { + ip: '准种性存提把进八从写革会信空从名金声。', + parameters: {}, + }, + { ip: '过之派进色油统办着者成金别离。', parameters: {} }, + { + ip: '大已单是去立称越条布众在年劳几离自级。', + parameters: {}, + }, + { ip: '一声写标资科报给机般问对设。', parameters: {} }, + { ip: '展量界手发维气铁过养联完热空。', parameters: {} }, + { + ip: '石信示起效存专及正半没农极基适型。', + parameters: {}, + }, + { ip: '包段前就交世长等育全实王很局合。', parameters: {} }, + { + ip: '常低完时做青科近她构满圆边已造劳。', + parameters: {}, + }, + { ip: '毛则使院平影受道路始往龙放计。', parameters: {} }, + { + ip: '种当划火声统已革自正术京状毛件油下。', + parameters: {}, + }, + ], + }, + { + name: '李秀英', + rootservice: '和者根断社选又也眼立些织。', + servers: [ + { ip: '标义表拉列题构直委此个型验。', parameters: {} }, + { ip: '都音题次百光眼新系积而第厂集。', parameters: {} }, + { ip: '便系况际状消名设观见确变置切律。', parameters: {} }, + { + ip: '制道亲百则于志斗种用办数山器查眼科。', + parameters: {}, + }, + { ip: '目原百过毛后六空导律中到。', parameters: {} }, + { + ip: '海反外等却四之全小效加国理土地须。', + parameters: {}, + }, + { + ip: '状设加改构技便率五品军过但声细自再。', + parameters: {}, + }, + { + ip: '处样色力山产下战半除界连样少达性。', + parameters: {}, + }, + { ip: '好清万步研亲处三原思量每条名。', parameters: {} }, + { + ip: '我标器口果二省活己常条型她于条风状。', + parameters: {}, + }, + { ip: '到得成六表眼济和活积定红。', parameters: {} }, + { ip: '非系快利品处认速合团那算等主老。', parameters: {} }, + { ip: '在名公即组对真开同办交火影人。', parameters: {} }, + ], + }, + { + name: '杜霞', + rootservice: '养成感第给状标正论是从队是都证。', + servers: [ + { ip: '照斯传体京及状即向儿证形半内深。', parameters: {} }, + { + ip: '年派立任义只别作是叫只府备持区百收。', + parameters: {}, + }, + { ip: '表知加物天场经民空权放如周。', parameters: {} }, + { + ip: '低矿矿口难料件安构电查直务自流构。', + parameters: {}, + }, + { ip: '达素酸八据非格即米各建指。', parameters: {} }, + { ip: '风什该候制例手教置示把必类斗空。', parameters: {} }, + { ip: '我对强本身行白济收须看角志。', parameters: {} }, + { ip: '平存维采称该求直天点百积知角论。', parameters: {} }, + { ip: '支于高产共况个备化过半织题员至。', parameters: {} }, + { ip: '却本酸技形动长生半便华深作次儿。', parameters: {} }, + { ip: '习音深许边风办证育到平白百族。', parameters: {} }, + { + ip: '圆商构必压回口机声量切支证候了都。', + parameters: {}, + }, + { ip: '查此对始其观无步非同行华有生。', parameters: {} }, + ], + }, + { + name: '曾平', + rootservice: '龙维油界易主体则通张铁千问斗口型低计。', + servers: [ + { + ip: '又五见月造速期走构做都很传能积几领。', + parameters: {}, + }, + { ip: '受见教们门它名地离体根多。', parameters: {} }, + { ip: '写济最领单运江土石金点教向但元。', parameters: {} }, + { + ip: '走各国等酸造连分名斯头个资好政真局。', + parameters: {}, + }, + { + ip: '种人车结方然持义适法认物育处通风。', + parameters: {}, + }, + { ip: '百志算查气通易她这率四话先中清。', parameters: {} }, + { ip: '同相处亲个存南员头利千受委前式。', parameters: {} }, + { ip: '实中少山位行子构际术制火派再。', parameters: {} }, + { ip: '引干全加段养图真用两放难话。', parameters: {} }, + { + ip: '必京程了少圆离达又较历快五求形干很电。', + parameters: {}, + }, + { + ip: '花准我发但度类别标命能调却根性许。', + parameters: {}, + }, + { + ip: '明对查口全区省及须海维角住向斗基。', + parameters: {}, + }, + { ip: '求书机知响反何便白明于不事。', parameters: {} }, + ], + }, + { + name: '韩洋', + rootservice: '六带划相几直经年整命拉六前引儿近海。', + servers: [ + { ip: '间织列日段细报般北离圆知方要。', parameters: {} }, + { ip: '么可品了越志发路正公族引干难。', parameters: {} }, + { + ip: '头业西力明由张再只技世正下提正何体。', + parameters: {}, + }, + { ip: '两文头边府过构实边属装交受通少。', parameters: {} }, + { + ip: '器处老世省代离适分史示历统算完场该林。', + parameters: {}, + }, + { ip: '体线备音段装眼不论方连战百代。', parameters: {} }, + { ip: '太约律她报再今会声素界期入看法。', parameters: {} }, + { ip: '目指持图月千得长众热利色。', parameters: {} }, + { + ip: '团色及利其样科究便代地回产所重边是。', + parameters: {}, + }, + { + ip: '复情件成眼花光共识海验题争劳空此越包。', + parameters: {}, + }, + { + ip: '命量人条目引由完研展正制议再具子际专。', + parameters: {}, + }, + { + ip: '门般第务展种毛根龙府没特广等年约认权。', + parameters: {}, + }, + { + ip: '第快长候手走学动林基最拉局果东装称。', + parameters: {}, + }, + ], + }, + { + name: '雷刚', + rootservice: '参度选通数断来运你重地它于变厂基少。', + servers: [ + { + ip: '者解总手广在我军常口县什运断铁方。', + parameters: {}, + }, + { ip: '提通无开时元数北矿世观正解。', parameters: {} }, + { + ip: '地重学领却般看越她证器始都列这建。', + parameters: {}, + }, + { + ip: '候好眼何原采许标始当发话民则个条文。', + parameters: {}, + }, + { + ip: '形行切七王用给南四月权可府数点领。', + parameters: {}, + }, + { ip: '月就二以实已情示且角能每就小没。', parameters: {} }, + { ip: '争数事矿千直加气道离电四所北东。', parameters: {} }, + { + ip: '位常复历上率验斗本山或统济许线成。', + parameters: {}, + }, + { ip: '类完京受共设完加还基容前交。', parameters: {} }, + { + ip: '始七样线性程省之等布效日商气特会。', + parameters: {}, + }, + { + ip: '前决持引连放通斯思阶断务布品要工技。', + parameters: {}, + }, + { ip: '难毛验斯为那生起积北所产间。', parameters: {} }, + { ip: '支示它就京不整太两往级过验般角。', parameters: {} }, + ], + }, + ], + }, + obproxy: { + component: '通代等作米度格众群导地识切物维回育上。', + version: '处量议细研能当及军机增交是所方。', + package_hash: '青只音美观根专军争直值素新保物好。', + release: '几满深斗气进主关直该平南除经水青指争。', + cluster_name: '如间导接类率国相国我什资可明太来。', + home_path: '史机物美持三全复查活出接新置声。', + prometheus_listen_port: 92, + listen_port: 61, + parameters: [ + { + key: 136, + value: '般业新半值别代进门科眼识往而受。', + adaptive: true, + }, + { + key: 137, + value: '天比标今酸回前连里今回成准。', + adaptive: false, + }, + { + key: 138, + value: '交油即火级议容本半区完高可酸生龙。', + adaptive: true, + }, + { + key: 139, + value: '往组她美质小但小内革总很本。', + adaptive: false, + }, + ], + servers: [ + '二认了向青广到但出少命问意去六上接子。', + '我海量质通斗始效无界则写内程话即并长。', + '保求色离红入文得员量置放民型划。', + '值术流也日如她派目外可进区量。', + ], + }, + ocpexpress: { + component: '备调教因通百五由论局程术。', + version: '边二名这东感区何眼该特强。', + package_hash: '每维动用改按气写金美平正外音因。', + release: '领着共去受非级物这金便采必备亲高。', + home_path: '理么头响那分队眼查斗下七周义什值五。', + server_port: 76, + parameters: [ + { + key: 140, + value: '所布上济易还采往亲装入原场律此造院。', + adaptive: true, + }, + { + key: 141, + value: '几空先感求二口电半合场离想身马必年。', + adaptive: true, + }, + { + key: 142, + value: '流小发积信党规且都安两米米维事又部。', + adaptive: true, + }, + { + key: 143, + value: '样业收高情一看就料头分战面例太。', + adaptive: false, + }, + { + key: 144, + value: '型消但般长织命调风除别易选会起保。', + adaptive: false, + }, + { + key: 145, + value: '基来团集马音热准种第很验家条切。', + adaptive: true, + }, + { + key: 146, + value: '叫每则中相示切四集圆眼重华花道。', + adaptive: true, + }, + { + key: 147, + value: '部界主然及王效路音观号质厂给就。', + adaptive: false, + }, + { + key: 148, + value: '变三清照强况合会年眼联制带适国。', + adaptive: false, + }, + { + key: 149, + value: '称子根论深及当太入然能日。', + adaptive: true, + }, + { + key: 150, + value: '人报求铁说位干知法在再进。', + adaptive: true, + }, + { + key: 151, + value: '精作实名写效听万步去近收眼万。', + adaptive: false, + }, + { + key: 152, + value: '但规数识或气数决拉听验对见收由为线省。', + adaptive: true, + }, + { + key: 153, + value: '段学场会好品所期证农何层色两造决。', + adaptive: false, + }, + { + key: 154, + value: '天见二二车细象进用利军列把斯。', + adaptive: false, + }, + ], + servers: [ + '路统石没转度心角书类质需车史调记决。', + '车热要资第性拉物中保维结效听层。', + '质真引白北这劳采成直属少实再者计断经。', + '传斗又发可离水只族管必头。', + '农第省反东白形须消名生家断广据立。', + '周五斯公县办委从选习作少回义七生精展。', + '这消研正史调气老口积省性状组制题。', + '也在保却就用什百结并马化部么组论引。', + '石进己会装据流一求引心展他权角达现积。', + '江出花族车系青天花已米交。', + '加连事意力级历放级织马切速油位。', + '增打委东级叫老厂江增住起构复。', + '行之事自将料是集克领新同元团也。', + '电包对写办毛连广收管民导头。', + '速时导府或去按况但识能力书安。', + '难展门查部前也理她因入铁热程为压议。', + '制商金管分不北化来观报物活型一节。', + ], + }, + obagent: { + component: '外克听精济到平受度素性安决派马采。', + version: '备活手时办离大为段几场林要。', + package_hash: '取较下革增一求段政两原出米。', + release: '面特深强装研办省速张标飞权元状来声清。', + home_path: '通区志不通划叫路种产农部红先相。', + server_port: 78, + pprof_port: 69, + parameters: [ + { + key: 155, + value: '龙发百取此资因型代影特增。', + adaptive: false, + }, + { + key: 156, + value: '酸运交石段根第多等社做说则思。', + adaptive: true, + }, + { + key: 157, + value: '照住非始族出精带我两去格共。', + adaptive: true, + }, + { + key: 158, + value: '层江列地科引而则社其族场同向中样展。', + adaptive: false, + }, + { + key: 159, + value: '照方带再能果准参活具选元。', + adaptive: false, + }, + { + key: 160, + value: '他何后内细油人制度政表且中将。', + adaptive: true, + }, + { + key: 161, + value: '比万联数去内条式书确亲件劳于。', + adaptive: false, + }, + { + key: 162, + value: '京此维也地红研包今求老开五需。', + adaptive: true, + }, + ], + servers: [ + '起便观整社打义何连西影有个须果律。', + '车工参于压国例设图装构越。', + '研况广料之力难些元我图比进好天理设。', + '看实高条科品规今斗要思传再。', + ], + }, + obclient: { + component: '装其他志容率步感政性教参深回明期那色。', + version: '取立际资级命受联具率经主却于需。', + release: '什就规照极美量素光四都再天每四问资。', + parameters: [ + { + key: 163, + value: '反等身克局动参市五日水又张。', + adaptive: true, + }, + { + key: 164, + value: '度给亲万准不须家实热县到易。', + adaptive: true, + }, + { + key: 165, + value: '干口候将称七路段气至例社几己。', + adaptive: true, + }, + { + key: 166, + value: '道员专子和用平集万发与音。', + adaptive: true, + }, + { + key: 167, + value: '思率此西争效主数装京他管式。', + adaptive: true, + }, + { + key: 168, + value: '手由你速要己入老金解识提党族上界加意。', + adaptive: true, + }, + { + key: 169, + value: '持情门反且党看干求些发资代存。', + adaptive: false, + }, + { + key: 170, + value: '变多党后火这向经产花步十市。', + adaptive: false, + }, + { + key: 171, + value: '式当切务七名动使了石斯已较张话。', + adaptive: false, + }, + { + key: 172, + value: '是四质局状八己带只用值价政知。', + adaptive: true, + }, + { + key: 173, + value: '具上起那干张林时标新些格张深金且。', + adaptive: false, + }, + { + key: 174, + value: '所还华别次要金由质提或出决西。', + adaptive: false, + }, + { + key: 175, + value: '报基权角劳口小小起空现见政列员量将。', + adaptive: false, + }, + ], + home_path: '但子科它成市易国设证金商听选。', + servers: [ + '有都无得后至气织华级在节放。', + '力平结青人土式全认团度平九切边。', + '市西质平设斗所那表便见花习。', + '以置圆门意走复要石统教办委也日南内。', + '那叫因党北领意适决者原头种识员入用。', + '写党上安通算见自向整四院天查给西情口。', + '却亲人员石道程置思新八许后论一然力。', + '我老统节查统基流教业步压正传到验物结。', + '技长清全务气养已起织技备是。', + '识阶保世区成严么第路起厂和生。', + '安原需天装维听局战日意代较。', + '江号科小风要话保志民准机受农教。', + ], + }, + }, + home_path: '正华号易千管看型月了合山装构受作出。', + }, + }, + msg: '热则中实变切约复叫金少办计。', + success: false, + }); + }, +}; diff --git a/web/mock/queryDeploymentInfo.mock.ts b/web/mock/queryDeploymentInfo.mock.ts new file mode 100644 index 0000000..acbeefa --- /dev/null +++ b/web/mock/queryDeploymentInfo.mock.ts @@ -0,0 +1,13 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments': (req: Request, res: Response) => { + res.status(200).send({ + code: 83, + data: { total: 64, items: [{ name: '江芳', status: 'error' }] }, + msg: '商生间提国器接他很即阶这一除号计。', + success: true, + }); + }, +}; diff --git a/web/mock/queryDeploymentInfoByTaskStatusType.mock.ts b/web/mock/queryDeploymentInfoByTaskStatusType.mock.ts new file mode 100644 index 0000000..b1e56e5 --- /dev/null +++ b/web/mock/queryDeploymentInfoByTaskStatusType.mock.ts @@ -0,0 +1,29 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments': (req: Request, res: Response) => { + res.status(200).send({ + code: 66, + data: { + total: 64, + items: [ + { name: '姚霞', status: 'processing' }, + { name: '杨静', status: 'error' }, + { name: '孔艳', status: 'error' }, + { name: '史明', status: 'error' }, + { name: '蔡杰', status: 'error' }, + { name: '毛静', status: 'default' }, + { name: '崔强', status: 'processing' }, + { name: '董超', status: 'default' }, + { name: '魏霞', status: 'error' }, + { name: '冯洋', status: 'error' }, + { name: '易强', status: 'processing' }, + { name: '曹磊', status: 'warning' }, + ], + }, + msg: '想题准算斗子越该话代质部书太影。', + success: true, + }); + }, +}; diff --git a/web/mock/queryDeploymentReport.mock.ts b/web/mock/queryDeploymentReport.mock.ts new file mode 100644 index 0000000..b1b15c2 --- /dev/null +++ b/web/mock/queryDeploymentReport.mock.ts @@ -0,0 +1,77 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/report': (req: Request, res: Response) => { + res.status(200).send({ + code: 75, + data: { + total: 64, + items: [ + { + name: '钱霞', + type: 176, + version: '有证派正都为理南多流手党道学的器。', + servers: [ + '得例界管区平节及根件青委号定。', + '生中国极心界通提道备算连。', + '论参传关消验反主好派层备区志局即。', + '风据热界就南情委置能口龙。', + '应来料开离及入好南维族并军效容京光。', + '物入素实用容手在圆公料地放斗。', + '儿条这管性增量步此接数红制难道上气。', + '效组温由斯点根反自做可响。', + '况造利专边由应再导民海马油此决叫转队。', + '感值将素织广气满真感族为王。', + '往选族感王计将温公题本热联经况。', + '连手内山去第安名新看论外总民对基。', + ], + status: 'error', + }, + { + name: '姜超', + type: 177, + version: '称用后展身走单金改委素在往国适越。', + servers: [ + '见四建称段则些几候种千种几。', + '素间解院第织图任但被活行导或量适。', + '马好出空少民平两示后两导志料果内。', + '置正证难际即个使至身者机统厂些。', + '等上决技它万张打当积住满很只素。', + '在便七积下思感广石由思布影。', + '管管际生共思支物带团取称社界代常据只。', + '深区较史理识米指包问写国统建很。', + '料决而与近常般表照系油应周。', + '般强住海片节应北感低中什第边美。', + '海然风个题场手又特型保样必会体转才性。', + '度备点即素装节外前门党什音过须。', + ], + status: 'default', + }, + { + name: '潘磊', + type: 178, + version: '京响行保没口在七织与动就较复院运。', + servers: [ + '叫要究正法统回起能家新使着月江。', + '相标红白进常别须铁级由二求一反。', + '把结组体压思划际度常连起由取近。', + '就所江矿存革法劳治满白类。', + '她位生增是光和活共向制局取常用。', + '且感真因七路与军各及划做求计快展。', + '习里天直强先公义算因所拉什。', + '世品用十条接几规题准目为都。', + '万表求极和集达量加角铁况。', + '法斯间用局日厂实音造厂学带商工关。', + '然响称存指指安解老着度通周西西加。', + '观起声形对型置把政数对理命院名名基。', + ], + status: 'default', + }, + ], + }, + msg: '运石听集大周去里新说军太合。', + success: true, + }); + }, +}; diff --git a/web/mock/queryInstallLog.mock.ts b/web/mock/queryInstallLog.mock.ts new file mode 100644 index 0000000..0cfc5c8 --- /dev/null +++ b/web/mock/queryInstallLog.mock.ts @@ -0,0 +1,16 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/install/log': ( + req: Request, + res: Response, + ) => { + res.status(200).send({ + code: 66, + data: { log: '响严管号同都战养志各再很己除具。', offset: 100 }, + msg: '我理回是科定积王育构引知例面非长老。', + success: true, + }); + }, +}; diff --git a/web/mock/queryInstallStatus.mock.ts b/web/mock/queryInstallStatus.mock.ts new file mode 100644 index 0000000..abe51f3 --- /dev/null +++ b/web/mock/queryInstallStatus.mock.ts @@ -0,0 +1,20 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'GET /api/v1/deployments/:name/install': (req: Request, res: Response) => { + res.status(200).send({ + code: 88, + data: { + total: 86, + finished: 91, + current: '标同有决心处听情式从温志切百石干。', + status: 'success', + msg: '起区院难门厂来书治结外说别过记质。', + info: [], + }, + msg: '也通统矿己厂这何志物特象子。', + success: true, + }); + }, +}; diff --git a/web/mock/recover.mock.ts b/web/mock/recover.mock.ts new file mode 100644 index 0000000..4f6d968 --- /dev/null +++ b/web/mock/recover.mock.ts @@ -0,0 +1,47 @@ +// @ts-ignore +import { Request, Response } from 'express'; + +export default { + 'POST /api/v1/deployments/:name/recover': (req: Request, res: Response) => { + res.status(200).send({ + code: 62, + data: { + total: 98, + items: [ + { + name: '邹强', + old_value: '那号道统作象历极选于采开指织。', + new_value: '算天际次酸情代格原经产也那九参当复。', + }, + { + name: '萧磊', + old_value: '回全程社该须军线属化素全场议水。', + new_value: '容斯九争院变眼知和族现则每空被。', + }, + { + name: '叶平', + old_value: '还天又体车当好细下儿太生反真西几。', + new_value: '两第得除容学者以比数运程干两山。', + }, + { + name: '余超', + old_value: '今基眼才之深部们较意号备术高真院。', + new_value: '极当类战而科科则六马从得越具步。', + }, + { + name: '沈芳', + old_value: '属思正非先信识流性真为消证没。', + new_value: '且那细事极包事解角接目那记京。', + }, + { + name: '田平', + old_value: '争严进确维打系拉记去矿儿社动事西阶。', + new_value: '人适去商有素内应今北电内山导状动华领。', + }, + ], + }, + msg: '导习片住又管性很观克目从这。', + success: false, + }); + }, +}; diff --git a/web/package.json b/web/package.json new file mode 100644 index 0000000..2daecee --- /dev/null +++ b/web/package.json @@ -0,0 +1,53 @@ +{ + "name": "ob-deploy-web", + "private": true, + "scripts": { + "build": "umi build", + "dev": "cross-env MOCK=none umi dev", + "postinstall": "umi generate tmp", + "openapi": "umi openapi", + "prettier": "prettier --write '**/*.{js,jsx,tsx,ts,less,md,json}'", + "start": "umi dev", + "test": "umi-test", + "test:coverage": "umi-test --coverage" + }, + "lint-staged": { + "*.{js,jsx,less,md,json}": [ + "prettier --write" + ], + "*.ts?(x)": [ + "prettier --parser=typescript --write" + ] + }, + "dependencies": { + "@ant-design/icons": "^4.8.0", + "@ant-design/pro-components": "^2.3.34", + "@ant-design/pro-layout": "^6.5.0", + "@types/video.js": "^7.3.50", + "@umijs/plugin-openapi": "^1.3.3", + "antd": "5.0.7", + "copy-to-clipboard": "^3.3.3", + "cross-env": "^7.0.3", + "lottie-web": "^5.10.0", + "moment": "^2.29.4", + "number-precision": "^1.6.0", + "randexp": "^0.5.3", + "react": "17.x", + "react-dom": "17.x", + "umi": "^3.5.35", + "video.js": "^7.20.3" + }, + "devDependencies": { + "@types/react": "^17.0.0", + "@types/react-dom": "^17.0.0", + "@umijs/preset-react": "1.x", + "@umijs/test": "^3.5.35", + "lint-staged": "^10.0.7", + "prettier": "^2.2.0", + "typescript": "^4.1.2", + "yorkie": "^2.0.0" + }, + "gitHooks": { + "pre-commit": "lint-staged" + } +} diff --git a/web/public/assets/computer/data.json b/web/public/assets/computer/data.json new file mode 100644 index 0000000..c9eaacb --- /dev/null +++ b/web/public/assets/computer/data.json @@ -0,0 +1,2123 @@ +{ + "v": "5.9.6", + "fr": 30, + "ip": 0, + "op": 80, + "w": 250, + "h": 200, + "nm": "seq_0", + "ddd": 0, + "assets": [ + { + "id": "imgSeq_0", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_1", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_2", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_3", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_4", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_5", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_6", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_7", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_8", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_9", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_10", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_11", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_12", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_13", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_14", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_15", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_16", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_17", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_18", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_19", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_20", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_21", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_22", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_23", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_24", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_25", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_26", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_27", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_28", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_29", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_30", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_31", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_32", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_33", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_34", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_35", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_36", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_37", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_38", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_39", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_40", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_41", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_42", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_43", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_44", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_45", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_46", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_47", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_48", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_49", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_50", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_51", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_52", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_53", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_54", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_55", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_56", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_57", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_58", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_59", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_60", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_61", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_62", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_63", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_64", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_65", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_66", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_67", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_68", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_69", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_70", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_71", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_72", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_73", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_74", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_75", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_76", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_77", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_78", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_79", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "sequence_0", + "layers": [ + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_0", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 0, + "st": 0, + "op": 1, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_1", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 1, + "st": 1, + "op": 2, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_2", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 2, + "st": 2, + "op": 3, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_3", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 3, + "st": 3, + "op": 4, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_4", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 4, + "st": 4, + "op": 5, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_5", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 5, + "st": 5, + "op": 6, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_6", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 6, + "st": 6, + "op": 7, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_7", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 7, + "st": 7, + "op": 8, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_8", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 8, + "st": 8, + "op": 9, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_9", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 9, + "st": 9, + "op": 10, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_10", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 10, + "st": 10, + "op": 11, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_11", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 11, + "st": 11, + "op": 12, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_12", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 12, + "st": 12, + "op": 13, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_13", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 13, + "st": 13, + "op": 14, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_14", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 14, + "st": 14, + "op": 15, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_15", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 15, + "st": 15, + "op": 16, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_16", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 16, + "st": 16, + "op": 17, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_17", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 17, + "st": 17, + "op": 18, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_18", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 18, + "st": 18, + "op": 19, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_19", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 19, + "st": 19, + "op": 20, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_20", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 20, + "st": 20, + "op": 21, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_21", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 21, + "st": 21, + "op": 22, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_22", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 22, + "st": 22, + "op": 23, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_23", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 23, + "st": 23, + "op": 24, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_24", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 24, + "st": 24, + "op": 25, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_25", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 25, + "st": 25, + "op": 26, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_26", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 26, + "st": 26, + "op": 27, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_27", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 27, + "st": 27, + "op": 28, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_28", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 28, + "st": 28, + "op": 29, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_29", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 29, + "st": 29, + "op": 30, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_30", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 30, + "st": 30, + "op": 31, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_31", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 31, + "st": 31, + "op": 32, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_32", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 32, + "st": 32, + "op": 33, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_33", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 33, + "st": 33, + "op": 34, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_34", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 34, + "st": 34, + "op": 35, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_35", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 35, + "st": 35, + "op": 36, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_36", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 36, + "st": 36, + "op": 37, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_37", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 37, + "st": 37, + "op": 38, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_38", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 38, + "st": 38, + "op": 39, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_39", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 39, + "st": 39, + "op": 40, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_40", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 40, + "st": 40, + "op": 41, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_41", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 41, + "st": 41, + "op": 42, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_42", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 42, + "st": 42, + "op": 43, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_43", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 43, + "st": 43, + "op": 44, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_44", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 44, + "st": 44, + "op": 45, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_45", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 45, + "st": 45, + "op": 46, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_46", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 46, + "st": 46, + "op": 47, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_47", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 47, + "st": 47, + "op": 48, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_48", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 48, + "st": 48, + "op": 49, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_49", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 49, + "st": 49, + "op": 50, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_50", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 50, + "st": 50, + "op": 51, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_51", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 51, + "st": 51, + "op": 52, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_52", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 52, + "st": 52, + "op": 53, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_53", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 53, + "st": 53, + "op": 54, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_54", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 54, + "st": 54, + "op": 55, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_55", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 55, + "st": 55, + "op": 56, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_56", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 56, + "st": 56, + "op": 57, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_57", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 57, + "st": 57, + "op": 58, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_58", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 58, + "st": 58, + "op": 59, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_59", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 59, + "st": 59, + "op": 60, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_60", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 60, + "st": 60, + "op": 61, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_61", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 61, + "st": 61, + "op": 62, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_62", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 62, + "st": 62, + "op": 63, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_63", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 63, + "st": 63, + "op": 64, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_64", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 64, + "st": 64, + "op": 65, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_65", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 65, + "st": 65, + "op": 66, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_66", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 66, + "st": 66, + "op": 67, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_67", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 67, + "st": 67, + "op": 68, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_68", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 68, + "st": 68, + "op": 69, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_69", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 69, + "st": 69, + "op": 70, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_70", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 70, + "st": 70, + "op": 71, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_71", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 71, + "st": 71, + "op": 72, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_72", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 72, + "st": 72, + "op": 73, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_73", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 73, + "st": 73, + "op": 74, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_74", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 74, + "st": 74, + "op": 75, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_75", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 75, + "st": 75, + "op": 76, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_76", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 76, + "st": 76, + "op": 77, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_77", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 77, + "st": 77, + "op": 78, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_78", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 78, + "st": 78, + "op": 79, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_79", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 79, + "st": 79, + "op": 81, + "sr": 1, + "bm": 0 + } + ] + } + ], + "layers": [ + { + "ddd": 0, + "ind": 1, + "ty": 0, + "nm": "seq_0_[0-79].png", + "cl": "png", + "refId": "sequence_0", + "sr": 1, + "ks": { + "o": { "a": 0, "k": 100, "ix": 11 }, + "r": { "a": 0, "k": 0, "ix": 10 }, + "p": { "a": 0, "k": [125, 100, 0], "ix": 2, "l": 2 }, + "a": { "a": 0, "k": [125, 100, 0], "ix": 1, "l": 2 }, + "s": { "a": 0, "k": [100, 100, 100], "ix": 6, "l": 2 } + }, + "ao": 0, + "w": 250, + "h": 200, + "ip": 0, + "op": 80, + "st": 0, + "bm": 0 + } + ], + "markers": [] +} diff --git a/web/public/assets/database/data.json b/web/public/assets/database/data.json new file mode 100644 index 0000000..40c542a --- /dev/null +++ b/web/public/assets/database/data.json @@ -0,0 +1,2123 @@ +{ + "v": "5.9.6", + "fr": 30, + "ip": 0, + "op": 80, + "w": 250, + "h": 200, + "nm": "seq_0", + "ddd": 0, + "assets": [ + { + "id": "imgSeq_0", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_1", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_2", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_3", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_4", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_5", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_6", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_7", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_8", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_9", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_10", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_11", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_12", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_13", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_14", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_15", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_16", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_17", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_18", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_19", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_20", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_21", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_22", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_23", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_24", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_25", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_26", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_27", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_28", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_29", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_30", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_31", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_32", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_33", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_34", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_35", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_36", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_37", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_38", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_39", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_40", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_41", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_42", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_43", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_44", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_45", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_46", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_47", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_48", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_49", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_50", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_51", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_52", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_53", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_54", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_55", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_56", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_57", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_58", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_59", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_60", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_61", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_62", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_63", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_64", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_65", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_66", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_67", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_68", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_69", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_70", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_71", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_72", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_73", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_74", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_75", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_76", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_77", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_78", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_79", + "w": 250, + "h": 200, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "sequence_0", + "layers": [ + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_0", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 0, + "st": 0, + "op": 1, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_1", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 1, + "st": 1, + "op": 2, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_2", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 2, + "st": 2, + "op": 3, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_3", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 3, + "st": 3, + "op": 4, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_4", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 4, + "st": 4, + "op": 5, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_5", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 5, + "st": 5, + "op": 6, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_6", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 6, + "st": 6, + "op": 7, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_7", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 7, + "st": 7, + "op": 8, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_8", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 8, + "st": 8, + "op": 9, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_9", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 9, + "st": 9, + "op": 10, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_10", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 10, + "st": 10, + "op": 11, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_11", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 11, + "st": 11, + "op": 12, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_12", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 12, + "st": 12, + "op": 13, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_13", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 13, + "st": 13, + "op": 14, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_14", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 14, + "st": 14, + "op": 15, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_15", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 15, + "st": 15, + "op": 16, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_16", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 16, + "st": 16, + "op": 17, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_17", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 17, + "st": 17, + "op": 18, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_18", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 18, + "st": 18, + "op": 19, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_19", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 19, + "st": 19, + "op": 20, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_20", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 20, + "st": 20, + "op": 21, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_21", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 21, + "st": 21, + "op": 22, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_22", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 22, + "st": 22, + "op": 23, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_23", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 23, + "st": 23, + "op": 24, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_24", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 24, + "st": 24, + "op": 25, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_25", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 25, + "st": 25, + "op": 26, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_26", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 26, + "st": 26, + "op": 27, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_27", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 27, + "st": 27, + "op": 28, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_28", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 28, + "st": 28, + "op": 29, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_29", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 29, + "st": 29, + "op": 30, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_30", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 30, + "st": 30, + "op": 31, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_31", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 31, + "st": 31, + "op": 32, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_32", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 32, + "st": 32, + "op": 33, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_33", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 33, + "st": 33, + "op": 34, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_34", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 34, + "st": 34, + "op": 35, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_35", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 35, + "st": 35, + "op": 36, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_36", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 36, + "st": 36, + "op": 37, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_37", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 37, + "st": 37, + "op": 38, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_38", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 38, + "st": 38, + "op": 39, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_39", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 39, + "st": 39, + "op": 40, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_40", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 40, + "st": 40, + "op": 41, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_41", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 41, + "st": 41, + "op": 42, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_42", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 42, + "st": 42, + "op": 43, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_43", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 43, + "st": 43, + "op": 44, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_44", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 44, + "st": 44, + "op": 45, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_45", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 45, + "st": 45, + "op": 46, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_46", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 46, + "st": 46, + "op": 47, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_47", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 47, + "st": 47, + "op": 48, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_48", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 48, + "st": 48, + "op": 49, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_49", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 49, + "st": 49, + "op": 50, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_50", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 50, + "st": 50, + "op": 51, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_51", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 51, + "st": 51, + "op": 52, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_52", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 52, + "st": 52, + "op": 53, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_53", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 53, + "st": 53, + "op": 54, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_54", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 54, + "st": 54, + "op": 55, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_55", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 55, + "st": 55, + "op": 56, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_56", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 56, + "st": 56, + "op": 57, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_57", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 57, + "st": 57, + "op": 58, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_58", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 58, + "st": 58, + "op": 59, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_59", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 59, + "st": 59, + "op": 60, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_60", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 60, + "st": 60, + "op": 61, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_61", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 61, + "st": 61, + "op": 62, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_62", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 62, + "st": 62, + "op": 63, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_63", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 63, + "st": 63, + "op": 64, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_64", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 64, + "st": 64, + "op": 65, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_65", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 65, + "st": 65, + "op": 66, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_66", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 66, + "st": 66, + "op": 67, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_67", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 67, + "st": 67, + "op": 68, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_68", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 68, + "st": 68, + "op": 69, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_69", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 69, + "st": 69, + "op": 70, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_70", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 70, + "st": 70, + "op": 71, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_71", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 71, + "st": 71, + "op": 72, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_72", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 72, + "st": 72, + "op": 73, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_73", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 73, + "st": 73, + "op": 74, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_74", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 74, + "st": 74, + "op": 75, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_75", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 75, + "st": 75, + "op": 76, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_76", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 76, + "st": 76, + "op": 77, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_77", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 77, + "st": 77, + "op": 78, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_78", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 78, + "st": 78, + "op": 79, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_79", + "ks": { + "p": { "a": 0, "k": [0, 0] }, + "a": { "a": 0, "k": [0, 0] }, + "s": { "a": 0, "k": [100, 100] }, + "r": { "a": 0, "k": [0] }, + "o": { "a": 0, "k": [100] } + }, + "ip": 79, + "st": 79, + "op": 81, + "sr": 1, + "bm": 0 + } + ] + } + ], + "layers": [ + { + "ddd": 0, + "ind": 1, + "ty": 0, + "nm": "seq_0_[0-79].png", + "cl": "png", + "refId": "sequence_0", + "sr": 1, + "ks": { + "o": { "a": 0, "k": 100, "ix": 11 }, + "r": { "a": 0, "k": 0, "ix": 10 }, + "p": { "a": 0, "k": [125, 100, 0], "ix": 2, "l": 2 }, + "a": { "a": 0, "k": [125, 100, 0], "ix": 1, "l": 2 }, + "s": { "a": 0, "k": [100, 100, 100], "ix": 6, "l": 2 } + }, + "ao": 0, + "w": 250, + "h": 200, + "ip": 0, + "op": 80, + "st": 0, + "bm": 0 + } + ], + "markers": [] +} diff --git a/web/public/assets/empty.png b/web/public/assets/empty.png new file mode 100644 index 0000000000000000000000000000000000000000..b8e666402623bfffefa17c50ff2e068e2f8b941c GIT binary patch literal 14913 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

)SNCXUFU~Q6u(jIY$+Y^r>>N2>ANZ1o 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 literal 0 HcmV?d00001 diff --git a/web/public/assets/failed.png b/web/public/assets/failed.png new file mode 100644 index 0000000000000000000000000000000000000000..0ba7db0952ec7fa71158ae0a287d5e8f2bd37b89 GIT binary patch literal 55696 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 literal 0 HcmV?d00001 diff --git a/web/public/assets/logo.png b/web/public/assets/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..f4436f966e5e78a949d533cde4f201ab80950672 GIT binary patch literal 3552 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 literal 0 HcmV?d00001 diff --git a/web/public/assets/oceanbase.png b/web/public/assets/oceanbase.png new file mode 100644 index 0000000000000000000000000000000000000000..b79b525f6b664678fe22a176d979daa8b73d599d GIT binary patch literal 13258 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

3hd%T1Roou7$o{A-RvVk6Um} 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 literal 0 HcmV?d00001 diff --git a/web/public/assets/welcome/cover.jpg b/web/public/assets/welcome/cover.jpg new file mode 100644 index 0000000000000000000000000000000000000000..5f24e1f8fbd7d4190374d1e6331bc9e2d829e740 GIT binary patch literal 236659 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

Q6JrUI*3v;qxN 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 literal 0 HcmV?d00001 diff --git a/web/public/assets/welcome/data.mp4 b/web/public/assets/welcome/data.mp4 new file mode 100644 index 0000000000000000000000000000000000000000..d4b05c01c7ebba0e787a166012d9c795486129e8 GIT binary patch literal 1162582 zcmYg%1AHA%)aZ?^25o%fG`4NqXmVrQw(X>`Z95GbTaDS+_DlcY_ulXQF80isgE`nU zvpc&0003y>y9(DcwBhivVvkd%0a(>~)#Kr=o1sd8q7y}tunSc&V9IT8$0}vAn2PfxehXmax z0lln(s5mVPP*6?ylhVl8@UuhM*3R9^*u)9Q#K6Ez%f!IQ{)sema+0fZKn9~2#LTBb=ZS_fGYv*KcYvafTGz1xd40#!W4#paG;^@E2I+mWG6J0(jIFH99X}xsHx461r%%Ap z-ujcl$QWegVQXW|%fv{-$Otq6IXdatIa-?A{iFCF0rqx!wk9Ty#!jE)PG%0D434}k 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⪚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

3Z_@{4nL@X7U(pxiSqeCb>wq=o>092hY6gdm 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

L!~Y7{hMa*BRR|Di4jvk1Gx$2Y+T>L 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 z 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�R)iK8on;B^rXo$QiwOy=9=nfIO34y$L1*zVMZ-Y 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

L#z!}} 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 z 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

&|H~HPGU2<)A^?n;t6;O8eK-gesN+GCS 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 z z2C0XUT>)GpkqWuH^*sw>cwq1qx|KI@vcB_@ zvtP-fq&pB^Wt}T5QeaXRHGJC2V@>4;E`V=eB1c!L{BYydpUp3WVkp+r9M>(G=q@#; z34Yf$>&zJ`(7PjvN*$O}$4(xgxX!r<9ae?h9#1UfzG5Mt_QE{3O{B9{8d?*Nd!fO> zZ%(Ms^Z46~$~toL#bS71?EzrE06puc5Xlx`flIqF4gRoAhxcC8xO{FZb4t ze{;geKcf15i-(ZysbRFR63L?-ipS$6-%>l|3X@R14M0JUC>R8_YXVN;1H6oAR-iZ< z>lD(XvIF%xE$w6hI5*pD?Q1tAvp$YC@XXd+tk6_>EwFq7pW;{fm8RuGR)jRFnTMpy zS{RTOaZU)q=J(1yl9C%B=9Y%H;5UNPeSe>Av)PM({j#r&%d6V2B0z9VQWkxZEl5z- z!TXrESUTO++62CM8;%=j=SL1)OoX3&dO|Sh1Be`h_=jAMoTyOupTCm6!l*fL%Lp!s zc=NCHm@Cb$!1AcgX>#b`)2zV$=UdSwC}`HQU|;hSFgX7hA<4g^bx?L{6AS{>&daaR zbT1Cs(Og#U2Xa`9P>fGCmX@);wNS0V$kubkTtA4iZ>BG`X1CfZb13_M6ir#0tKDH^ z(|M;#&DE1T@eS$I$cRe44gvXb?t_(;nkP%CZ)Lj#(6O8h18ezOQ#>%uan0B_vCY`B z+Dgya@5$cf zhcG{}^Q9rL;6O<1TRRNKd`VUst!lNafk$19AN$0PnZYvLgt53QN&PH&zX2pA9f9Va zS-x1i@L&V2Xwf9I-e6Zve7w@6@`J0LLTOgt>Q6P79^*&ZH8r?VP=H`rYn@Q-LQA*+ zozYexwGx9fR{SSE{R^N^h;%%ia9n~8BuRj^gd>g6sVCwT7ujw|o<+ z$f!|PZ$qznT(cC3TO#s&y?Jjf%4a@lfgQGNNQkHg4p#SpVv03k^C~LTp~dbM0qv<- z@hg@W8=2Z&RiYFHKU zLHNs-)rd;Opa9usUuC9e{bs=4{;>aWzSK7c&$VEmH?S-c&+d4>7Niwu`s+*XaCRy8 z>KJEr>7n)uX-0wn^AY~yye`S4-S0%0WI3?ikZC{3+?I-1ozS8MoFJ(AMBfw3qw{9R$3+~b&cj^}p8nKrXH zzW1PnXk!!elTF)&6b6NC_=x4}UlGk$2bbb|^;5F|p7OJi-TL>RXlVnWGMnL&m{tay z>-TdHT_r!FEMpPbhfCy+-{cyJrPX5h6kBp|vsBvg@cEiFY}3w(fXMYv-vbeqi_F$? zs1}AXdQqXYn}^*YibfYv_)?|!7Nl8Vr%avDHsISDz?Zm^u}*vxOJXudJa*h+DK?(- z{PSer7fn&Oyszx_PU?SvA9s=102y3FHOts>Hkd{v=zVXLU#rJvQZ0m@*k{=~6{qkA6T7&U=?!*q)~>fj+LZ$4_5&w&!9I!`1d!;*se9OSbg z|E((eekjH*HR`8#-US&s1GBB2>7o?+%j+4CHCB3X^ZI{ut~aFSTI=Y@T{PB!MS{Nc zEbY@Mvc?MCI{3$a6-!<>f;WqM-v?&wG>@ij-x&_u!sBj&q7*%^mAE%!TsOzJoqQ2z zjd}I*c;|wR8Y2>yuISMl)*D|#$P_3S(jJZf#;zOH#%IP_a@HU6*EK!dUG>>UG6iqI z`{8w1-21p&y`RtEckx|Xun)g^Q*c-zR`~Uq5m5zIfF5x!Y-{-yRhof$?qT>0)o4Bd zB*v$-fbTI<4f;{+Qvn~)qRYm(yj)BPO|&`$_mMQbu}dc? zBjI&~&8wfIYZEEN|OSzd3o8YrL^}+|f+F&O}SK)wjI9PEZO!0!IBA zN5mRN^&O*a096}7Fo;QpISicpNeu3ZCSG4iXGg7K7i;eG=2PZ30da9YW$=0?sN|R_ z2ff*CB4|8HuQ-mst^?QWsBoB3jcJ_&x4ew276S)X!9+Mns`u1Y1-zJXhXNHkYcLx7 z3Gea3UH%6Bc9x7fZmtcVVtgY};z!!0&Y7+FPwro(7Rkz-dIT_W@Mq9pq5AT%9-i_o zlmbL)!HlM8bbH*j8~nZ`sxs7Jr^0PT(>xoaGH8JuM?>Dt=EUr;-Sp3t;tBz25m8by3nA>DDm37LN!3`ZJVfR~O!6=|*&d)Y+1Q&I?L|YM@SQqMN>_=JHsNVy@r(d0$ zi!2s#s-r<*5*tG#9+VS){ML>EJhBNQz%E9)jK*uo5ndjVcu`yin4qI@hBGvu-}a3P z&ChHEWkLF889DN4(Q`}ZhGvoRMmYQ8-lWvcJD$U_gQ2(hHCsF0QCj*M1tjd76V3U$ z3_AUa5Rwx$BeI)^ow7s91{W5#OfsW7#l)v$(%R|83{suAT>&dj<4};D(NvKn+z)u4 zf1lYUOkw-knwFMVrE+h*M?RTOP|))GZK_Fm7!jYDMs+P9E0VFv@|lR57;QU%&_x_? zvwPKCfD*WY2-zxUb0`A3G-%ohnq^{A1Sa?#X#aobb6$-$Mz_H;L?(1d@+;Kmj%lH) zyQGC9CFwvK0@r;%W6BBOY%#181&>rHRB=YpO*JWRghkV_(NUvCtfM{}EW0*=!;GBP z@XHL>dY@7e|Ad=&*s*1+wL~hR`n!R)crjIANP1)9djMIaC8{4T{2?WIGBA02mud4! zy;1O#InhP5?%J@W_@G?!;6Y6w7x4|gcm=z@qejS;v-|vnsCami#~>o+;aPGRzohfK zHqz@lHaNJ68XsP~7~XTA19U~eCHe7f<(ax=D)3P^Rywc$83`{PaLA$+E}Ba-Eo+>{ zYY+JXl8uR+JKTu;fO%QuCX-iNm6f|w$kD|)>r4TV`y;x)QR4P7NBhB@V)gI`dCLN?B^VSk$=*60y+|wDS>8N3f44xNZ8(JnP>+H(#DT<6zpY z+WMF!MCz{gCM9!*?KH<_o#^lIuQ8!UNGHXB0YFI=WzlNr6x&6>5D8fx@$Kc!^?wCV zSLchv-~TPYz9`W3Sw~DN+KSn!Ue~9*WubQ@W86Bk>w31Q6$TEVmNr>ntD}be(X62s zY7VMkhW|VoMNg`5p(p>NK@`h%h1{%?X~9Qgp7@WP*K@d`nCR0_^pi{PJ3NO9g0@vMH#8IYX3e>xFaxXd~oaie1X2Pr0Na7tNz zJ&pah-5Uu2+OowSHKqF)T(EKro$vwCA!7CU@D!nG{~T@i1}2Rx@mydNO9R-+-EAy4 zix!hN|71w&;GkiB3&=e?3p_U-EhJ;Y7l9X%>yxV-SFA86;bgvY-8q!!gn~;ltIVw* zI@G#T=c>ab2PWz>1?rhr$P2NVYB~E=dqk z{J;{BYZ$ADyl<~GKT}tP*MJylX8{FwGlq?AI81z+jBS@%hRY7w>QO(-b$yhih}TM2 z#_hjjW?dp-!8OvVTGKgI6#5fsOKaK4_<)Hf&23{SsY!d)@{Z)OG(U9$yeIPNw__I? zGH|63W@MJI=&Y$S$2%{DS^}EuBtMF;%_;IG5XR1^p-ZC{ljcg@kGH@uMd!Mvg7BIT zM5)+{LdR|qt&A6inH=VdWs?Z81Gr9PBCI&~z-4Q!0uch^DAt#>Gk1@D>S&6ao=hQhdCl1ns?C`go)*8VnCvD$g3?n zLySXzlM$GQp{Xrr>!ja4o&H_1_D%*c#pw83lqhR;vNvn2|Cv7nhnpQxF)ACSZvdx| zHgYAI@+Rspe-^T+RM zx}Ywa&2s9Bw2_u1q#@5C?idLl7Nx@p!iO;ln5KJ-@fgQDu>2Nsn}Y<3`c!Z_o!}tH z6w3CM4;EvBxBok<<6Yi!&`hCNnOMz|gEyZQtChf~+kFM;1C-Q@naJpyALnOrw7}I8 zv+se*?JHtYlOiaS9bB;+BP4;IGA!FcMwIOO=$vu5k}jwHN$8<$miX!zfRJ|;CQf^P zooPxX=C#J^eM&qgvk8^KL#D;+V_ta6nUQ64FeR!7aFBoq1ORm)rNdvswa(w(KXHoC zV=syV^CF{`k=N@GEs*)JLh)i`H*Vj))#`+dTD-<%NZwz)k)7<&CtY~db_lJ;+?0RA zAki3FEQFw{KDjz=SXT3ym4baXpP;}R^@lxIl*c=DOrDWv=U5tljVOlhL!+Q=`vFlz zgQbq>)?xo3NEvO6t#AxfiA_edmm<}8Map36=rC+@68(iZcaCjv=N@t+LBVW)e~v<1 zFVrOl0hrhuQuPn>0x^e{a1COok#A~l1VOQI~7 zDQsh$B^dQg3>Ru+`3?ya)V3r_OvrE`(%epCjmFj>xc_vugm$V0es%b!MoydZH-n2+ z=UivjNWb&%K(3K#t*G9Nrs!s`EZ2F#+q8i9pTSUB+Tv|hEemb#9e;B0u;v3glM3N# zO3}7i3~3hohWe$ulgjr>W*i%5co|hYu~|=;U)@$)}c$AqZ|xxJ;a@dW(YZ$Y(9Nh=owT2#O~ur$u%9B?=(v8kBc~7UxB6{M~X1 zft|Z^q>Muq3jp3V^o}QvHIk^@Iw715h9|h;e884ngaNP{3H{V2Kj35fhQ(8mpPwKQ zgeP1y`?sLLe$J;0GlF4e(}uSIMz!}%gs#AXhR7Vs+$@;yX~5B%px!3MbNDpC*Tidf zDB%|_r(0X$PeCD7-11ZP2u{zpjzh2n6NJ8|AMIFXaz&%4{_eQYJe1-Kp%20Xs4Bz7 zH)s97`7<{RQjDx2lvNWgu-uo_t(h0aYz?6D*dgbeIIh`YnL;|K3Ljtg(ImGkhvGQb9J5%|vb5Z)mbu2_AH1*$CEPKvV{m2uT37 zQ6c{<6X2N~xEtDXszD+_pSmC@T+Qm5UerEiNbhak$iCR}+%Y(is|Y^rY@02nmw`LI zc`XGFVa%@}-25!quc|YCa((5q&a<|jV^5Ab>)nEEOzu1*O_F*J%)Ppk{EBvKb`+B1 z$b!I3DE!^|L{-DITFbSFO*vwiMd?$yL>>QJvm%kMc{!-~^Fizl3vQldA2vzaqE7l> z1KM{>&(qzOFrN|>^(3KHl<2k%4i!L@E$V%McF*k9OCb9O)A*2{xLl>9p4L=Z3F6Fw zTt`rIqGY~<<9EH*YOt@>W-$tpBR%9p)A4u?DC1@)}a~pDn!QZ7A-Zb;tP1Fh3@EN{g#2hOgpV7fiIV z)eLWmBo>u^1|Kj=70}62k}umV0BEo1`I?cfcLe?`8y?;ZR}}jPNU6?j4^*sBvCu?m zY$~ppjNC1_s%2v{Ww~%-bqM5YzkW8dYJ9pe0x2roD;s(8opq-;v&>0k_xR>OBdq4d zwf~D$?C5tR?)81Fck$US5@AeeJW#ESvj*JngjWy)`?V7$wTVo&H1!$DOctQ0f|gT& zJTJs);(`+n1snxf>~Yp*H8ize?T&fE7o z^N31^+^E_%FHYorSj-n%Bg<@@UaPU)>*||-?U&11i6?Fmy5xdZcrNBeFLeO1sAkG?64^%Df;>ip;H0(6DA*?nGRFqc|R);VY zE^2GChhJ)%w+XB;H;Qi;8F~j=9H{il;-bWl&Q-Io1o z!kGJ*Xw4!F7Hv}8oF+Qy;dL8T))uASJTo6pQZ=D{hsQk>2(3-?+|B7n z`1ra1NArw}ws*-?$@(OJU+LYno=isFn8628&kI2fsDrOAN~$5d3aoV)GRv)1~ zCXLd*_RQ%|o>o8xtKJh#$s3#MA`n`Ka>UOMW`)9#w=>^~L&^|NAlI(~WZP@n!=RAU zDfYzEM)YtuDGaChpL7+qY=#%R-zLwM<&mDM5dSLM#WPtN>!fWXIwy4d?YC1%w2rCY zuhW9)LxLMFv^T(!bLWR~J)bN8^mE;5iN;!E?#@^Y12*2hZ6;oU#;0m-N~#kmMAw#Y zBB#$SrRZdI+~n;{vEb452_Dr%sf3UUB(y?AbLIpO(|xc zkjls$DSil3kEMo{qTHfm_NuPC6xQu?jAqw6*gOOymu0wK2%sO0tjF0HdGIzwGAEbi z+cE?=UBzz8OR-YD*J>l)n|tdWM?id?=8Y#+Ouzv7OsK%Vd|60#T5bxD(-2%IfJ5I)mHK_f8d(UBog$>YmG&C99 z6Ma^CVu1CG08Zq$Q45vqgVImedMGCD4HZy16gm^7H5MemHi;CzA_{B7v;}s z7W~MuH@{v^1{~69Fz3cuQ9u#P@L#9e){zV#Ig>%^R_<=xtIxR9S)w40Js2YXEOiF2 zR0ilLUG2i-%)9d)Na{*;7`M_Oqm){XpG`5QR6l zrrKQOo@z=`=Tc`+k7WkjhIFTX$Jo2?vH<2Lg(!gyOxc&IEG*_;}26 z5P{kU*|XwGy(4enbnMrhG7vcp^-5Y*oUib+fnF$J0eQ~K$(;BLP-Cbn$7it9ahJ~pTUP#=nbuDOi}Yj150dLAYF}o5sC9Iba@JBz9hX(oG7^n0SD&O7NJmi=-?IT>2)aMguzXasuygC zi1=1^p#*&uNoek&q-|$Eu^OX@q0yi8vLEIo`}jSC5Vm}Lg^T}|!k@}w^Qua(I}WcD;0EQ zqqJC)d_HCtfW~lP{xjc?b?9RT?gFh@O~&RVJ&je&e{|euu&|OHq-tb|GnN(IEN4OG z)PY}L_SuCYdY0e(&daFd!UKMihd_}Qwu_5Dm|3{pbmYWBp}+gv0?-K;)lMn3y}-1{ zPv*?38;?qBnOKHyCys)=Qqba)5CgaK%W?qQR z?r{0(L^~1mENlTc?z-BX@$3vrX`^(?7bTw$yPfn^o1q!58?U|SH(%R|iuos??)15D zK!+LRmd}N(jxoQ+`y)F=gRs{CL4t{TfAfYSt1e0JT|zNST~+_X%&Bw6^VbJ7;2tDW z{B0wV^m1gN;S>$eyJXUL4#u&Jq{J^detj6{1MA@y$La2auRhx*M*1@RFy=58=T_es zaSv!a=!J3U&i+v1=eQ7*>qC7OpPhdsvVpy9As;GQ6>O`f82mk6M;6BG7 zdeWZzm`SkHxpaX?M$EYMRBMk}Y-&%B*4e9|j*k4fUq}_dpr4om?W+-WurpEDWHSB#vnJaY3a2&oS|VsnAyD=nib36C zxt`3EeSa|3_$671T!=9y76PG~#=CO1hy%DPX$%F9{rC|TUM+b`E70eZc>Cl7?Ej%V zvRiSTo1cez7iB`bIFq2-Dn7qSiz%OOviJKS*}E^2wjw?guB;$eVm%R-id3j}%iM$} zPdvXWYLf+9=ucxF#BhByV@Hzk$$>i0SoJOgdarX5__(XnJ#z$U2gG%7@v3j7dUhc- z9q^eX?J=!jiZf=te$lN1yzI|%DUYuKJ@?Q`G-UMOOLps`Ihg;Sllpro7$4j682ndb zBh?$1o-)dF!#2s(lRFe;iIje6uh(?}y#|zbxRinJ@3Q@_B|CneX@DGek)00b!r-A0 z;176xT)z-1%ZA1LvK#)%;9p>rt;m}wMI)|Y*L{#eT;vi?B?PV;10(#0-$gOzdsD1d zwM~mN;c>Z~el}#2`>guTC{W}}@Ht)C=@ofX;^kAbQ@%GRS}XOl^y|v8<}=w>o3s9U z8siNl+p!g~CWE8arsQ>3?nJ0s@ExH>nw6653+^sRbg`8p5}gW{rTd!=`X@4pHoms9 z2J&~?i!U-d_ldJ`4SD{4(_8w$_y{6IWBpuHQ}Q8>NPHsW=(1KSi@uja;>+cTuuwYY z=r!WQ37tKXwD)NC{MEu$L0>0NYYIscAqx;y_szV>2i0?cidmKBG@M`b|6T9Juf&}> z;aK>nGJ{wr&ro9X>j`7FY%089RHB!K1=wcReK03N2HOV3jeZSr%E&{qpCSl_&&umd zvQ#Bv$?zu$orysWkxu9pD9h>2kplW6!~=)Ya|bpm1FH(hM7GexhuO>fqKSDz5B4OX zZgM?k>f}x|0lU90a5~4krXKi5m6)N^si{)RtXMd;d=S%gvaVw3tsFDw*H=O<($^Xp zj1U}K(aqYa8H&EQ-|}paHXPBc#ZYaZlmwyjleX)z2EEtkYMJSTJDnFpz6k^k#lJEk ziu6S-^HvcSP52r*w_fuUnEd=(&>Td74pEECjxqVn$Q%Bgioq-HxK*94!2DpnD2ByI z!t(KR6Izm*7x_cQ=Lyt@J_S`zB2G+FEyBKkd6m2%|-6_Ce3bRYj3 z@pKwCnd$cZWX-CqN|Bh_o|#?}mP?;jEyLGD%cqy#cBLs~#1`SMK;YeF3QWhN?1xT7 z2yKJ00*KTG1Kz#5VgO~#NvQa1CzFFc6~BboUO_qcmK0a9K`+R^Uu`6J(mPZJPy+4@ zqUxtNe-Ho;@?6p>E~ZwbAl6R;vo&tzct4?~`=hF^B;!Md8-M4~bW}zsms5L*jJJ|_ z2ZkY#b>bC>tHrs>rJ&2Z>QDh+dDN@w`VG`pUMvr~7pvj~K7E!G&qt8U%FeRJ=7}Zy z@xs38S(@`papv`{=`XX(6N^lLLdui)zOa-&OOKd%i6)L7etK+GgRH3jZ96zY{v}y8k4^ z*1_Q({9It{2!Xk%apKITEN0NpoN67A!aQN^O`h*BSrDt2^oK?5 zut+z9dV}J#!%fj{3Ls6Z?SdnuP-!u4V7V!0yOxT=V+PYsp2-uG4JV`7tu|NN7}f`dWFOC=&qkU1#$PkbhML%%z1DQa6Zq^sUf&C(UbP$F{dyN^+ZQ z{@y{<;*Hn4>=jcoaf*!8SESDXD1pvQyVR-fNFr`ffs(PxWZTg-c1Z6rlCigDujybU zxvrz**ZsndrMrvf>!2^1d&i7Q8lhxt==8%f-&3c#-V@E)*Jpk zGbcY89CpE^h#`KE61kmzewu7RUZoFRi9O*;MiXZ2&O*@@Um2i{H>y4Zhb3cHnl+!o z!Gt!RUf@1N#c-8PLci+PDyO}g&SfwQKNiv-b7xU{DR?ArNz7Va^@T54?2TS8;Rd7Y z$oyOLrAzA|*DT_?Q1m-Wh^Lh@&??canTxHb9}CU~<$-LJz9hSBb+`(;+`W<>pUZe^R%5>P z1)kW)sbaN_Lq2(4L7cYPIK8*NRG8OCMpG=durq|8w+=89^@7>5m8RAVa&`6aUg42H znG5hyk^AuCS0UeFe|jx`Vq=8ua1nsQEFJ9j5AZtW7c<~~0q{5BheRLdP2#jl?^hfe zBixN}4CoU%sjN5^rke=vNZu&PdmalK9Dnm|tAnk(tH6EE;%LqfY@c?}XrIL^{ zJ902llXg^h;Ppe%90%i$JZPHdNaDAtaZ0**HNK*RKs93n1iS)C_fOfYJzQV|BhD-F7 zexPB_&@-Ts(Y2TOr@8!*eQ%qpTU_fqYmd-ZZ5-KA0hkmkE^7{v(SKy|ya}G&-w7_W zc|gljm~MnX>%}A0TYxzT2@2PMCwLH*GC^7IXDr|l2WzejSD28R!hS~E!%4OeNv=Yd zgQXE#DeWg+Zsk8`RK;G&cf)yZmQ^~6u5d3ak&nN}cFJU@nCaPU#K#lBK=Q@73J$Of z8ngBpw$3!pP;rM~2&VWXScr8gs3ZXp?fFn9cP;$GYCU*yvr4F-je-H>fI(@d+3yF?_6o%|)!sd>?H z7qI9@x1m-T^yp0Z<=|Z^(2#EXS=qqz@g`VL2N;R^nIpp~)=gt1C)z;lk@03bo*T+x3$iw~$64J^$6Gn0m9HJSq~{Qj zago=SBgWL3Xkg`5sMg%G9>BcI*Uc;-?<|R_kA?|_XphQJm5q}qV~4?@w}<5Yd%6!!;C>?cmoVK_GyXT6tT z{pGx4SeY}WxJvqHXu$(+1IZBt9RP0ep5OIkl$M5xq6+n3^>79t#EHx&k}#45lv5Zj zSEJkL)Vct8k*@1>lF1_9d)c5)>RO$#`blH(W$C~(j^3GTw3tg2U(I+=Zf=@x7=#W& z5u|8W9Zq9yQBLuj`tG~i*7H|=AadU^(ipOmuU{i1vh)32h8C!09B*=b6?i&xze!s@&Z;Gs>?P z**u%uhoFZ+wxUZPCJxz7;|})#0c{*wHcLhr3i&&XHAfw&kNY@!U1K0(11FLf6}3rL zF}Q8qNI1P-1QyI_Q(2^_s#eo#6`&Da;fjb?{3v-~P4K8TLWpVb;2X#${gM zxg(ygG|n9c!I%>NBo8d_&AZx-oZqXatN(VWZw`sTF|tjiOOP^$nhML#CK*|kDe`d@ z?D^0dbVawcFNl-=#=m{8De(mUXgUzTb9>MEbIdHKoqB}c5vIrQE&Uh`j0+;f^cm(| zg)mV)aa17{_Lm@Aich&(jtW|Y;m)|4dKc)%+J<{L)O*f7QGJ(1WbpbM?tc|EBd6QY zqv7nle6+A~E>Nikio@tVwO|*nU-yH)JPujnv#+A z9)!4zbA>-j;p_xU0fo-jwQOlYX5juC#$)z+-(1ELitK4Inj3s-^nL;>sJlSz3T^hk z@D93z+64)>(16C2f9Qng51FBQ zDR0@a%~;h|^tFr4N2Xg>d+sh=>m-v&7Z6VZ?!k0WdS_{i`R3NHDZVy$c8)42(ZuJ2 zBSO;!kOhne6btqC(Y(6iQth=rP$#S+ArwLHUrH%dMdZHpJYK&UgujKOvYjw4$w3BP zev^*IXf@t0V1L}Het_ZcHR1OZ6yc(c%X;|23*anwH|pP>+~z{jkXgpPRcD6ewe8W5 zWY2coaxcQKd&4|1|H`PJ(Fz7dF^Q=s67m1+bu?=)mmFm%SmXsA8;gw&I$-AQt2wx z-%DFFQQBk5*&5zxSp$z+x-F8fveG;|RhFXqUAqL?(hzYrh=;2&zl4<{X*hc@S%F|B zp}3|_kV8mNHa`jq*C17F#9=(kQIJgA75h#El)P?HX_a#Y9H95=$|tF&mrGlvk7KEj zrLb${sz=(3HNSJ(lQhRP#062N8CvN9!=T`2MUt_VV zrp#frO=aOmur-pE)Lrt<8qi3GT*mlJg} zYYOM>nRPc$7$Y_ZbfTcdS5ozmEN-lg?f3t>C|TESyhc|Gt2R=~+|Mf=b0@5kIBRhH ziX{N4yIUi~xY5A7@%@dvVwMNKySrza+-KU&ZtfD&_HN8D_6=C!5%H*gbZV+i(yY+p zF4|aEE@P)YY|CY2c~05tr>4M9xFA&DFaLX??N8Kibb1Xy#~#CAM}XujnT_xWMrC!`$J>`#PDFZldaF2l zDzb%vdoScSB{$f!*V6WMBIjPslFVsbxeqS9QIzByFG-)%%EHDlabd_#S$uf8e07`478LZrPI8SYqu@P0WsIL=Uxe=aLWLUuIzsh}^(QSfgKv=w^z5Qwv7h|pqVqc z1Xc6ocoS`pE7)#+_4l+ud1~hEvXwC-^%j1F(`;woOMQjy|4)j3%0$z{?V17=N3&`k zq%lSc8Ae0nPJ0GVpHRae17VbbLtWs?ka6vNVjfDXJ{I-8;*)o-IJy!ir0?3lWT zqh$2%L58iDkJ+PH?k3E7AzpSs+bC7QttJs22B-e@y29~#WqGW4oBZv*2631Uz4(s? zLk{t}B6*S!?%ri)Iu206`|t!wsmiVHsItNUww3E10HdlAxT}1MH*|*i<})cO&ACt1 z{0Uq@2A#Lzluvaz!?APNr-ktMRNE+G5Sl7WH1V@RZ_avBd^=QxncWEyqI^-NnMHOc z=x_^Z%mU7)x2`Yr_?V-ZC<>T|1S^C74ng?Y*`(>ZFucEb2;w;N#l{W8_S$jB*$D0jg1n&Z%eatLoO;v&dBLc5hM0}Bb4 zG}ss9_XCuc^@kbA<_!i#G{VfJDj(Q!*?fmuFUToFj{c?h(qkjI|0uau=4;zb#65yq zb{sd%;W*zPukwQqPec);nR`y(Jm~#7N((@;n;gZ za)e&k(4@q*D*R8paHZru$XD19YfS*llIy0yAci{@hdNIm)=K`^%89*9SZ7wIZ)5JZ z>bM_D{gUdWeG~4BUu0s6JE>QPjjOyU>o!fxSccf!0e@=z_)<`SG=++dl>ucm9I2!y z?VW_xTZz;Z-=e=2oTHOx0{-b1uI7Z0XfL}pDt0_2?*3ca9B=0EOg?j}Z&Obe;3QE4 zfXj^T*F$n=pGOxCL{rRAspND@mN%7N_}M8&ik*t}WVg48$-@FN=)GDpTBLAYkO>z5 zE}%9)Y?v$T9580jBn<{RFS$fBmrR}l!P=^S(oiKTw_kI!lXWW*yT2 zibtkW#bxxWi}u*A3lVf%{@`cLY8{LteR$+S$ysHr?pa8G1R0D}ad@Pyguk_dgZGex zCqj2eg@7Zm^$iw6mCm)U+XMbX$!M7L6q89TKU-6SAhfv`RK6G&2KS`8;f7;YvP21a z=6yoSC1P;DbZM1dx~Q9}9o@(~ewwz9FF z`v0_yOw(FVF0YIn6)SROLErvxPcqVpdLUgta+Jd6;zr@%#pu%dkg0_7{OZg@LI2V_ zskiZgKU&Egpxp&D8EN7ec&yijp$Tbvs^1_e?);{sS&A)<=<={~ZJM=00i^YKwxA_;1YZ zA45l!oVd9i*B_q<6YCCp5%>YQhzVjmU0766cFna8D05>Eu6QYVmgYKo#!_tg7hFRV zXlkR-wl)_8CG1(V;s~Tv)@{d)3Tvx+Ooe97y{7%=6(-~PD4)9g{@eW|3#cNz zi~c%Hm{vVy!A#Awc@BccC*cIO$;93efj3LtkL90}6rv6?u!%aeFFwNaZbhM1Coyt* zrS@m<0?)~38J~(Q8Dp6#H{ssXE7Kr+ttofc{dwQ&&ajSz{}B)AhaKSUj$}ZdZ^*Cl z0;YWZQM3yUSvqr{IO#w8gdMB&U{p+dC3`?OsQ%*M`AEE_uG;}3tvE8aKPGgonXJKv z2eGxaOyvWtbbO#J$ZCm1>J9P~e%VOy0ve3%I$Dk$%)n~qY5&uTCe4SlE5o~Ae?m!B z=c@j*Q)&uuMBFqe>Rk#!DI=DADP$eWN^rZ>9Q)U`}`o zQs$qMrOL>O&+pILb^5h;S>7naRXSomLuD(~Dr34j|R7DpzexInh-$ zK*sCGKA-orQqN0d0ny#9Z+$>=snNdW8F>SD)RtC;Yvfrs zz0q38b0nv;ysR2Oo|15g(S1*pus8`Y1Pr+9w`TtSjNUM_fh12GmYokJszs8^mf2xg?ALl=Q7hLiJ-R6$KR;;(+k<7VT zEKmNJ$1XrgRR9p7M1mi{{QB_`cTQg!?=P^#v44~5QO9Q1g_Q4wxE!M6$U)Z0w>cGZ z*job@vnQh7iT`@Fa5;oT%I}qLWp5mHDZw;_%&`X_8D;F6&eTJ|pO@7|sjM$OrF!w5$yZ0Q^BZubm|$YzuH z*#teCJu-of@W+*0c^BV#^zt|chOowE44Cs*4KwHhjGK)Qf+5Fp(S$^(LD=KU`JK~y zJ6={AJ$W^A};@k9kSTnA-FJ)5hklRX_Y-*`#%*hGE!}``^azBFr;Qj zoqQ#ny>+eEs)?-neQCd2^LPZ0t~+0@La}Z9f9ppKmtGf)HiM|4qbExbv>U5wYwvnf!LKoZdSt*AXnluZcL*CmXXg9{7r5DtZyW!mV2cu<-?RBQwI- zR$PANQr7!~=MBf1)j-B@Sl{VCuv*dBep`hjXJg4Nu57Nc0$w|wC}_g(zdg+e%NG5| ztXNHb!WtXq!xr4F>yzF?GGkem_2bmpnvsMW<5ePW7|7`bs))I$?qEhK|BD_$o8i)Q zz@b6}$sT$tTza@Avayw=JUuw0mJoOD)P=L#;yHIBnLfE+gBY$(36h0X66;Z`SdOpi zQfDT>VW+Nemm;n8<+OL^a=V{d-p<^oJx$xzm#&C&xjQdz^+n@gGRlczW5ieT#V1 z;i5A0iD)%Lw0aRYcw~}n*oF{+_kU7PN5DPvzrxUH0k%Pn{CtXFYBuy;vJM~iW~z=% zavF4v45)dKx3?lU0V1#;G#?}TK-W3z3MXE}X9US)Yf4sx93{x;XlWK`_zX5D!g;o6 zy1&z$-%Ys-zah*~yI)CZO(VJp*+k7aoZYk)5Oxevi!lc*n7{_L1FL-($S)S&wE>BQZiTH(>fU zyg%h&4#q2)*9Tr;fdBFtTL(C>$=eLB2{okz$qC)c&BdPOFmO1w1)MJoYADRQy8=~V zF=O7gZ@i?JvqeB8y>_RSDc#>H-~sk|IwzV~z}~PXtVA+<1 zXRU#InWcS5Jka~Qfw%TspZ+gWx?Ao%+LbIo>p}dEAZ}3`(>%IC4M4S9G!~jSAlsuD z>8&Ad{D4aS{$m2OMOTKT%=#qfFw=&VA0ra36%8i{#($PScm0srX%+y_fg0DSU^IyrAa-Iu&GJ~GpeygSUB_+c_!d42{ECl3H5;e z%CsuU8q;rM7Z)4}fa5e+`@Y6k4*9~Tdn1$svypN2wM3x#_OH}NLFos0Q{PTdz%q$8 zjVbTlZGc6N>nuCo>b|46;yr$TSdRWIksmIQs+J%kD|4lK3J(HYrI73=lh;I0(ISQ+ zjRG6m$#{Tv6k$p3nb%$x z|AL(~OhQI0VLBO@YpUCw7U^Qt3gMn?qQbTua9dmXIMnz@2bnOL*fol8b+4yl{5Ztb zNG`k@7Ez;OS2%!-T@c1Y@U}@qcR3xd$L(s2&`NaSGx-qQlklswWo@r+kV2}EXYUv9 zZjkP_bLl~KT}ptF@Ea5L$! z;#G~kxW~SPS_1s{^D~I!JZkP*u>)g*?aaLU&>%cX;w`dG*XdDf`{#&_XB+TfrlyGf zs*mc`QPxy;OY*vO(5AUGSxLPg9>%Ctj_Ma>Q49cBtc>98w0uE`>SJZua#m z(drAzV->rwmU>P~C+etLp9tcK-D6fnY3Pru}>`=5rH&#Rs7I(#P}_I(r6~ zs;p>F&xwk%%KV?#cps&Ygh`d(7Jd}a{ZoP`BwKh754jsIZ>e(!a8XInh(!9e?p1>J zE^$PmX>>vct)G-7k}+vm{jBx`4}SE&WwD#hBNc- zn^2hKL!{PjJMg@)_@{$#U2=s?;>?%73__tK4-ezg#` zFCW-E=iB$%wt4!nO8r7a1G_W>!vOd`P5)e6Jl2~zIv1ha9#3X^**ooqbU4F;BMqsm z)wtS)HE0?F5sHfQis9qZJC^{r+gFL5whxLsUqC`wIENSM+7%}LP)VFbf94q}9!2P# zq@*lw>CV=cHpURUL|(3AdLnTDt2q^DB2ox2w|^Pv9&h!#O{v*?%u)u-S7fq3g2@bUZ z;yHSn87>+Z0G%dOjP}+Mhcn(R03Nlsf8SlyZ(D+fRdXlhc3z!m*>1Q^t~6K46NdsB zv`;X5ak?w4(k)ebmGc*17^P(+=6~`j`3j?E>li$ojBetODW$Fg5wI;dBC9U22iQxC z?uTJwWoC$K%JI~>4Q6EvC`Rsq%uHYt^YRQg85F^w(ExKg1~5qFcM))6X9Sd(>3kX0 zQJ6oP^0U4H+r)$8tj^45yxZ zulbV1>mP#?D=&7A01yg-rdfitDRwKQX~<#3k8jNNZ76ciDqTzDEF?t`13cwqTxULt zP2LI)QMI*nCfOxH1~6@0BrG4^Vi(Xo$*x~#na6S^FrW`hd)#&R9yC;;_@dazrdc}E zN${4Il(t^*`R$6i0`v>?^GLZ{EVi0(V=hszJ;rALmXpU@u~&oz8w?qYTs~!n+)0X4 zelfK*R!6dc8gkdLrAnt- zD`*YAL+jXgst^Y)^d{Z}wOnt*kqJffeOvDf5KUfIRpFMRFy^XZS$Rn?e}&jkLv}up z{m9=`=eo4p8ZSjuS+_`ruOIwUm>^9;S2ncik8{ovrabK0Sz1=)JBcolmM0d6{54w& zKmjPn@YZmMr29!OCjCH)-M!8!g%GF!B0Gp|f#w^;SAY93GW%q%Gv4kH`4}AK&cI9}qFQm>G5us>Y6dBzP7iQp zW>dR^d#}G15b%2^T-$EfkZe-Z9Qs!L2z36+{CCE&@&t2>)BQOG74F|BKB_eBa`RQKP#E1Pjyj&jrhYN-qn({$)$-vyJ>8YSi-va*}Pu z7)L06d7$SQCy`M-lZc57!-x;pv|tWq@8v&U#^!g6@n4kLd3b7{1A81CAS>(Pju^e? zXB6U^@lM9WSFhQ-5~IKegn_ZY;!ujrM~)Q2;PjIsJOxiV=MUzE+?cKrs|`4eXv#oE z4?-9(U2fgXkz0l62}lO~k76CwSJwd+Rs0R6EjwG9xVadpQ;~Z;8+qjxkDAwM-8`kC zRnJ~At8XHRXXEuk@iy1`@O)YiW)a=9Pm#0!WPPHGcZb@!!Xj04v-s#ImbeApmbbIKTVP*T)@Z9j{T*N# z>k#Bq$08%zi}AXa$2#Nwq-xu1KS|Fsi^}wMV+@aoOgPDKcnC^6F7dyM%MUP)W)WEf znK$G#*h*F;1u!XH6zk5ixwpi(wHTqf=YGl%i)z1hL*xuP4?5y|qSxNt;P#rg+}D(> zoGQ#O)heZ(*2K+zh=Dy1M+QuZNeo4m4c_C(v3}Et7X5t>46wF$mm(d=p2T0qH0%fH zF4%XgR01oJc`hVYlSUXKMNuJIh`3+u7ch~7KM3GE-qu=U1#ScBen2uv zSC%Q-kwNX^(#m4}1%COg2%n3&#$Z2M>)Ol@%vVMJ07)9j8W1ld$taap#QItZPl4r3 zu#oWYk}?h6IY83!qP)!ujE*{MO|X%2Z_)qR{=%-eyG-ihdTH`gJuPml|BN8xaIFu*G$p#3z78Hq;aG&?S3vLlDlh_1~kX#^m%-53(9s`;v1^(kP z?VBw-QNHOfMGZKBHP@BQN;rgkZ3MgJL=*g~SuOQ;5Mp73$<><3@r|r~O2eXPPsvob z5v?tlnX`S{r|_gGO&sn!>13cjzgD!i4|=6*F_=6q);+Kmxz{Rhpp)4>g-;3Dm0z!WNOOgKbG{Vw`}pC{-VS5W0?xeLT;3CVg-Wge(fw0jzO|W z)dfFLN{5jm{db5uqd@OOh)&j_wZ#I-N2pzX;Y|0{sys)7!6(G}6&`)yt3Az4%zO%# zDkKvRHmR;27PlWgcmwDG-h;*THya&erkj<8+ zPUp7Nzd=z~Y3^y@;j2zu0B{C-NwblE)rrbmn}eOrDSphMvIZ}m`FT1&0oR_M?-=Mdu8?~#uN#Gzn$LJ!}58kQSwT3%5-xGq-SL2G*!p8`++2XmfP|%8UUt+D~UGCbV zcX^UjBEk?E-_XV2OXn=Wuqb^B;N&PvE0PMN>C_)~Imj7#@wC+3R~$(G`8Fb7*#yxe z#1_ONG-|c(aS+na+;4Q*2XDs4UwACVwiM_y<>f{!EdRDQ%}ylv7QGtmjvYVz@BR~b z8f`cgOvWtu&Jzo1?gt3@tV^*HOm_>w{4>a39tj{B<@=k7wx(;gTEA?$Xr@(d+f+Zr z!%ySZ9kV!Km7#4+8pSTP%N5M|p%FB2^3my39|`*uzgzDgp6zq&zSC6*G-rtTlT2@OWtYVBXRA%*_zzH0EanG5vHm=;-CMV>VCjg&l$gjG-QL zqw4KCQ}SN&EJp zH0Ob7>SWihR+lcmj{z$mA@Waf2=93dqz41lKu+xu7te~{s|cck8K!DI$l~x6-N`j8 zDYsx1>_)gQ^@WD^wg7V1Pns8(Tu#9{fd<@4#Bb1VIhg1I$6 zE9jRLys4DxpXe*Ce3)?qu%?@Z4goRZvS+Y*4LnZFQ-!Q{Qrlhi6AcxAq+j5gB(VEs zszudd@_TSJpB-P}uYx4mZ+z5soU+@PFgosK*TO&J`M+ytSQZn<_ZzR1 zXD6zF`#SVKD);?EfY+V&*@0RW9t&jbPaM+jUy)nFKhHjZ_8)QA3TDIoJgBT0HrC03 zsLshSaeMye`Fs7`KB*K37ea-pH3vgoUf_Ui#=mMaw!$v@8APm{>}YwoY5COg^%d`( zS!u$&94@8`_qc6{@j;0%y7BPhwf~JYlv&X!qq$-X#$#}^Zr&~6cp&v#cj_XvNHoJ& zKffy=fdl)kLi`35YbymNpB}L$Ef;pGtWzms(Zz;p&VBC{gj^b$;T}4@`;&Iaeb6z; zyXwoQO5!i@WR8qjmVO>LrgXN>boNixl$|Ha2Q-sKOX%riS%UI=H{hJ42mV$ec2*y8 zhRtQ8LZ{n#A9SQ!4Hq+rP zPyhFn-irg_mJfEud)1!|fpA#m{;oPeu~s=JcYHCEA9u-|<<<})^FgMxie<8#x#&Y? zL(eK^WOy^VuiBzmzCm==$DTd*N!nWU8L~;1+8V43C%ayp8%p>Cw%#qbv}I6AOxZqm z*UhrcQpO2w6EUl-ZKtl{UxQx?o$u5=&pDV82#?P-GsHI`F@Z+ep=e_a>-&`!V2TN> zM`)grZMJJ_$nP~N!5q5wsDkGj;&-X*=GOhFR`2C$mKG zGAkS|B?l~Rbk9>vUQUz(Bq2w9CgTt(@R|baQ=%=<`a>7yWF@c$=b>DgB@ND+)B{tc z*@M;h3^xdy;;@e1sF-3@hYHf{)LW@!jN$A1&Y<5RaRdBnjRhEqsbcobvK3{L8ync#|88+^7R!4BFIdg^rj#7C)SraAmMr0%i zGnoua*}5}%#53{7dF@SUCP7LWNgVe=VVvt&3D8!U%P+Jz@^#8N^Ar?sCypdQkij)} zRdqAIG6Ry^Q_F2eU1&|I8oU5*K#{*m zKH&P9dqknHSC|9B=L$}jS3)gm9vXcAlKMPhguSQkz6Owxu(lc zWtPatT!T6?c2l2rwGcJ^72TH*RMD@cdoueMV z&eT9ok!bcQyuW+7>r1U=a@Pxk8_db%Kw!5IW`MR{u7Qd7bDHy##N4pqg?N%+z4hMA zt1Mn*I}S8uk0d6xyk-wWu3a6fgVUe{oMpQdq9mW1b>xJik(ZRrF#jYtgr?72NY2 zg3Jcv&SC?QML4g%IWZB^>oBv~P0@rVx&!gCF0om!P~-3x_Z`(w_ZtSKFQs7ZWLim4 z^=J2V=1FB+7Q+YHK>R7uMalRk7Q?OKrl6v$WE?s|QSiyrvDRF8wB657otTn~6>6GZ z&lf8{I~Vn?6puuRLFXQA#6*{4OsJ*N6Tu3aS#t{xdXmXz?uST?WUL~IhYd)wrcq^I zQUMu~I?esf*)`qc5^p02L117`50jf$!YX!1KON9T5HEst&_d0@yhx%=pDV5_e+@sv z6rA37Hsm$1aM_~GR-hSf>xi9U}hVL*tobxlTwBu|xf$fAUndzv85J}C1I(?Ase%+{&q zQR3gzU#5!$0sqkGV%n7&EandV9Q;3F?QNdi)Ai95q~HH|v!~!TYJQ5Tc%@(knB`_( z+w{$+mF6UF(b6r#e;(#~iNcKV%+fF%|#~OW+H#5m~}? z|JTYh^ zg6Px*VWJ^{uoOcq20mg;Xigile`{NIJZPrJ7||^w6%?K0MI9d!v>PN3uYS*QhXC9( zbKG<{q+Y%Gg;2F!^3t5Xa7(FNJh9K+P;>#nt6nM{;#=1R)#F$t^2=r>oPE%O5BvBg z%ZCXbFu^IS1FIWC?PNV_!$FbWIN?4FN!S9m=@{$i=@iGobOqmBK(sPbe9kbS4-kj5DCFu6f8~UZ??NXGPlit?@BDSlzc^^30&JnEP{lhbi$5^N;QQNcE>)kK- z>}o6iJ`rE38EkAQ6zP4OtmfU-vco1lr=d)zLL<$f0li`tn^;{tjUkjC&@T^nFa`;E z{cxo{@9P4(x<%|btPVbTh6Wpav;?BKLzCfBJOW6UxpW6=H zXl~WbA)U=dydDp(@x#Ue;@zAz_T-uv7`FHx2j&iI30D+zG5pd%#{~&lE}x-QZZ^x; z6UHn)51|Ayj*$90J?W57>|BLwslUTK0l(BpW1b@r6@LTT6vk5T)hKJ^kgDUOjHZ)M zb~IZhbEvk(g>#dAm+X_tBE8BkYGYMNM}#(!X~U?GzWKltRzV{{m`DgHQ=)J4UIICR=%@%t|t z_2$}|zh@~p>{&VMRuRSmxLN61?M&pu#V1Z?z5Z9%YS{d5p5pKyG(2OAeOjJP%$JA+A5VVU^_My|-@HE)}bUgtzI6XVc7|e8_L$Mx(&vy~XuyV0!bDx=FQu2Iz zBHsT4=CD5wh@241`V{{&yy>yiRWN)HX8Ln0khlKPsMhbkF8KsGl(YmLo7Eo?3>0dM zkkC2vLScRk9qq5x25~d{8dQ3hOwIbrV=WAfS6ga58I@c*m;NibOYzI+SynWH4JYBW zs&6SrLc8PQ^2(f~Ewl?(19U$>w>N__Grhe9F}v64dIaY|Ll+k zcWWodZ&%b=VNv4l z)%F(yx`01laEQXvj|eiXH%NkIx%}h*Bg3diO2`%4n&;y3DfBr zH5%S9)N1J+oN1tDFeEIjl|Dx`jY5>qWb7N-6-lQ7shG0bWUpR$+g)PJQ4+nN$=fi) zJjEAr<(FLta|)vF$W6DBaii`Ie>Exo;Xzk{a)CQxA7YSWiI%HDrrG4bZx9D#VDHxm z{=8x4Ul07%#(;8LZy8zy3R6HVooWZ^T~pN_RxY1dpu&Gbs~Bur7&>$ZajS9KLuvrZ zSbSNfwg;(o7Gfl2+3F2+uy!3oJ-Z{7n<*-$0mCs}G00OZ{%`KR;bTb%x(6-(ay$FS z&wIIG#yrJNRk&~O8w840dmC}U=FYWF*1>|Dl8wfHS?QX>-r-{zWS|H8AW+#Gzm>lf$w zO7hwhj3hw02f05^#fjf;6KI{d3cMd3t`MFdvJo=j@}d~Mn8_rShtYKozN8g0@iF`* z4(RXqvMG%%P@JUCdtRsfFK5T!h-jzrU@Jq2!(eb%L~nelUeIjF3_3KIEO8F#blKL( zSo2rax>|h2Dax)!A{JY2W9405?8b;)GCk#oe_9)3w+n1FDeOq(ZEFk#g~`zAaT@rN zJ&S;d<5mI8+vwWaGj3bsJj%aGRWf;h$)m_TiI^+k<8J@=q0j?v!-}cp?3w-v zlVGJF7EnwRjja+eijwLU7hU%h6ZwIq9v(|M)YHvZHiFl-*aLVXRmYRvBU(KS8H@or zK67jG-Z_`kMmt%v$4usW8k9&SXa7r9C2mNkEk<^+5_M|WHg-+1boVT3Gj_Y%Wyy$X z<&4bv1q;Oj$O5Kg;_0iU^-^*}riZ{DOfVG*NwBpGciWy=_$gS4FWkpr5IK%1$SDh9 z!^YX40AfmQA!o!J`<#%50zUWxtuMm}!?PD1ap0pxoP#*YxNb4pOWn%OzDm?Dn@Io6 z;&!Xmt@ypcJbAC>iH88bALq)LoI(3_Lz20lYL_mKlWlKCOa?lz zR|*I-y?rP#WVlEci+^f$LQfj2hVx*CtX$t=lTf5-7iv14N!im3#VZs$6XP78=%z5psg+-ZbTyOvGHOd@}OhlwOewS+#Be1z(bSw`NS+y7tMx<$u>IO0M+t;UkQPnXuy0Z^jY^OH*e zTGY6))6$)#dt${ldAO5mAv?^M+ukQLNYO_i@N3E1fMFDQ4>y zwtAjOuHaXdy1KS0z|Rp{0J?HOS5uM9m<)uO8VGW;fR=l zd!-_a=gOIF3kLI~1*Y0+(wqvokWZE>ZM!;Wq|l+EZAa^ZGc@-dBCSV=f|k}V9?tyy zfbaTXUX{&?WqJOU8#!)tf}zDu<54aRG9)<800jSSO;L=aKWw*~`@~jJB zf@pR{Ym<);84bVQp%m|KyFNK-NSLb0S&(UMKOr~L^R2CVCU8aA5`^~k-ePMj@A6Nq zJ7vz|OrIO@Yu*in3~pgf$IAg<`@zeAtL8#7Ja-gm*qe@yrMZf?VScq}9->%i=;w$V z0NI5w^a4=6xI&fMnF-60XN08gUNBGK4siz`t_Dm=Kke^l?Q#}G1GS3(+~~|oq0eX5Z2`Z zA=>@P_vWHSg)-+qSn4VpH)Qn+{LVBeASxUYTQBwQe*Y9;z^DSIKDC)uF)(+q8wvc( zZ%3fU>x!bFCDK>e58C=GjbNSIFr`_kUdWQXaN2xW&(QIw(6rj##^v#K@a%&@Y`cK+ zLkO5JHjQ{B#jbEj{w%9h?t$77tXMJyw!IxNrn9H(UzukOxPyMbO>#Nl%#{(1$5aqe+*jZz0gGQ`&Ffwi^+9 z0Hn4RXz6^{pbiTXFiHHZaf=NAa8`!?BaYkMuw~%EaUdYV2F(Bf4B!Euq}+#Te+vcv zSio?JEraec5Et?yxA6{SMm^SkPVnV`8NpK75(}(Xg__WSDVHAKKeIVZTIr{c9@^`%r?MIh-6h^L34IJ`S(d%O- z)D?iBQ0RE=UGymykQ;il(fo&0Zj6$qI8=*n@j$izMvRa?kM6Ov*DsH-H*^8=ibtoJ zW$N#}@hLm96E3dNHmVLVM2kcx#DCK!9m{Bw0St95Jsrz5 ztMg5{X=6~uZtLlN(2O8CNBs8r}>9{YLQ~+OB#ok@z|NA z#E6eEAPKT!Kszy`ZPgU_!F(wJ>5X;Vh`ry^9X){xr|6k*5}_kxVRa&9b%nG&oc5B#2pjUo&$(!;*efIrCkqptJFK)?b8 zElQ&tf0tf(A?kk5;}3VBhkS3%_jZxseNr_^B{M!?^E#|VPTioYY;A~= zp^*6(aqKcnNW9MQbZ3{CQfVtR+C*5hT@W}G5Q62g(H&c4`BazrG#|tX+07XgaEt0g zi$`t&0JD$)z`MN1wToanf8{<^m%o{2dUz(Xj(h-yfw2 zag-01v3M(GJ*kXIXeuplS&z@}gXZGi+YPek#mJD#xcq>=qCCG@Q}*O~nU`LTP566cve| zD*{V`F(aT|5_+N}pqGxsFXdbS!$4yc(Bif^l~ks7vTTQcinh5Rhf<|=nW!CZAn8=! z+s{=vYh>o6FT)Ub?bOh2us^ZGV+#sL-pdSZp8IYPuEoIKWqJ=Vz=Gsqi^!FziXQwh_XPPqXoAnT z^L)=`UtPK8c@+fJS>0da27Wtmcbk__*a_;oF`(tw_Aoq&I{Eyfh4ltiI?=1i0-aPg zAprrPQ_Q2VS9XYD?#o<@Wl#ZY1|4XJFI~-rlQXJ^mL(9xn42TAYHe{e=iG6kJS8+j~b(~>p0*|ML*+8mOnNVA0#z-$Wgup-l|0rrIFlAbP%%{7qf01ppp!3(Gn{<(k4c z6to+U3TMvUujjS14}L%KPJIr8!{v-sEuRDTTG7L{5^!Si;_rQshOhbyoVNl=54&~j z9zVj_D8KVy^5;MqGUA0jLZXTx;1^~gWY*3ot*OJHbB@6)uu1qN6zT#}Mu-XmBMxHJ zcEEGu|HCo)R{6?{*z$mhFVMqy_75ntItj~JnWI!IKU@=W%`ou@9xVWoaX+MZdCI!KS|a0bCg|DtU6UrQEMOxn(Kg98)zlSfmni<5$JeA4?iS|ENcp&gWlh1 zNBhao^+zBiPT`rX41kM1+PDX;fL%zlxjv|#e6jbtiV&0{CXtZeZcIpR6Jf6TvRfQ+ zHCbmy&vnmlRY?0mwXY1V3sk`h#78C z?3`l3Jqq@BC*X^|8hXR^zbDVDhanw1QV#qBOnPPyTROS;4KuHaf+X@aqR~MLA(}n1 zhOr4SMCTO0XmZ3_urMS6zW2x1`;~Qi4;{TluLaJ>FnZ)XLIZP%{FqDw2J4Ce=`Nf) zh9<8{2IF~*GVASYzlz4&6`qMZaz1J)F|UmzW%5hezO(>I8wfVZ;%96+4~0SBdT{O) zIGb#sfCrlRpE6T9E&BPDedoBj3ZQ<07>-oAstLH6ga-OcLe^OrMe5s)NF$e%3^L%l(e@$kLH?!r!e zh4dt9j@)=(ls=F($F-4iyWpcmUF zeOu1|l63p4Mg00Ok3{;|v*GG)3HD$laMsR2s^)QmGMNMYSmoaf`_sE#1SExM+-T6> zTdKef5~GJqTsTTW3P9+ct~(+tl{SVUb^i~BP1e?1ge<3(_k&FvVH6WkU6bX512ywb zhDJs$>*#k^cGzYT!O;nSOj?w$RV?C|RbDHfELj&FA3B2@1?Gx( z^(8p)F1#zO)|}`ND|={=S>3E+Is2~CsrG?x&iBbB!a24HU_7*~^5%H_J9O-Noybm5 z7v_s*RV7W*l%xM;7k{Pu#vd6pKBAXxTRb45jvQ5|QkpvyGy&0!49L4N6yXK6Ikc(1 zO}Ov5u=m}Q=2JI-E+pYItk3(yypG0n(UeR-;o56Ws&8?zsf{RGue-|76(kzgiG28% z@j)!Fb2q?70n)_-U*O#8L-v?b%y)MKi{`#zx$h*7!qC%nxBu=wK_)Z|B<^|KnqCzk z=Q>m6b>_Li?$)gD(Cn`96Sj7`^Mk30NJUC_qU3GZfGO=+*AxaoxH6h7Sgn`t&jF;2 zNdIXZmPr+6YB+fGG``O$$gRs+;Vp-&mt;jsjHO%NWM>oHI!I5Ck^Ee!DBl`nH>Z}u zRC9>boH@ZSAn?An0!(iJ&a_^N-!UsN4cE;9-}(9ls~fLTVF}NArs$f)u?C6jGqBmU zUkMq@_8sQA7QylN0&i+iaG>JUo-vWByq2^r5RF1JLSUCN^C9|v(h z-Xv8w!_&=yxADeGOTBr4RxfX}XgBW=$mymqcmb~J8c{b7A$D#~bR|c|qe59)EFb~@ zFZo#stbJ3~QvL7HIhQ2{9S!3KFAduaL4}%CGG~mFrvc6t?j006ie#0cpUQm*8?8J^ zYFTj)REv0Kut9wkqMWs>U`s|8fph>w=`Yzr1*Pb~?*TSuB@pb83>uZ9bBI17&nDvoKKgm&e1>+ln_u3;@MtKZ#eYFAiUoTU$#fZoe z3i}Bgo&a?MpuW_V`!b9;yfzpcBaPljG@Hk~qGPvXmPG2N@<1#yszGnjA@d;omoE&L zXG_#}pJtoN&7-U%vBBHJOa2T+4i{`O+5TG3LmuS`7M^%|pJI!osr<7#QB*^zckxUz zEcgfC+a9u~NX{Mp+=KJ{1IbFj$?-}^8Kk`~1~kT_R1_D?{DLGcw#kGWM|gUOAp_%; zgaeYKivH=qQGD~yZX|Tv(0LM!6JTB1^Hg0Lc@vieFd8~Y89Xydqu>Rnnn`KkVe~#~Hl;1PgT!&a1oK>+%s>N+)#A@DZHS&cK>U`}&N|O=?1Tvo zC*Y=WRXU_cu6EaAKX|Bnq^%wUGJ=J)wb}-~EXb;&)k>*)UE##(LSzpb+5n^-U8SY!1@o>| zr(5Ybw=&0L!F=~m`o-|GUi$LWY6CQo9gkmV5+4O!{}a2N&-`YI*2~x(2X@pYlMWyV z-5GbK`-L4JLMPZ88Yp$7ex%N@HXyUNs350)Li(%*i-Zt>ukv$tJC(8)mi8mC2+p&v z=bvfVOhk&}hd`$H(@Z5>&gFR8W`D<> zyK+my8Zj@kz8YFK?zWC;_i&%8OM3^uIM=#%P`p#}+E99k5wCL3RePi$Fg`>FC0QrM zO@7P&xI?w=g7OlGw~rYh@>(?1$HJq(@QVyr3RuvN0d76TY;rbXPHnIU22s`y>#GNv zBFaF%*H-b1vNdkH>IB55k#aYqq%Z%RX3G;DP1?)m%`|o=F z4x@miD;;Aa`LYkDTI6BRw`!H$RomXi)_tH!F*4)d{LgBVpl=HzWgq_EHFy{ZJb}#y z!mxMRUyX(sg`L^8Z6;H@n8?|Mu`4rwa3ajgwW$b@28xg*5g(Yo3RV7Y2)!;Q9_43d z{3It7!sF_28oxyqQBP>ym-(LM?|&TOg(vw39{W^yK@ta5e_d<1tHF1&lNo9ZAOI_% z0oS~t?4yvVrQ~9qY1Qn?REWlk!v#tF5}gOji$TPeMyebQx`S)tN|N_CU<|Gd7;~|u{kzuOE@_OeSu zlG(iq;r_gA@$+YSpbWeEX7!|w04ntKtNVCm8cfcCyYERriS5*RG7w_3!!)YDKCz*Y z0E~i1-qiXCaMvPz@d1h>a&e&~l#G4-j8c^}Q~I*WGmuBlC*Ikj8wXn99fduc`OQV4 z*)xZ`(j*-~J-2t-dZc0H{4C*7znBn``rbHz=yXb!Pm?Bh8#*ufjVb!Z zG$Hk2IgmTt;oFl|Ho`;Q8p;Dq({2oTf&W94SlWG~=Xwvw+7WL6Ii(rq8hU}^cC~wM zJOGGvoq-@|W&~ALvr`*NP9Pr-(HDmLjQ7g=%@RRFJla6>*MWG7Y@Wb&4gIGo!r+JC zB)tD}S})k~&{|)LFVZC9JPjlX0OSj{Cid4M0>o5)_l1zaGa$a-+a2d2O#hz_5~dVO z_soxm@548NX_;Lr1+xs?gxttsYo;T1!cAKy7k&Aat5u51_; zdvN6EVV_%E{+ZzY`uV`-VoAQ$y|orWc-?exeF`SuC2c|M>GfMaPeW44fgGh|!Zlm*tY&M(=Nz*k#OuP5clRH|BsW^N4P>s>Xr z_GaWHe5fRPsrm3Ofe=(c`=i{QQDOFc08R24Qq=f*|C15xxwm6Wc=@(nyVh+f^6RKa zHFf`r?h6f1+xIMtC)W%kX0PFv?dd9)=jDI|%IjR6$E-&;R${n^d|&HWRo-)K)Lu8= z{1BARp>hu+|XOZy+P!BjiMjp+`nGM8SsgAmHzUH3uO=-dGWVwOCu$ZUDF zF1nRT;pfVs5cjvEzGk))PpnFI*o9w2vB(DVGFlf(pd6y=6Wp6PIr5=H6req?#YrGW&!R~I1K&f+9nf~h|g zX%nnoX;(_z7Qf}Z_)1!_D+L*?K=ub{1Tk39P(Lt&?rq9eaOCbb{SQ!8^dkE}3(#5} z#OK6P3F&x>*Xt=zIIy<jHTV;{hz z{YnBbCXTctfCY^dQC5_QY}-p{IN7AOP5vOwguE?Hg8cJIt-A-JtxoQGXb&#s86;(J zH(VFg@lwy~IpA;DjMtEPAa_@Xoi(W%e=lUqCnp$% zhAf^cR)A%0H`Sv3Gf2HWo`=erXt19+0*{G^d)W}(7R~_Wrfa&67Y8v|l zBC7Xkg@DTDwu=I|<=%)K0;sFTVh96AIiOz$UEijiOS6-a5!+u%nD=tp)lS?5asbq zmBw12^w(zPfz}lYvtSX4efP@SI&NeAvPo~Als2BlPK7I^Q$OqxDRe0l_88$A1fP^& z!KB`h8jW=H#Am)#Dw{=V(*rfZbMy^`hBAO=i>|mum=irU#!A#^cM|Ek7G|>t08Wg) zz-I@gfvw1F8R-sWmrzA_Yk<%IHnV$$pV`uIGJbZQW>+pUG;lL4H71U1_hd=Z_1$5If9!UELD5fsC@ogX6`M~0mx&UBc$Zbfr7t_6e>dgM zGpDVtje#;M80RH0u+lzVpubs{)Hzw9R9w0N`ky2yBNDJ5^)`$Sjk8ZblO|;aFKJ#y z2~n}3>COYgD+mN}XL1Mx(`jL5I}D|yhNTVf-0#lN?SGg_O@pLb1f7D5-};VGvEa2 z+;^5RK#-`SE6~gLLF&-9pXXaGULsF|ljp@Bjb<0009300RI30{{R60009300RIh)zh=b zinm%Z9JGiZ!E9IRG!b0Ghs;enXaPHaIYYw07(GAtBLC^DM%M&WG*x;ug@IIMVgTjm zyaq-bfp=hg{1`s@BIXHWEoSYiWq<%5p2ix>*a!Vj4Y<^>I zp3^#0bL=?_f$yOgar%WFkLtWle;XDvqnePf&=ABviKG?K@C*PY~R@ zw!rzH{-F~s2%EUF(U{b{g`$6fxvUdhq5M9*h)s?-!lU1!r*z6N#rw4ta+%@Zx3u7fu;VRgmUou;d>yw36x{! zrz)$L_f#4o3biB+Ion$t=4(;i>TJ28BKp;*_@9G4TS*M}dhMyuS=94=fo3cduGlNG z@I9@#`S(s5YiW=;%x$L%(0K?6Ub{D!tmMIe5mPl6w?5~2+U}6DLUSJKyD1)G&TxK6zle!nKdXIjW>ZX}n+`;jKS6xTJ8YCBXU73U)8W%Ozpr8R4d3##{JoU(@*PCW zcM7*6*JFn-ihr{om%Bw&)I}-*yz5?l8Jv0=UN71Ao|Jj=(y6A*6#eYqqVdKctXXBf zyoBr#*h?xZfLH;z;2A4QJW&uPJPxoK+M8Mi?fil$^$j!XPKE;oAhmU~udHO^lEJ`F z2i2!!|LH{T?3nlqT_j8;3n1zaSfYac34GjW8eY)Hlso`{A0nY=o^KWtCk=aaMBLog zc{AWUldt5DAnx;z<5q!`S@~4xw5ynx)|l2N&Oa;P+5hdJQXQP8Dxqjav!EXrh5qpR zV}HR52$1g8yhLY_uIy+Xx>!u5&<=YGXU2-BZxXx!DXq3@1Yq#W{WKF*B@b{#z!s`R z;rk!g2X+LRSNxl~%?t$ijf=fU5TGxiF|x&XpO9f}{h=1anr=yWnXXVjI7tIivP4X_ zWs&0MDDaMd$G<`%=m22oAo2FF2e6A<2^%+HMsO1UYW_4`Hj-`FHh%!%H3{|AhysbX zDFk zRYW{xc4FvWgFU$Iu{@&V+AN?uxk=J}h1eJ92I0Gijy(%81$V`tQZp`*2itHwBuyl|gM~0YfdrS!c`%nhF{q5&<@cYV3R>r!Jh0~T_lX6u z#Wkeps9UtL`R$w$QUno?GBD6s>#W1VzcdcDUf4Q)_+ubE8V=+GYT^vnj>Hx-ig?sV zS``?Q>E#1VL|+)~K2pCY-ZZcBL_{~^FOK49-eZcG1o`*R$lOY*nw+9bP*@y`OSr{B*Z^kF=Lj6-vtLsH z5EVDD-1-2D4~q%4FZ9u)=*O1462UiQ+f*91YMN^O`d(x)@1eZCmH0#$$=5int3oDHX+wL;tKoqv3s$vw?CkVbP{Y{62AFW+4@Ms0N~5XP?f5JV6E<<=5r< zh~K<{EZcc>8iaH41r4;9G?Fc4K<*RSSc|W}?^E@YjX7SBPEVL!24r67-bl4!bx7{{VqEELk6)8mKf%q{=3m5{d}c^beHNZa9TS4)2V5~!%H4hRW!*j3Zb!_$~C18RH3 z?&u9If)u0BG9^pPFVC*ZZi`M#b>*YT9@f^!Q`S2GXB zM!mzIb2Dj&bX~!ohUYlQj^z>VAt?sydH>?vk7>hJczEXMVOF3rEH9w?7eP7P$!f5OC7 zKm{;V*J({2YI_X~MS0_LU79*70w%`&ZVykhH8NyVkt%yL-uqnSL{u-u-7)c7qAd?NrtE+Wh z{Dm(rZ{3jWmaZ5M^UrB=hdzrzvvZg{Q+%;(kPagJ?6EMY7}Ph4T@rFi=9Ln}r(^?= zPY^0LWL%XT-Y`EXwHt&>4x{MJXXsilP0W7+-<`QSAIw1vDmy?HuJM97vo7__r8EScq3cE4gp*Wy z8LXK50gsN$%s)^vJ6ry0SQ)(t zljW8-7*}jL+$?m9#*B8bq@atmuvuO!cE)@G&7E|vA%8shv)BdHXw=yc1*pPY&|2_MDP zdzcv>Gd;W}7yAYOJWRETOymP8twrc);UrFEN^R16f<9vC<1dgMi{6Un{v~pT_)M9% zL*J~#U~~Eg_s_S_A{8x2Hh@DSH6J$M%hhv!qCR|Zz%TH4w9*uE35q+akE%mIt-H9^OS<8YtJT#Kj#jjJIdd&!cy>~Kz$FB<-2~QM0 zhL%4!3rP2%MKhj?aqo`Ykg@1zNxxk(WCC40F=YSh*C&*K%@j}Hd%ud@ps%wooGMVb zQy0c=z2uvgOP(J`y9q42y`oO9fTBB2ZS2e|h5?oasuGK=nO|yQ#|_%SGIzU`Y>HKS z>SvO(tTtEt+(=gb!j<`>{igwA_p^yNu3p`bOQ!)&%l1=l%x(^`@~jdtoXhU2BsgOd3JMmm^6ph|$!6CRC zToH4Y){He5l?xOLLYm0?5>V@A(Up2&(ATR+`L5AlNuR{rtIOrL+=hE`p^UIoGobiU z!`@d;=i@Il<}HScpYr8JngX^}jPUk;9CAID40v0$lo?tN?pq5OGmp|&w!K{Ok6aN( zQP3{uNsEuVLxXCD*c)IFl!uIcf0Qr7BJ`u^w@Ni6_=j_{;z7%CK^jZJpGZ40OVSa6 zCM<@eT#aIP%kZd>?U!H$ja=+qvhG1ZMxA5PUi?_gB_EkRzp`{b`MXc33{cC-hP%i9 znTxNYyp(qLpX(gZ!MO@63D?qsv#@3|naMT+CW8TiCw6_dVhzNdm_j280?_uo9Q_=1w7YtFUxx&lIHXkHST@%zn4PZ0@ z*EFGC2uMHYDfA4p+q)DkB%;XjlfCvd1ZJ+Srr>zQ2GX9qQ1dAZwTVPQ*SFU1<2DdM zBmE+Vc|aBqu3v-AXmHMAvM6zFu8lu@XJ=eFFwY@k+O$8PC5}iCE!Wv@?962*yQt+N%Yh zVIIH600EYk23=h#tjhxpNF*Fcd5&l+rVa-uL5iDe^h=xlO#v$ zHDy+EhDE68VTjO`WOpOT*{ayh!KZe5@rxejg38+S115#=%6?o0h0cRC?1&t`CD~hsgUdFcSBi@+h&1vcDNwySM19? zA1U7%M9N8$+D-u`xsQq_2z@Fqv;G0Q(dKBFw>_BX`v%f;vf0r_248jO`@6l!moM=~ z;{*0*q7MYF_3~C_lUq#jBKtlgvV4Ft+1ArD|~C}ovBP($Y~FNxYf+Jz{(EhyWNzbXGpbyQE)q%i z$NQ>U{2nXTOoDq~m`vs~Ibrgv{kixvbG*Y4fB7~VCGwnnW4;XI{2;FEa1v?_N!(m2 zx(+t$79FY;j;fV#ZGMrX>|p@-PL~-+2v1{dJh^nrIK*PD8P5f^N)J(&`PsG0mIWH> zo~!vEkTwlH*Yu`W&Wo+7T0qw&!1o$0+yH4FIO2XXM-+W$(HJ>Ffrs ziT??`Wf$rb5l@|mgLx#he{@B-Jd2;uLpdymhm4abXN?JZAQkvah*6?KBvpN@s_6#D z`>}BfbbmZ8E`Hg^LLX{C?+VTRcU52%Nopb7J80_(5$yMup-mHDPe^S{$gV45fcGlM z33B_hYKwV=lKb6(H0Y+BwcD&1eZ<0eKi!8k0%fNZtm$l6hJ|Qw>t?FWQ#{&<*x^>f z=7id2+D3C9O-OcoXp^o%LcQ8}qQc>pf^$b{fWiG7wc%{+Hp@-ucq2hEzPQg_p|%!Z zg1WV8|H8M{ADj!$Gey=d0DpzhYbxR_IG751`B2#aP{-PMJzhL{{h)W)#ZpmPuk$u0LSnn zl{zv;j}RELA}=WIEY+s zpv(+punrZOAGF!Uk0KaoI9ot?k9jom?4ST6XpRP}vBoAOon$zA9&X9@Zy^@co{y62 zp8x?ytLpEBmT8-4_R-45APBx5Q64a5!j{RcV{C!`>86Lew$^|L{yso&m>n#rr)}~3 zs(UDVh^S?WK!|npT9~L2Fd4w2>@dE1z5oz0fnD{bq4g4(SDTBsD8u9Kd;>DQO^-9(shJ)fd0}xws70 z-Y9n^N8>;h&-|WtMcDkZc&IpDiby8}3jOKfq`Gv0jZ<3{jJ15|-kb3(!hEs+J$xx7 zLu~_k6#HN!s7zc;@oK>K(&a?vEaiLTQA_IqB2W?D!dN-{@>}ev9ychE#Tm8C0^A?x zr1}G~Qv0}T0MP$SiM%{>_F>ScS!wb#M%h>{8Lzz6@rmx6u5=U`69f1o4Q&aRku#rs zGey!5cb{R0ub74|#?y&w&yJMUxMy~5bKuh$UJ- zhRq|Fs(>dMDX2@?$T!qDVj$sH%?-oV&nxWAe-l*yX|#5K}TNmh~LJ?($z zvW@!g3Y_adQ#V)JBqytU995UHG?jpBYK**6R3}@<54Ck*#u6lPVV8m)Po7X z%0cHT=DU2oEd7l`@*EIU(2-tT_U26Dyv332XrP*tty*Xk zyd_%{cxtvAR195HXMFoK?B-jUM@7+^CE?AYt$bCCi+eg?IsiOIM3ou*$&X|mwTF7x z8`#{+u=`{B!KStpOF8Xu#RuQ3#2$x7gnC%&y!9(tPTF_t=%chxDCym(S;iaMNA?|ZE>9W0*EUwCG=F+`Vd~$64uUV{`Rfufi z7XT6iYTdp7sh$P#G9ESPz>VXdiNVm-7zZXUS-kmLOtn2r4!5dKfzY)=uuIh83Qv{N)wKnI{i{*l=j$Cqm_iM2z|cbRkg(b zV8HTQzWelOnnZCZWy>MwbV#>25GI1lRWlHCMIEab!>f_e|Mh>dgfmw3jxm68Fv_T1 zIS3xEcE!6CT+)#yd4Gtwl*}_unfy+sZVa|4=Cu_(=nPh~J6M?w8}rx%C~;|j6(DIs z!OEPUwQ}x6!QqtRAsFHqWmKXw=)oyjo^)n{&{%NL1G2+`shz2N{p!Ep5BHG7otPp= zC28agr*ZwXP}cXeTw$b!Qs&6FE8wgX19`0+&$qnZ#bX(ZG!|nIvd#t}6byo>_C|kj zHSr*;C3Kf@dp8|JB#`hp=5sxU5la|m(CkDwNsYt#-^gP>s~8wbsju<=+x!uMR*K{n z&!szDeMjt^N#|frlo)LSH?-IVhPR=_O_dQBVdx7*B!wAJ2Nkv4SHF8vdL6lfY5n}{ zoJH4kSf%%JZVV!wgr6hIRYyGqSHc4IQ+~!|*hwDBS=3|L9|jf=ctn=U@8U4IQ2+sRivkGGh=%!0+NwTi1MZ_fnbX1^ojIg&1k+fJhX?+jB!?(pfgqVoV8em zpUYVWnlzdM{8eQ~;3hh315HHnZHb*k1J-rB8RaFIzWGGpLEK;}1oKh->!fJw|9jW8 zNk)d%^K_|__hy4yN;7T}WIPdA%5y{qDSypTWSq^11c8z(?!a58i~hOXlM~hCGSs6f zk&@o(eL2zj)Xx#ZN$9hXDC5h>lct>GMpf5Eb+@Ef z!4AFRHP)8L?Yge{5eFC&nJv~9rMmBsc~MzD8#uphU5PssD<7qwl^6R8XzbIQ^#?b? zpsXlmA+l%n9fFV6(K+q~C=vFfU~h9r zwxYn?-ujN#-Vw+&rQT!uA>)M|Xl7}FQ#d|i;DyAWEU%merU zO|M=D%!%-f`MMlj{itubdec$a$6IEjJlU&rSJbN*jvDV1P1P>}5pT=V+RP$xrd_lO z45Szpm6`KWIP~5mDZV5)dyAaaK{nSQ9#&qIE<@sQ+|>nfoAq-@x|Ljp%5ilYe$Nlw z?EX05{ORA#?v+DhwSl3MY5RuSdMprZDkxuAw^8_*%X&f0K^+=g&%AV@$Vp_swOY!{ z6pbTe;3D4K)G=mss0qV`N!m==2xB3c+gj`}U+P>b)}YZce2bng<{TcT>GJjY)54{C>=~i9cUtc8Tuo1kBg~ zfb%{qK2s%092THCRa>6=`MZADumkN(r3pcuty@53+>+K<-Zc`}H3H{av75RanwK5>K&^XI|hR z9-sy1&y6a=WpG-PmERsJYe2s$klRJ_xpk*oXbGXxUp~dqdhIP!RHI{#4e}EbvEg37 z?>$q_Hs1$VY#5OR@hiEOL(hQw=(cHk!LAx2y_M6fjdJS3wWQ=#Zi9o(&TjEH16rK4 zc(-WSGB-)=$M_Ornir78uWrT_c^dGR<&c~?y+G72zOyKEQ-l~VRV5deLX6O8Ev%}| zC+Q^&w0vDFu^}_@g9B?bQ_xsL!MYul-Lg@QzNiQeF6xbZ;EA9ol`$2~8Y@vK zNrvt~ncMbzoY_tqa6V!s(<=$K4FSO7RP~Mc<*b(;rwbvtzP?Amby+73D|h>HdUb>X zyu4zxOlyl}VTohwGDXOoakhx{G1)FiXT{q^EW9R8>>2Up^6sR?I@X%%O8C3wxMU#} z5x}=tLSO*~jq`;HaA-$hW_l||=Xk{(K*O;eM7a2{#++yi?VF*H&%K(+bq>gGB|(0) zvPnZ{WrgYS45RJH%>NoKTBN(Rn}?M8iVhj~Wi5Q#avB1%*U>1bCil>eA7{ISO%U6P zyv#fo0B5GZ6mhWt!{@*g@Di|LOWL7WYOY5f!MpRv#0?pPO{p{dKy)wBh>hCqR`Df} z?bEDG9X@A+<&oQMo@usA)0&!ug6w&PI7~Xgz$oE=1|advqCr5^$A5N+5$|<4iVf1h za|wT$xFQ9OPrtaCy5O9tQdjPPtP<1#+P&rVDDBs6jKJhl192+v4P_ZD8Lv;r3m^}? zV4&{z*HA&0g(F=y>0T7N)BTKF^y^@*h~4sqC`Bz>zp7w{3xD^B%Yx-Vm8|3Y0ivof zo6$~pNnK~ex%i5-UAaH0B{^n~&wz`a(e7hZUTxsBr8~a3l0m1anVF?P2ZTSJC8gfi z5S^Y7nYUWjMger4rH1@~nu)2%N4YoDqbas%s0I|C*a^g9Yr#q@r`N;3JC3MN4KXZf zuyOivuCnj+Iu(XdnC;%AKsr~?)rTp4WUe!d! z)o@oU!A{2sgcnDG&^{SkIYcZzPe2Ina^q#d^JtIMx$^J~Jg!g5MbLC%&t|6^YUuQm zprXx|VSH|zf4Rj5#OmLO&L4vrs@}k!d#w^oZ8%eYpJrxFhWrHLR^X2Mtwxr}dg4%X zNnB;m=CHpJg#X^I>y5nQ`GoU|3QrAGHUuoC$f1=u!+D&=aS#>58m{&Wj3AC>&C{#B zY9$5fifDE+Rn!bEq-}(U0du@>5Ete6dbxxSTX_q@{#gt&7^A4AYt`gk=t6*!L5EI{ zGTl&3-YZoLTXV-qYZ=up``lOJy>=qZBKYng(Cs=9ln+zYmPE}vBV+g=&H%GWUl=>jM5`P7X6vX=ST|Mg_ zQWGpjb$y(!k0t?oY7jR$E3?Gac!`3PsiN~aVF!%Z7MV&nwlj^tvsx@)l;W5qzuLgW zQPG9!l;&O7)p2aju_4)zERF5^HWME8`rwkawda|pfmTD320)*^NceHmfz z317sWMzv2)`JvLofl&C1-J&=Z>2kQsRI4=gOpG^SpM3eD8U4YuLTSWi0A!CwR6^a| zu^@XHA-MB!s-9FB(4bq=2n>ls8W++aj@_zX5c)lBY->deO=7|AxLv0O!>0PTJKT%O zOTm|B|5~J+tA(_nK;ZVeVIE}DM<1gAPF)B(CdLO3{k@D+*`6$mWwiir!_^+Fu9_u@ zRIqu>y0?`mv>6SBNIRDIV58{;Gu*=vk2zFRhB~K}^xI^yC=`&yuiG@2>$jejyyn z&tahydua>Vgmr@Q^)D}#xw&RFi>S@<&gMOJ993Y4UtUzbdNW^4E9d2&cQK}Q9G1zH z|HRs%tPFcbEIv|;m+LEU=Xm8vMWR_!Zb>#at~WT8H^v18F=TmxNjb6lw~<{)b8RNJ z(wq75sq4NFW(Rt^I#2r}D`hldFPr1~SS{6N)`n8$i&B)R7gPEc6c z2r9m9`4Tl47$Alyc@%*t@^<=G>a16pulX*RY99&B1W*h(%&b2=sNtykbbG1;Y>J6WWqCFa+_?ftrQaq)Q3L z&UV;ic722!Yej%p@#VflwMEL9C_hFt@+JG$G1yczWxKsBDKjqPAWrSt8%^X4ZspG^kqesEUW zrTfa4H;Bf(%ewNSEKifMm68gwzy|Q04(`6LO%SE*)AoN@QUfzMwPir~cZvU2;!>U3 zJ^Mef5~=NJ=4k}tQz%{Nj@+NgFkM3|J%*IwI-6a|(&D0g?Q%M?(f34_xVrVfVIbp~ z?|@IyR46_mF_ykoc&S=#SllloyZTo;m{(ERgV7<>wTVVBDsUlh0(p9GwFoh_cvT%6O zapKGoPLXpuF)(MUbLhk&Mdmft$xMXXV2ljSV_D0fK0$U*d|;my-@#&*@~*yhD%<5q ze7PhDOMS_wXA<(1-O+2{Kt~hd)3@q$<}wRn75{vUh)ZedB?x2h=D*9x^1mZQ270!D^=}3`wu4HLSmP4N!)JhVte*5Wd^+H{7m18>jzznAiSj^qRYO5H=4QiKuLWc$ zB&dU81d-ZYGK4P71vNAiORKtKNRmMY_mpK&^|su@vbE}nB3;R0Fz}_mBcy1>fVF9z!_W*(pusC>15q8ugOMn0ZH#MOn#DEG>=w3vE zr#JfGo2`mau%xt?Y}rCUv=}EJrj>=e^t7jG;nkBibO6ADjRPZ@DKAP=O2N3?31u_0 z2H!OgfiK;ar!2)EsXv9D6SVo#fK3Ut9HCq%;-I8x7cyw(af#tA6KOLjh5DCRn)&s} zk^TZJ1WIT?{m zWUpL1yKJR;45xVswl9S8o$say?L1Nh?)_5*7-2C_Q#<^QZpa3DQ@vA%vftG~G1eU> z%ToLC1FRLh0)9I^Fh1idPGhM|h=@4d1?Y=9bwva3$WFgsHx^UGyW)%o;9e$8uN$ zQI%-c-WDn|yDBDpZ(MN&aE{kSoWFtHy6hpD`Lxa~HlvL0N?iDQq4#61OR1ZAetQr= zA(BJMBPq4~Y*Z6f*#SUyf3xjRxKaj22eb|7NaW)9OTrz2I>nDmr#fJgyx31eLAA9c zAn9{VI0FV$zzdKIK*F$>u~Hqg3UAZPWa)W`L^DFf9b%;n+01Sk9ZZ!SBQG1KS7Cxx z&zJqsu?#s3Ma@4rgVO{zZU8A2?3$Yi!lU(U> z(#(|;gi6HQyQaAke(Ay{8tz(GJ;a)>|j^%B36y#;XZ8TG#B(yU6n}O$L)Hd`pTcO z*I8RN&h1Hen3U&wR5g<>>)Z$2V*K+qgOJzs<3gt2s!5JI5KLF&8J~Z5Ab5NZulMr+ z5m`Rl+<34qIrtkXF~*hXkONlApb+DG>~*0d$i;vCi*{mkpHzoMzzLuz+hM!07`Qit zUuP?qS)P1&J8?6n* zA{A7vVTy)+;9n3Jxm1J}JNiArDl%pkX|Z&Pi@Fflp|kL%tT2Y)g`!hW^j&fc4L)5` z>mS`^${lB;Q>caleEWW&i_W7x9kBD-`TxHK`Pm*u?2;=Q<&M{XeaM;FN|%rlUMY-v zY&GV&m{9_I#rf>L?7&=&8=-_O28X}k-W6ldjHzEiyXXE$H1qOnNz>jHOQYM_3Aa_~ zitbLAC|k`VmY~mIz-d(d8O+z&x-@2~W+&}_xdU0oV`cKPTf83VjRe?M&E~{~ znza@o_OUvPrmWXQ`yoq;_YW`qcf@{oFP*m;82zX$a-cmQ`v!@MEC6I5$won2O4)$t zK=w_{J|h+}^l0o#Fn?)~21*z@gYhnk44!4Rr<#}7a=@8l-uNG9q$ASKgTo_LFS8%$ z-?6gDeyY&11Oi^@=IW#2LW%Dr)4$;|yX=)aKV*|`*+-am;MwOG()wb)N z)Kx`{_fWQZYgM)Elr$ChGy+f4$QuBv8o=u!C(K$;n;6kfA5lf~_Gh(s)o%?PPa%@y zCiNXYy{$=+ojo9fy%F8CSgyZGcM;@EwMbSFP=D^AvUj3+IEYf7B=j4>(og&N(UJ9< z<0)##V>d@XRkT-NL`@Ju$i`XI1&+hgW`moP{<+Mpv!z$-@OLVUp=Orwhhd^N2S9`i zudGPXOuQN!Pm%8BI@l&=p|T_?^DRqzt(RrN3nvziAt;LUY$)y+ksI_}@DOhDEg6_l zpy0WiGISPFqYP! zg5_+P{M7}b1pTW{TnPJMl5+NnXg`m;{=n#(L#CA->C?rd~bYW3!aW9s#ll=|sZL#>Hc21|VI7B;6Kwbh)b9EfI0y91D=oB^sQBOzt`S^#xSHv z1jHaE*GNT141)EB^NmNF0n0^ME_=v&LPPuZq#EVE9eo2L=iJ5rF38AQ9_=AhNw+O^ z7T#otQQEBGbgCL5al+b7y5i(V&F3Q#^a?z#VJbUfb8zuemd;Z-Rh+=0MTmvln@{7| z7=uvbjYJ7Q)_GJ7D8gF_p)Lp2q2B&OcAD*1fPMJ#UB%Kkgu!oOnb~)%c2flIx`LOq zoFm3AoZLLX(F~)k(`hl!oO@>!mGYvWVv{6n{A$jyPNJI0K4n%H?0vmWBOYV0>gpBqjEop<8q@O!tcw4XhhtiCK3WzC6(Bt`Q zPqz4tl&w{oQ^;MQWk-B5Sw9oNCq-}_X+}GM2^#pmF8r*yt3o;aINKvxpz|=vVia_t zS=alz12mv^pN+FaGpVNE1Xg2(-&ZbHev<%>H$>^g*J-n#&bnPD+*yP#M&1AQ#%brq z3|h@7RC`~RyMgE0R!gS*4zM}*uRlw0T-e=a60w{W7t%AL6G!GVhp??$ZpAoL0BFqLS2GX6TWZ9I=~#9|Q&ng6l8L z6il3QH|~)Q)H+jhS!5TWbKkEL#_An3KPZpl+WC6*Zy|#NX*7cIn_&=0afp%6#>Q6z zjzc&r@-Cgp9BLhw+bQ$73>h6^3vzwjrKKf#;Np-kR`1U2D0n9gdq@7@CW0p-)`Wz6 zRZ_iTK%^^;vi4Kjg`kKfM6!`kxzyfvoVWbb`ExQ5qK$aR!HK@je%ZH&nBEb^uu@MY z$Z|ijHxzSD8OF*Q1Lk!`KhBz|=byy>MZ^z>IUmZK_P`q|eA< zYZ3`dZ#vxM9FxO$HrePxcuvYHKvujDFQ50L;%yNl>s7^E=NsHPL2&|+yXI?`86S)F zK(nyDvcncdSS^dh7)3-C^K*R8_>r1hwBia97SgAiz(|{N+{meajvFg#pJYBc4AQc4 z8K^$J%YGV{y)fx8LZuT5Jgc)rZ0Il$9_5m|Mz`a3q*_AvlCVole@>SiBx#8v+)bgj zA8#gZ$HE~h<6AAlu?0(Lir!H6a#>;J`kgT?wxWt8ppDzkzzG*~e|OlJ^^l>ZB{U6b z*CYI>&CHRFuqvzGq@9N2|@{yb4sJfEW=XucG~;U5O4he zDZGWq`{kOXJs4{+p>->I9-xVFPQ2xYVpBu35_L)*7CDCk4u%li$bW`Y2@p_KrrjmQ zLH|es%oQd(G+$;helK^snqyFjXoBrGNW%R51QoTCXF%Yaqw<{C4)GFquzd`10?Lws z#ecqu;w%GGL~^vkHnN2+*Nd`!c2zkA*FEz({k8Cq_zH6-#r_ww!uZg> z0FRVfjI)aMz5)c7aFc#QK%~aPVK6|j=nPt!ES-F&t*brIlWr4$S13W zkOABNdbsXCF>)q^G1~!;Fd?x{^J0` znwh-a4Wgm5W!f;w>M&b{+}Y2u)c^kDBkJPN$3tbjEn`22pwzy5r0q&w5vlKkqn5IM zVHzr=p>f+V8K-OtUQH`+Sl6c=K6dtgH<)a5`hnqaE}WHMs6W+v@aTVaR6~K#Ch7t+ z)IHQB_Ve(3s-jGB_`XbDcID5&0D>k-G|VQ7LLs>(x_fEZQ_q%NKgXmIcYexH-PZvW zWx+oA!jglA5rhJdQaR}CVH%jns)=nAOB(i>kxq2^A|0g|lMEVGRcLOdt9lA9BCh+* zKZR_Pe!LQRHe*_&ec|v_hp{2231DYv)~tIBPUWvB_8?iRJKF8#+^2h2A-dNCzK;y+g*ubx|2H` z6sWZAe&s?^l=OGE{Bz526<~iMMA6QEk6F@LMsu9*-tEl4^Yk>@0&WWlGHUeeb z=4lG%4~xSJjXy>~&Z#Oca*n_eSfk9i&~glJ(EK_B%HJcAX6Rb#;mVBd?N0sUm6D{n zv`FNBZSDSJIbsps_Fk(41f@mteD$VX_DZ&i$xJ*d86Eo4+4+ zrAZF_|X|-PCLXuVpVo^ksoN%Z(KDulkQJ{6*s}!!-aL!O7q24_7wx6R z`hDw`Ey_#4Jz=`vKfT?Ggh^ry@iC!1CG4IKn@Qz=Ia*gkhta^S-Gl0XuUI4)Y=nNOHgwAF3LEEQjLk48;d8+Q$p3lJc z!}fcF@ZsXA-Dg(uSC-oM?aU%ubRYuIRL!QT4g7l<hkEt>r~g-x2-DE`XRFrxVjed6Z%k##?87( zI0XVFot>bRFn5Q}S>$AE_yUv4rt%eLTNh{@pH^m9#6M=wIJW2QR?_J=3td6&qB(f= zs$&5^x}y``nVK#v-np6G+!&ZMzL=Y(E9-yO`PfQvXzjsQtnv^2DSQ7Q?ViBT`!UPm z-qv%HcB2;$Z@3tI^U9!`y9kEw&HgW!CcKz-xkf^Ra0zio&W;gYi{eW+O*Ds>C~aG( z^5YejqYKf0A5A|$rh}Ltw(F;>U#U%xs_JoHSfJx3){S#fSDW&X3i7}1EL*5CG3&Ur zc>OVAIGpAN|EtYQt23#$D^e@b^Zco8kt*U+-vs*!X0W_3?bngBR8NXK73V;Zn$t|- zNFz#%eetA!INm>>E!mk}hy+LY^LZpJa)>}J)1GvUf^y*9LOODH^4!i1$PG>@xp{BG zl@n_RNdG~LnnLR3k5$2#{V--fQv`2>ce=;XuPlxHf=ZD&4+f{xJVZvQo_)5SwQ&xb zy>~)p!ymv8BP`Dv*}~gnF7}9e>C;S#q|=iN^TEWZ)1-NPY+UuT(-1EZFYR$1nD>g7 z$znbGWd0NF#tv}Mt0aL3xtPmA#i7_FzwrYWU;9QR8)jz9LxZ@@&d!$Ofrt z!tfhKxT7Jf0(EOiQ$p3moJGu42p9<3biS@hD0>KS988TBda@i{XYx=Yr%)~>T+~P2 zkpO;vN8y(b8(%ECOPja_faZ&5@?YOGW$k3_3G=ou)1QjnDW|;C=b1Dp)yen-upgfS z)~WpgpP7nsD}efSJ{7SaLT4%i>kq|*Q3kRsM}Ov;I*Jt{5WVE8p)}}%C4tEB9#(K1 z?f1}V+**q$yt*zG-G9hS24e|jw9tD`#%J4Q>nAH{U9hv6E}2vqE;=l>wT|{|kH_40 z@#RITkkQF&?+NX3Gx98e)~0PZP+v7C#2%aAYI2EmzSBP@LU-Oo#PgLRAg-Z!;g!!j!1y`WrKb~Nl|kr`#Zu*evZ@xYo$-jz)8zP zY(i-9|JF)HWY{(pgTC6m5hJ`N@i*)09DbVyReRnWF09wM!3uKA1gLQfC&2mW01kj_ z=irlOT38hx#ft)kl51IgTonSj?d0=<7?pN=QpuD#!aR#IysOeki-eWKAXC4}4P?c#S zeSE^V_9RZQpvOM;!_(ZCDcC@_M9wHb2M>#s4@gt;i&`+Ns{Od?eLC&!0fk}SRZV6{ zF8Z2RF742fkV6Z|my^Uu2c;U30{>Ig+mxNa+=;n5(T_A6QpH-@X=p^rYQczz_7bW= zx3*tn^Z>k2j4r@4a5(C0BG5~6!bDgIzqCW}7mk=js~?UlREx-Xp86eT)eyW_8ne)n zsH5eE=9V3(z+#$Gw`V^cR7f_I7)j5w3lY3m#DsFs0`SHi3U0UTsLEYPdTjswZ*zzHmvHRCn<#BX+Kb z?b`7XdW)PY=clkVgmu=D=XpLcDO~{U{baRL(=5!$3D_8J_76S0?-?t)i0W82rdRW^ z8KA>Lj;EYnQQ56!pbba4M6<%xOBjgt&T^6TPWpvaoJ5kju-hGa73t9cBKwp%FSf;-f7emUg4*t z({TR@Xoigu*i>Jo{4-olBFdGzCivh1P4p@i6gM@6THej-5Xl`Xz1=k^(H3*%+-#Gr zFi!ZE9F5)JzzOmqE-zcWkvb2f?6Fv$>uE$_6FO2;;=CAG5^DyJ*IL)k{%DSsZ)=}+ z-|rZHz$QW!zGkEfYN4*4#Q70J_Wq~#V2X{Xyuj%ZQsMo>C9Q_RlNy}3e{s>;m zts2l>>do5r6lhR9ncbvJFQt~WBdq~{dS5!GT9F6ws|?(59VA>{sK~Wk9vi2kR5n8y zH=m;l`084v2%{5zbJ1>h+529u0=nY(ZJgKlPf5pYT;UQpEgLpW)#8%hO;$y^mu_q= zG%=knY-l0X$bCyg{`9>6rkhzWVrYWlN=B@CcHP=BT{j<9QBo za?m7JyN5~WPrJLz+o=yG$UTv|0i#;yqb9Hgz%YO~EMMuiHID>RDg$?jQqP(D4PaW3 z><+WyhiBJLbdejj)@BCxHZK*|)ha}V_prC4m|qA{T0ysS%L(ejyuSP}dZ3yKZLFM4 zwa(A9@kPoeS2}UD9d5*Ck)y)H%-;8UAGi&Fmx_Ih))`FMqpC~wLOpP*UfIAs@r>k5 zsYK~p9e=v4TBT>qKKHj)V?`#^rY-|@^UEIdodStOqtNtCQshU;*k*p^tPtGh39Dlo zdCpo3Ub{OKWVL^ zcV_Y44Z((HedDD|a1@_PvLGlWZz1KWW0W|L)nErw9lD-k!&)AC*00kpvi6+#&c8?| zpJU_}0341*IdGlo@PZpraA6PObz(CzI%I?XufI4WMSQt1g^vb@Yn7P{1V(Y!ZzPtGE|a|>J|8Uft-_m+|KudH!3T~yHM3(Xv18hS{OmmtXk+p`yA_=00w;K=2nb<#8HVzMF#|ohF z(5Gvz($d`PyVDxqx(%GS&)GeiW8VuE=!-kEPIuL96g%-p3l%ftJ+BfE59D!^I=U&5 zNSx4btuPYeTIn2^j6?&d`{A=Tsqmhh2k;qFH-}QhrYPyy+-jRWWNjbYew%Ac#w`4` zYo*-4fV)5aW>P^A`ekwDa84p-l*v2F9dS||rbg89qQLAq1E8a;{rspNm(W?SwW*fA zGP`FsG|<0?G8dyJ&yY(gXE{6;o2^bxB1Z8PE|v_-CFPb%`_|Sne0P?1{R4Lah^rtN zaSYCBHZGw1L7Km${SD$e)z3XB$4(~+K;6&NHA-~^af+Ec<=5K0RxN=xS`3-OnkIuROmHB*42-QHk`nnn_MP;VOy+XU7s` zqzQ{v5$W)q!LwO>3Z+s2XYZUZGU!Lt5`{xQ49ph;^}o8*2F3M0T5yMCSLS3blg+8O zp-0b(3_bXm!ml=O+go)%0(Kcrj_w&KlK0VwfgfAie;yDsl{1HshZb8StkL3;L35MN zYt8?jp;3F_mrVOcSpvlBRIJaqr4hF?$V;H5FrtMGA|1R@_S4Nt{17YN9!1iy^)gt= zI3F&6vqMPGQTxs}dSx*yQB_N7iH13>0+LBga=dN5$@p@20>fz{C{iH5PRuUzga`;W zd$Y-8uSWu{=?zn+UZ?Qv|AoHt1g!!GyxzJiD!1?mfqYak95;i}>8v*aH>qA3|y37?3aEK4qvp+Ieo^Cn^QzJ{W zyl%LePSRXDe#Nds?kyr zp7%{1Kax2cLmmZj+H&r8u;z8K&vtR@B7>xk({nOya&U=dWHYA>5TwAVcUa-y-wJ%% z6sCn7oDV!_V&2{?&HA<6BUnJ-1)-{0yfJS!HzH@P;d zvp^Xsr0D-svJ&uHJOW_KF}2m`5P=DUIqS2nTRx~bOAVJi^wu`8)tuj{C74NJts97c(yV~t4737 z(it%{9)LG2t^j7>!Br95S^o%@a|$HEZf*d14qo9gM>V}(jD4xJ>qb+k=(Xlax0lcj zI|FQS;lPXd&F`Ri#Ru}4mP<8Pq34e9Zq%qCQwBlU9r5}&W|QnMe2LZA z+9>h56`NWhgr%QkEVt>g{S~p5pb-l9@dC&iFGZ`U$iRK~gybAx+dHFz+^aB_KZzoo zi`mIDoA=#N{q^*(Y1Dqe4duD|-gs%isk*ghaQqh}MVDshp}S^>%U{V9Ai*5~L-NAG zbQ?^#537Hv*@^%3$;a^Cf@g{unHuy+0~*_F1qpYA#nyrXHC%68;qW|ERvL@=&;=#z zSkIiS(}DwftP+DIq`bk&R2>|HEtS{ohom7QgK-zY_O$^0Dq7Sq6~ea8Ehp)GUnB&OfF7 zhw-LErEkprs>v2YesB&iMv6Rm8et=yh$&5^YOI;V{%9G|u@+n~({@>RJ3zt`BYPt& z&r*Uuw3YnhZNNQ}=nz1_IjY74GxotTAj##4E!RXudDB+oHY}O?`nq1YgnQ9oR<`a2 zr#coSs!GQJlGQy|YEh}f>vOsrVqwv&gcwp?(xMi<@oxHJN5JoW3|y0o4$keTwRPn{B}k3W5vzP2Ztb6^ws*7|WJO_c~`60e>Z)_eY%cvjL&01$0xnb@{P!u@_DRNBcm? z8;`|1drJ3BT!7S(^@CNciK_UdotT-VlZ8KCF#Av=1vdO`BVCdPQ(nETH7)D=bCJXY6FX!eM5a^Kx)e(_oPU#-}N?YfpthH zGE+~HM-J<|Bs05DK~e*M^eyaddkX;B4CK9g$EmUc8SWF zGL)h7f0<+m3q_3?t9MW?3pa`&^qA_>x(mg_*RaErPZGD8`Fj0szPfp*5ID-+T<@Ws zxyISTVSY0pd|_O~mNTv>G$C0J4?|PQX&oMcOIF<^s&kiLUM0Ink{=h{G@)5b`dc0f z19^WY&R7ptzkrGSoJSHFm%=B_cY4SAUc65ynQVmj$+F|UNGYAG`?)@bWb^L z7G}h(o^on43_OYBDJl}^v}f=}RATozY6nrM14$cxK)GqY?IS2F3h?~7F>NA*0#tvu zP|gv32%=AM|)cVAbsFQJbXwh*@2dOV>7=VfgKP6S75 zwWK*b>aNfr7O@WPHsfpB>rrC%aXcHrB}qI|sG7#n!T63w||Kp@RS)<^(d2hVJw44!?#a;>vIH zy^L$%N2uA>^x;>=bu<)u3ZQG`?2Z7mSmI}2OHkj&I5^>zco;*R7HA#=uGdm>Q`HoRVvqu5ejkxlcx%rwmBP^VoG-A3QM<>A zP*S^v*TjI71F5}-FrY1h0!KCs%I!|#vg>>WsuPBM)#*h+qf;vw0Ul#1UoKXs&Hz#6 zSeBuM<96v7F&r~@7G3jnaM+#|*Lua-cUn3F(>-;?pSPh9)6XraX?`tHa8Ee{7#)tP z-48$x{Xi+3l|n2| z{v@<-A4D?7tNR#8IUzW~9&F;-C9aahr!@H-^F`C^evHgS$4&}Z3>_6M$}Q0pqSW^s z$3kv(2=~?c7WYZJ40gk+ivywO1+?QfOdPqg&Ipn9O#m-I(7!s3QSMz*JY0n3tKI41 zM9sYx8oV;fpOw6`d?T+q#_^cmVUbJCub_6tzn`^AQu645dir``0}*-N)SM7oDVDC? zp?-y+{HNkgsWKd}(>>zUDjVDqHOu-^KZ)4Z{^9c_J3RDGg?JQ)+?Q&a3Fp?4$d)A% zP#adw?L!v?>n-{Lavu)2No1%8$GDy_3w+dkG5v%TEBk9VIysfP|FW9>F%v%qkAfjs zZ6s{mDra?d-^$YZ%m&yviQc#jdjYSW_|Ld_!_Tq}LOI?n@sp>}yM%N<1Tu(e6t=Mm zP*MoIyrz-phe#tbgc=mPS7F;H4}o6+HT}>WQc48M{g;x8TK!p9G$|RAHqQK{bAZ`6 zu-2GMGQUTjGA~Q&g7ZTzm*5U-TI$Y@vaA027_)LZo~M~#Sxs&d=!auY?;w2T$SY@C z)h6Pek{yAMs}S161A6Rz}bWr4-w^OJ6OOD#*(-OXQ; z)rYVYlo(_dh3D7W>-;Tq{SRvFEL$QmxcdZxPypjQSBU&q(oO6N&mgN{G>1+#^>+hq z^&Dn8=}C*yL3B`7&8r7>STl|}aG`=dv|R7|jQ8kV#P=s#Y$G&KSsd4(;T^N;gepnp zH-gI5^x6>!IpasOmLz0cnB2X_-IN#Bn%3+>_jXU7U@L2L*R}{VNHw3YD#?k+I(I$F zsJ+wrSimzlMEY#zn*)%3g^^D><9qoQEm60$fJFV1g^!9K7R<7lsq?Txo?(?sD}2*b zpoS#-T3POr#sJw2X2nJ*v*gP-rH%&oQvR}ZQOWG5IwZcCKlnco*>u{oMOsD9 zNo<-}Ea#(`y1VE3ULamfp0nj$o)67`v@B6viR?T!7_y#12fc_n4D6#muRX>N#?VG&%#M7Vz5@1s9^0`9 zFc=x@j|^*DLJ}JwrE^7m1^IOOL!osr?0a(nYcKEe`9LE&Sb^<{i9_LM}&tWHc13ni5&FVJ6WX!0S;rG@mp6}+#*a(zu1R|J&*Ji z?OiGRBBJo8E>|Pwc4f#I%P`$tN3s}??xf;cy{c(yA#sRXL%9iz8=yh0m?S+nI zt5&)?3u$Fw9DnpL8wZ?15x7o@@Vo6N*_oGY%=xt9P!CJ*a8%>$ll9qYHRBuQbXN-3 z^TR1tefN91?S>|t;0bk1XCDabl=Om-|Kdpy&0|iocqJjaiOWI5G*z4STV#^*&^2H5 zxy(n#IhbPC_GoFH&``}ps0m-S|4t=@?B@PS$N@lse;TAuY?j&`Rr2?Lomb4Vy1_tE zld%PF%LeGV8zHaDX<8T0t6KXZs%h29Sr+(=@1-2t4i4Paj6c4U=haFo&^wd6%gO6! z`JozW{7SDQ#e%nyL!wM+6r?h_K-Ygz>#TmOASZgJPP(BbY~G&@$AcZ3YPO2=@2k(; znRKL$1cj>7CiLzrLYZ74M)xIPG9%Rxt*%C>l=_gXw_5%(;+1C*NE~Mg|n!eX4-LAv_nhhDQTyc6t5#g2E`e7B6)Ls`g+N0U#36 z=op7!PuR^U&tzlZ3xhy4Rc+6%T<2J=X#OG!P!eijaS2ekm2IQ@J=$wtcc86NQbO~8 z%1N)DY$p{Gxj{2*G3U$KsO=m>SenU7*2v|a^sgHOkjs^)^)HMi1{XhBfC8_NO%sAD zDSQnf9N>>8#pKYPE!D&6S!O}NnbC4d7_-Fc&?w}M36{6^&KQUN|$w%|k#6pdk ztv&F6aX8SU+V6zfNa~d09T#3PWCl`V>h}_hOXc|uB$MF;bKD-Sk^&cyg;o5$%S)2q zw`Kti&g(!Wx!<_*w0a8!oQuebhshR+D)?V&Gm0JQk+Aqwc5Q3N)Tzq*Ilgfp2f5|r z99Rf!3_vsgsDD-UK>8;BaEma>zkEOwd^!jCjMZ7a*~T^ad53vY3Oi|TZCs-RG*6*|auyeh1 z$HWWua5_}IJH+zexV-F1Mt8L)CeKJOtW5&k5ea$_tZWryTfb?h4U-u^KM~Iigtp@L z&_!^NXEfMp&h{DzY8*p(@Iswyduus2OL8h;aisgq0KW^GizYB9mJOtb%?Mo*T7zD;DbNFo%0K4mR8uFC1#-Nk{8FENFr( zgA5hJ_Xy4I$Z3;~Vs|(8ZanVNqHU1icg9Y%6nekS8CaKm=?uulCQK+1T@U}QB zU_!WfazUQUDa+JD04k1(oF^|%YbD)^PayI!AH~a1cytG~{QsAQ7JnaQB<^_WlNEvX z4ftNYH+Lnr#3~2H)^#kQ$yYsp%v2W}S&n2MTum?&$2K*qN%ZHQ{s!}B}s zbQOSoF23DH146=(Ad&j=5YBli07`DHDHL1_W+Zp;_qwGUYSJE@y04{}Z^QehSKlsk zvvi&=vC#FV4x@*JGV7u^^7r&D15#Q0Jslf^Xw7gHj-LF@@5En(lWQu?viSX{JAqKk zht8`O+@&b3hOfv#ky-j=e$xnR1u3@!%}iKjgg%WfU;CP!tteWNJCI za%#qaT~@+P-e)IS_bQp2)?%1rb~uX>7A9vKeIJG>n`J~t{wjU#lPlW2`6H30*+3%N zHU7&zV8hB4#Z4ZVDneH-iP#zbM#&q(>N+Iy7{~deN|Xz4s(7G;V?Fh7c9gb^mn~Un zIBtf@7S^_qB}nI9k@+0sto6&sU&OdfFQL?|Xrvybs2hk|Mx;+F@ul%?W`M4%*(DG@ zK&PK_v~WBC?w>ExSuHJ0vQoW15L=iSRHUm;M%IehZ|!(|vRN_fp<^0oa6vMYcZ%OF z;4^Vct(v73MGR$t>I#lM9@tW>9%O!tAUI5qx2ywf$iSq?Fl&rFjVC%c*koJfcDaoE zK!UwUe$08uy${opHb~-vK+8Wij@YEAzdw+_ZR1C`ne~$V!#V5|kXZu|=R17)v1FyA zK}Z%}H(!LuEGX$k1O!B}+$q5wa+>f@)lF%iWGl?K5o@j@aGSolo_t}Ks+mVA#yIVw zI;d+)i>pCB#mfSLl##Xv+J$dRe)#Id1h!E(r?TY;O%f04NRL^iwM+m01R=kP_Q|QY zcu#k@#&4+Iy%Cc(RxTGdoT1XcL2xN}0bYhsYNI2I{JsH9!D+fQFy^GW$}}@o3OOpk zhL0R9vF2S8+g42&Ym0xoN<}<(izf{?6HV?WW~n#~GGvsD(6FKj?}QB%9P`!V#yxE6dwu9LUX0|uiZ z0QWy4Z$|Qe&{Yfgy+ZeOQNYN`fY+qX5RNlPY7VF?wUJQTUM@8FnW{PQAs!HartD>s zuU8r0#y3t_@BZA<--AijB?MhjS?cwmn&qjl<5gmVKX=t*SCBn@9T4{R(Mv*gocCZ; zxPI%4$xtHtRjPDK^qVp&i=^^kcc=2O#v`l6lwBjUUY^Br#RT@O#hP>kn_e1V)}4*w*`>_s`z9)ey!CN3Sog_=4sXRxBd^P0-xk=gM2K zl?LoDS^s$l8lfUTG@W9hjJhK4u7jlMFQ;>Llnk}!r^|}#C1M&Le^-_%WFV!MGE-@% zX4V0WMz5mh7It8LB3~za>zVCUJHkJ5VGwWS9mgo2D=>I3s*~cZZkLt%L3|Ydt zZfE@@kTRMP*)rtcbr7Om#i`RaQt-_hpAsYWk!if|=00&HA`RRfu`6%6T-A@LGo2W8 z??i-9+JX7(`&8_p$CR<>?ds1ksHcSto!bMN$IT$CwPX$?DoyL-sU*X@z2?^ScW^F^ zc%?owQ;c~1e>3|#V(q0R2Hg!^Vh_rPap~&(SA2il+D{T6Z1gv_{YGfAEVbaFLiI?p zuJ)Y2w7U_ODH13e=Em>}GW6oV}|NA~wEJnf1j7NNu{6e~28Sl@iEL(3J=vfR9|) ze9&C#{Vs0aSqWWjbV!*oUlwwQosJSJHiI}=Iu1>au}|6h)1RuH2-fhW6(21`{fPrS zNdBnVXB+ke1_eSr`Qo6X#?_hIv7=ske$rlJcL?k<1?pv#d>S!CZ@I_f#UJu005Ldi=%*aJcmRR0gE-P&tN=Wos-Rcv`X%DH| zoy4~@I{8}oW0*qH2_ssbilqzWJu*nUFg4Jd24?bC;G#fY^@Ztq9Kh@SPgwX z;{F$)21#n=Ie2s918FrIxkUX`#ZAA-bPwzF=o@sDjeFqgSxRb5L<2D-n+7mbN*?h#h^ zNTICXhYo{YmPPaRt+4*AuBz$-FSF^DJFr?5bl=AnK^mErHB728GYIuXrvWxK_C9M5be zk+6#c5s`omt;QijZ0iI8mBgz>d7|x~oWm!ADyd}{?iiH-yc!5^D(bti%(CeJkk4c*Kia495Im;H5Y z9f$^9^C7$c69{R{V`(~#q0P=wceF%`J|!Q3mtDz|5hZ*4Tem}!cpaPjYsnt^MSrxo zf52KCLZ^X3!t0rI=UL-7cxNa|)#!qn8aJGRlNP<)#X5&0f=?T{jeCjoDw7iH9lO#7 z_ybFbK%wnk{G8*pqtx6!CZc*bqXj zdj1r`f(x(~6(<-o7`q=g6|(pR^!P5_Zy9fziJ!#An6CSpK6Q zq(-R6gq!Ot8X_?ax6=9OMiPLm8c~|+jP}y@LwJ-*(+f8P9*?+P4yg9| zB#UV&pElqz>&#@M_6&i?YN{DSLkxi21e@1R$i2#HK8bLDbPD5h(R-;Gz9-62nw;Ij z6G^R2`YkMLd2WV6p)xcLgsQR(Xg0p686F)DyT!7#&QiO5%v6#lMAK}O(S0IG4~)oI zCBp@CgtS8i_g5xP-}{yt_S?cx2&syl*(J^fSHrBy=>HJ41dr#qh5XNQea1^WLPcc` zafW76>Q^gygt637n!JnvTMQId_5sZKF`_&1S-{pQr!sdt!1Nr2=D$q7nM*C3gK2*} zGYwy$&c`1wy1%D81I~hU-wysD&|Nv2`ZqRmqIBlP3sX!g*6BMzZ%FxA=M=a}3pk0M z1m%zishxmL@cmTrfUdlIC26Pe>1woZ~nFk|~=_2?WN*w~8a z!(HWG=sDl~t6_kYll5h^1Mso=P#8#qrSJP`bYqh^VYyZt2djn^?4492N#4JwE#}+aQ?F>El%Ow%kDKjOM%UkO;Z`xQD2 zbk6`Td;kctx*KJ1krp!i-Vh+gQQ!6n2aTSBN*TiOH2z`6+erb=zcPsC0c<5AQO~H0 zZ{604>+s^8mQ+(~VgvYrQLF`^bmGMpG&Pm+zDn?kWkT?CjS`C9B7JbUTjSn$AzvmjEQh^Bb39vjjJ-OUaw^q9E zo}3zMK5| zLh~+t%dV%}^F-k#O3wuhB-WVIVv~JKAhQt>HW~KPy+TMg*@YRBw-EfYr8DNm+&=Rx zjWIZz_ieNxOjZhIFVt|sUXv0yW52DFbIY>IbV); zpe8v~p6E%Nq{hfs!x8iK|lU1Z7(O|KJraitoi^`UfSE^ij~0A(rln3rCrp=hl?+XL1!zk+e-2LkM9yhlcgo0^*ek8I0QidRDR0~Zivz8TU zfRU!L73Ua0zE3%|D`7_EuR|phM&Nuw9hYppld9T^Y*5l03VnMN&-&!-pecg}*ivQ5 zzeEUf0%AB0h~v1Di?Cv|Q0zNJ&W!)PTqtLRhENV~N8ZJUJvT|QP{1zkpZ7c5(NZA@ z8(Pbj5vvjSxBK5;>@2;-JXeF!8QuE7p35qsmGr5fLj;yFGrjhZ>epfn-}>5{UPpF?$=O+s82;sk;EqtX7Ro5MKzZ=ExieeWV;(cCv&(s;~-@|ah? z{`Q7xd$fDDfWO#U1en;xZqEFZ#?2Ifn(djz9l8It1e3JzL`q6W0r0m;m$C#4Gy@}v zg=Bhy&cxJu#+t+O*SE{BZ8uG0x^=!3)9j{~qB z6789WnGhE!-Yp`F47Q%z9$z=qU|?_$gkLU0tTX<$d?3H^0n zb}6pNU*eV+ zZZ#h#B!1P@ug{0 z10M&S0#od0onsqd52976sp(PBU&F-r3b`Fuyi-0ULu#nl8LG9t0XhwDg%15e1j1Jg zG=Nf}B38cun`eXNjg*AwneeDxc|*L%X3HX!EvToAYAS|7CK@)&d|U$jEcbX&4zUWj zD?Wd~f%N^a=~BWRPgoI))gP4)Q zQMq43KaFy!X2b>xAcujtRv!Qqjn01QyGu?EA)L}km3DV}Z&)Z?NfIS|?Cao|QCD>-4*~wtv^CfYj_LVm; zgX{<`u}P7FWfWU7MOW;ER<6@XeMTB8mlDE)te~9EtDGf5U`MhP_{U_OF?dfb|BEb{ z(U^LZtN56lF1FBY=aT|Sf%-Lds%Hfq2}E)w_?T5gyZ`32Pjk|Ec#D{-ENA{u`}>Xo z2}BArg6=-Z_pzOzwPaM*L!#uvg?lD}Eu|6-Oz>RR!_A>x(I5T2d{EG5tP|W~=9EPl zr(#MbqS*z}btEhiF#9cd1za{_uUuLo4MNU8KG^{t*=380(cdb#m$-qJ_EG!~`YQZb zN!ji(ox_R61_)`lsL_pDWR)_KyJD=BT>w!Ax{s||>O&Fsy#|kUdDY9_^1fRBIEeoqer~3C@;1HVIdUwWPZ;g%Wt9f5sD$ldx__KGyq5}^G{u4wG@Mff-q-6 z)m}$B8I%0A4|Uh;UDg}i?W0o)=74(kKqD~PSJi0|pwX+Q!2pL8=H~gGelJPQ&0>$& z!tyr^nOTH9njS820_U`@TBYj;Xhq6>ckp>vI*|2xUI zUH*6rjHCa>mh*nd)1>eir%B(L@uWWx$T~GU{6zmOXnFinpAKlM$Or}364P!!jFcXW zabV2hkI}3pUkT9n?&?~tM%ofvwJMUqg}uBofh)S16~e*gEyUBROVB$9r3oj#ooYW) zg^*sb%^1L|9g2rcACXI;m`n5nb=YbRb-paVrLCK&Cnd~D&+&IiRUrwU6xK!(~EtABaS`zcv?j( z*5~f3FQ}>4AVeWr!IV-=m z(`d#w<5IHBMS1nb-7>|c4gpmj(verg=lW_rYo@$LG?4iehOMljrDF9ZF=@12yZ8ur zBAnJ#S`gKxL_{GIg%}sz_m=gAsjSXC-G9BL!#jLQ=;jGH|0R>Q-s+*2aE@k2S6FT_ zj&S{oQSyV#%0ADoS97|cbsmUe+^52HkKSb|wVw7L&uR`hB3=vzcM7OVA)b)Q)=E5E z<4kJa=$X){qV--s`1A1Au}KQl^>U?meeH>Mxcuw>Itqz}2bVni4W;$94KP)Bxn=5sxhQXSc2j_9Ec52ik8 zI2Ki(wZ=#WL+^)gnKBZ~_r|)8!cn*5i>yqk%0R(C`bUHREH%>ks+akwiYJ@t4_o$(Cj$jo;hM+UY>1ao-)Sfy~THhQX@r=)_P!te+J?5pS2zk#Mprn84?y z#9N@K?Fh|BdI)<&`+?Yl^nTO@E7H1~_+!#?sB@0skSv!^-A>MoU&AA=xAjcBY-1{T z_$)9A#7NnC5S$lIX_nX9^1* zSfN3^Ff!{f^ygnuJPv`N56SDbP%=UO-#rt-iGExUC-Vawnv!#Bauy$JFY)fQ(5%s~ z@}Jv!_ZlJO(&mpGr}$##i0EBe;UGyWj?~jKP>Ia_F??El@rOZ>EB{9C9m6|6RDt2& zPI?wn)hJx!P}AtW+nHvYPIJmtwN(*%c{@ag%Y!j;S>CNf!bi-+pR-j+%1bfcAlJ?; zKCim@+V~<>d55Wm48NnYyDkpxudl{HC3}I}#_?<{*ji%pUB~pV-YUIJ8otKz#iw&_ zA4T&MZomBQuGPIlZ}q%%WQN$iNe9BURfTKCchlYs)~(vI@Zy>W0nrhwhC<}fF)myHd0WU;M%TP81Ifv z?JDzQue)D@01!+m;7H$HT*k)?BFH9#RZzQQ1)qsUju8^(5b!K(Z+5o&b$AR1vde!|z_bgHH~&AL3*j8&UA zxvb3s+Od9lsG9)fp|)_nmv2JI(T3h;hG?F~7?)PP(iFk{pXL4XmGkzt$K)9AXp>o- z1CKUQS39hMDImmP zaQm+2<$X+?*K zs5TU~j$#Cgur&|hrSk~b>OM80`lMSJKQn=fRLCMgQ#>6yP1y;+7;S?Ij&0d8;EX{;AX>hkHp|8;A|NkUka8cl6>5{+he)BnuB9Mv}Sd2)oGH@&qZ#8~s044EU zo9Slb@EyG(>XYd$lUbP_*DfKoR#cm$u>h$vgz!6$bTsySTZ`j!%d9)(O>L7k9-C=Z zSgjjxiRUKPQTP~OWlTs z7L_<%n@Qgn)7B4-{B@Nmv{)03(>#4t(V=Be*Y-WTVK8;(lBKOP6gHqw4SY4uZCC8^ zgW(7IplgzEOx%Ijttj>~65hALU8pqHjjSOGnKlddrbsbBgXHh$V_FhX2_EWz$h0`W zFrFl*%*|nftcLL+(>PZ#!+rFO`040Y1fMNg_yPPWa3lRngCPsdf^5hy*&H(*ol`4) zn&hnWfp3AnoFzrkj#0Vvt4(mvG!dD-yHk#o?a6d9O887HZqUlCK?)=*R1F8XQ|0Iw ztXBLT^Xf03pR=+`Eol`Iv==QZY0r!rYH+3^h{;&}&?cfVSgm-wyI(RZU7u zurO2sjnpmt^qVZre{5z94*bjcOcZpB&0~Pc!Yi{Um>fd|`^Ssa zYRGXEOSsNe>66g_WA#UtuSzdwFPZDcRE%er&-@yMe)$>TwTeC+ z{LSEvr&~MNQU>H}(mu(A0SlWQ3)NC|W!bh&v#8VkrcoVR3%+zVG-cUw7vOx&e{@09 zg2?F7p@V8;;K{)uo#<2@DZ?rCnM?v~=p73_XnqQJ>l%#j(G7NUaHg=qV+T-sHs0z! z*L2||pG`ZXd~MaN`q3Ao;Ow<~TK}x&L#TYWa<#r%!bW!1c;%be(NTs!u(NDajTmv) z`iGaFod0sbs+sT1Z#g_IWAV~aay-nxv!4D*WLdk77#c1gOW&C`BbVVEaw*kdzE;JO zWByR3LT*l$;<-}604*Hz6<;j1-8a@bRki(3Q3Uw`JL6x0G_k<5f66jV)iH${U9GGr zNTAx(|1I1~@!qZB)Q?WR4cCMrAHkke!+_Gdz=E}!=T6nlnNAOo?JERI&}2O{ta$)F z7YjQ#l=zKW&O5|vAWRZgyj~%xY!|3d51Jd$R*Il1-a-_G0=~9*PQR?L&zQKaAAP3> z=#Uh3QmWyj#{=;UL|yMPIJ&F(YDM$?v(gQ{V^*|BES51A8x&1d|uTt-iKWB0a zYdWVZGSFDrj!&fy|8MGr!)gH@fcJgTQ8)#W{CwDrH+LdZ!>PJ4W#1T^;&>wI)~yJmrr$E$b1{g!H1=FkE7#szO`)XSXvL+sUpP&CS>)3`(>9gj$6&@uOpkTM)A zCyBX-b^J0L<`yZe*6FN)v*l z9>4jI1XxG)ZuPOy@HMbs!FYBEoyr+h?OsRtjKbCFB5t8Rv(;bKK%UuuvEVOccMWZd z;^RFc*ACLk!UzmvjMidvS|oO7hoMK)1{!kx?9?$=a8AlbG8-L-x+_4F^|rh}Pk{Eg3?L|H>Kfon@A5nH*X!Lc>h z8MJyL<^{NJ|Ho$f?vdKCi>>Cn-WKR%DnXBuA0A5JtgU138t!=tAs}kBHz6wWdplN` z5e;n?w>O%NAySxl*P^pQLq}+!1|eI~I=+y26qXso(1KICH^jmrG{%Z|WXue3&7TN2TS3f#Ni$#_EEh~}=HKDpOY(P|_Uwwc!DDe;;`>Dm;nFaQt|##{og zm5kfLV+e57GP(+|4LIQ5IE5b7HzPmw!WsjT&JFpwvAQqK7rs0{$Rb&6Xxss>mc{H_ zsA&QjKmc7HHF&r@cXv>V+2gy(Yttp5^3RA_+k|Vsz;(dcEPrL97>Zl#iU5QVLE(9h z#C833l6;4J{=?U{-+hcQG6kb02b1Qc`UhGl^6|ye*yo!HE`krw4=vRh)f#!kIQscW z!`$*-^8lxEb?=yHb5UT%g0CMRB&Cjl(N;QirQ?lHvVuPHC}Scw43 zcU`VJ37g^+LYFcH3yY*yk;~|bQX#E!wuz1A$v~;l76Tk-yRfgI89>%&7w33`KBZp* zl%DH|0i`IM4r0i`p<=!1Ii*;z9?KhLx~5JGB^p6WnS}oMxIUb&X#xZQp{gvukGj=5 zePAB>ja?2IJ4r8#;L49|;QHGCe$PQwdQ4zWhoH}F8(L-c$y%AB6zDfcnynoMQHCto zdve@d-?VWcf7Dd+>d0EN>$lr=WSV)417x+LF_eyvqMq*yw$y4K+8`SFf7u1ln7jH2 zYTjH+Q|!r2BEFHfxEU-l=JN}J=b>*&D`BZL7Mfzjf3AwXYsLf!Hm=QhG0|@S&C5o9 zZe)yX#fC=6F8vLl#N8^w#;6fkh{49-nM1kB+}w$`BJ}yL#MWyA$pPM_WhbGvj3awAu?rpRfpgC_4gtyJ3sZ^2)2 zheUmuqo%1tup?aJ5Ws?vrO}Ir*F4Z&V2G$5$9r1`4*e+7-6bt+kl&R*(Wmd>k$LS4cMsZ7GQsRbte@Qeu+-dyxIA zaI&Fp84=;gPq2Z8EmRC7vr=oXNZ_DY)Sm{N4b*JX*$78&yz$(sk755|lM)l#Ie9Ps zbz-Hd-vNC9x`EeWk0 zU`&zx*|XrO>L`Li-s!nFAwBRIe#rSJoK(P~y9r-0zvytq@hNC$`8GxvLPc|S_Ztst z5XRt34Eqff6s=ylX@ML~;Q{K|?r_A|prmastkq`sl|Q5d!xog!cASjL3ELamKgCXX zN~6wPUIhEao$e%!A^b~DqcaSGoJLxW&O)wW?z>9xeW3Hu?1FrJ6m4iC7y(A@Z*@!b z@Rt~IJB(HrA2JxUHnl__)C!(r;o~Bnd)FTqf^!8?-~CPhD}3}x&71Vnt3Ewd{&R4B zX;m?~HI5+?eIzw9(FerET(`C#v~7w1zTcDZ_LP?wO1O!jU8JT=kWUVd3*{?T^`~Mt zh3IKpv?{`$fu{W#{oqsL>In&u00~$~+667A{HOoDvOg+2ik{X?H~s6u@FT~yf?e}n z4{4w1@GU2vzjDQ+)f_IL{p7(_B!h&7N9alUwOv#M@$=`nA|`DQ><`6$9yT(T9yhXQ zPjR5_!@N($F%m7f)?dt#04@I1e`eclh7>5$Id67*a&RVI7M$zZ&i^g z4QK6#47C+)UeL%O8#wRJId`GPhe2rGhJpTb&mHF(mW4%M9LKw@#(2mlR8&!g8NKZq^b}sAz!sSZ_l>TqeRuH7 zLqpZYK?jUnpX4ai5~hN!#9EjGzlKx6)kNG2Pm|{IM=hifMc~*}J9vq>H1=f1*u%9A zuir)ek{_$4c433_t+rX(c&lV33M&KYr*Oj26`^%HtS~zatxH-Or?0bcSR*qrxBl%( zAH3cWPT=0ut{CXoiY;6L^#fA)Ln!^RoN4~!QJ`#k+$gMYf|fUvc29WM8RB=HZz*9X z_nK~#wQ@?rgg*8!Qf#|YiUeTXc_;~yOuvdFRe#});NVf^np#*VttSaV#H1F<$g)YA7;{?*(2?DlR6kaZv^lR}4A1b*c+;A`6{cu*Z4DBC?Y03{ccTID@p0+tsvNtd}EZVHr)ZICE0=|sa3AQ+jBTnYb{ zLx1$`1y*1A;x~k)*(o!pF|>Q_(L92?q3M^OK<9#IA`nSa0M|!TuT^H0olaV&5KjG* zz#u!D^-RQ-fYIHdS7suI-h06xTqIf=RfGnuz{GPxTgY>(*Xv|FsHqh42S4~G?p()W zVED^Zn99;F))hjHpWNG+yC)}W5b-vr_1(Vu0hd`2!`7vOxs+G~3rAp}A9RM$=w@^f zp)`?t0N_i`J~v#bn&fM9d6ZaW>Hg>3e!`Nv9T4Sht4Wou#UOCD6L?UMOwG0#bM>~I z^LpdPqb0qWTe*mY*pFlcOpsyrCs!J~6_#53G(5Ty<|~3tne;sa^sHPB$&4FO0wA}Y zI6=B0Gf%!S^=(|N(-x;tktctfvR2`;EB#q)qXc?DCU|EFk6A2{O=x+1A9GSZpuiH& zJFahoGkbXbns2EL2+6dE!svFtJ%|)}N7q9=!!bvs_<^F%w!`@Kj7n?heQ1Uj@!Xh> zObfUrt1KOq`Bb{?yWRy$8K#+(ov|R=*WiDF5gUNW>i{!X?fH&YjbUaqW{nS>dmgAT zXQ=ScRg77?mWIXzzw5}(CjTWYscM-7Vj&-#cD^s=3!!*k4eDjJB){YR`6;=|V zs3uHMg4uto|F%K)`p+!}l}m)$C?_)9Zz_(IYjgRwTO_}bwJu@+${=}ceQ-0!?(W?@ z_N^C!sd&+zK{;6eVEmt15Mh}#5eEoHOhb3&A36?&xnR$N$o_3o0V#ouev^8k#M`T? z$~j^b+z-&A?l=`i`FgoSX(QgOsfIWuD;K2a2`ElJykHDcNT=c)KbcPTGgJ;&C4q~1>bs-Z0hd=eZ8=X2Fd(mx)O%ksF&d;mQ_!oO%EZhaVh^coYu z;7r<1csda)`!xvp;YF{^fA>%}P}k`5ORxjL4LsrnTy!49;&e|TZ~_lC*Diu)+QwV` z+13$Nwug<0E%NNd0$2!a(1FrlU_b8)B++#uMKr+hGJ|c)4!5F^lzc8q z2F-rymv{$#HUba8@umUxNu|BOG zG#`QLk6bx<-q!e;&4>I&+4koaK8(){?g_63Iz^Esdc@3IMnbb8SxPg?p1Chtd@;W* zM+r5mDc@X8#DWK=NNZX^Gnw^|t>B?QDJ&&`$rqZdEmr{%()J;Uc(B&%@+7-NSHzijq?`-Js;Z>z+tnvqIPb!TMS4FDt? zYfn+tTkBP((nvAH>^(6!`uc2VL`OBjOb}u0%acUrcaaLqvf&QTl5@A`*EWH10{M5_ z`%B6sw{R2Duo%fOEZ4Xl4F74sry2YtTBnkU-5aEL&?}WM8@cI_f)(`-i~CZ2(qwi zFnh0o8Z;3$zSQ?UMyyrftccaR?-*~fEA@Jz!%bTcmb%s))i-!pgw`;%nFSLae6?av zo-F5L!EyV%FL=)JP+5QFHRcI43=}w)_pka?u5&~t-}g9-+@?KI|F6lX+kS!7!y3rF zUIhu9WpAeD4Hg}xA+&}3#OismKn}@`sJ9owf-{*Q6GtUa>Qu&{l*Aw7N+0F<0~TEOuxK9Y z)R3`)6tWR?mr<7*3M_q*#hb?Xoo{TR$c&srT_Q{s8XX7-Pq~aNbBuZ% zZZKFjUW9*YE?QUI`Ru)cmrmD9a4Ts#aDdI|{`i^Q3)Y2o&xwD&XjdC+RNWOM{l z9AIKU`?B4FR(#$}t;3^DiSoz$NIan9D*D7%dY23hA-qM?<h1_5NYG;o%IoE#+VsSi&QKP^>*uJq5(6f*NSutXG%=>*M@WPD z3zK#EpQsMMU}K|Mw*jB5O?0o`RxL`*DH08fM@$)UW4B$ni0zn zrAGIjC`Z=cQeFt)!I6$&qefYetnaL*L4?7I>?B4lEydekAl8?G^V}eW3b!_9wdP^V zeYO@A;#+F6Wf2ocd<1k~HX=YBE7R^x8euULOZtEuFqz%)n@;{r*zpE;vsO)KVlF>x z7rt}?VV>Yf4M=!k%AA1yxbkgmj)Sa}rusvpE|l;Rf5->U-clJmGRvZnaTGGT3rh0*IEFNA_`Cqk=Gew>6h`}0-CDu z3&&15iO>lE6d9$-9t??5d|Pb$Q@a#T|H$uch6HSVl$K;{oKtl~%AlAt52_7j9{5Va z6)dV0@qBcThq$N|+nB&_04`Jf$0|A14K9c%J&5X^Q_bTys? zpZ1AiC*5TfTjdRj7!k2RUoN8O>LJRe<2`}x<5n~1k^*)8 z{hiNEeq@vZT=gGeh*sSc*`H}3QegkL#wN4gkqtU@gvhEfEju=U6DYB**?cZ(g_7JW zRvzBSvEQf%r@FXLmf)x+(4l4^(D!A(bkz2bv1DBL)XtgrX!IQ~RBalp9RTXcc%S%G zwy|=_31*eKFMP8kMZ_@S+mxm`_NSYg!whxXP$lM&V|Q1^)-}84J@3bjk(Mqv=EcN> zA2VFb9(7dk%)YWAw(5zKH`pkOfl$#%i}UF(m3{|Ttv5N$HmH}@Ktdi@C8Gt2{K+ki zOM2Zk0zV8YCJ?&#c|pTswm9vnu30nAUF?{)VjZ;omW z-v!cCJu`QI@~DJA>;%yJIGM4!Fgsa30O6^F@QL`Y5l4&XVL=f3E&EN8L=V7~ZpfXu zNK)F}27j5U)3y5H-p&YmlOD_9C_d2@g(SC4`o|Ur1R<*b+aCn_yr)Q38mugqs2$1{ zXQ?p_-p(?9Di+${e<#`;J~d^v>#7YSIAwDL8VR{m+Ha)kEyxm90T2uhqqf9|pDdrl zG?^d70mh-*xO7@uaMSyywP8HbLj^$T105lZ5}D&;g~Q&Igvfs$9jMqY-HUKG&MStn zOpt~Ib^r|qLnJSh_Y{-NY+z41f7pu9&z-P2UUB5HfGKV)z7CwvZ+_++Oe-89?=RW9 zQ?wi=f3gSyFdh74Vgq;GB9=KgVkr^gi;^UWv;J9I+wi$SHR#?O3__v8n@|fvzL6>QxaiV&;r`~cE zTK#?s4|H)TD!Edd2WsH**O~3ur#|~>O!q++2upi@;G5dFOExMu10Z zmI)}_An_rRueI|Rwkv7n8fT@J7(UF8F;dMXraR@PURKHTYbE9Is5e!#X@W6`XT2{n zVPJs3rn;|?QQssy)uz@FmqNNUGWVh@MqnozfxQg1vVvY?WGu76%nd3nFl0O$>61xx z6^C$XYn)?fR9MA`G-B_wysWG=16J{A`pgIva?q=0lJs1+N-S z#>S3t7PbIA+qAO%e6}GLuDLtDE=vEEWOXmC;0BxLV5O1ok|-9iVbjRbzAr-o%Jt-8 z!$Eneb$5ry$P~>=<@8OQl+2l}9s`J}aV@cea8zU5(ld0ZlXVqUr(5z?EtJ`Sd~&cG zpT!DvkF^N#TxJJO{$6dZfADo^PzU=%bcoKd+4{n@5HsUZf)PZdCyWgK)1@!eo2Vxw ztd2fcM5%>kC17qr@a?EjxnC^`@^vz2qvJ2WW`LdLW^06-Y@mf-ydVlptdS@E{2E-|li4srNJRbCxkqNtf=U-a=$6B6a zC)yT~skK=8-ZqTW1<8giX)`u{W|$KLSZr5HHc0gvs|&y_bQJxYJdlM+b`5N}^bRr2 zbKq3wWYiL<08!EKq?`hwL%!Kl5IV+Jxr8{Fxx}IuEY9!ISEniMEilP1mb}Uf={^2S zUQ|b@a|NQ@<7>Fy?jMnQtgIo9j@%u1$3Gd419hCMt!|%gvwa8q#|^7&ZWVC~%Toa| zF}zrXw27OyqeB=e$ltL?`kNYsl%K=Yle1$_7)HW)e5*X3odKS@J)7xnP!z?I8tvP+ zZ=sX&#A*AQ_Tvx+4J6Aqwf+BkyTxZe5%Gb>$SwBXZgOK8(|!Gy?W$hPyF$+ebK8=B zsf}IKLS){l%QkitU4g?dHlnGnOg}Amjm?dyPoQ?0v799VWvsc=zg5nE!#uv-kM)gs z-X=N622P0PtA|KQWPsZ`(@c}YGRYi2LgGX^G|07TvEVLUKDmV)j#| zMP#_h($vXUb`-|XF)}1k-L0sjJWw=1c41Nq_nZ^W=86u6DcS^kuvQXj+~pwqEXK0Z zO~-AIqKq@OYQ@#aW;L$n1sB#O82-40*H~R)=@OduUIgfV?_vYZ#D=2E9z?jG#(F5xsVjP|9{HT2iGTiLm#j*Ax=L~B`1nq4)Krs3&=NXC z)ah*F?_kv+%gB22{}Fana53t#?%06Y<(Y^-E^B1O{IupFTb!g|OT4VvK$`bzAtK~Q znsDoEl4J}wVo_g<=WpR-eW`1@vc_~qAI5v^Zx)U4cT6oTTT?*^@8Rv@r`zy)qloPO zfP49uVHNv@)$u^|NjnBoaNBxoIwB58lbLYc=h*6`#=tOo9O)UmI<1jCSFQfbmLz3l zu0NPbo6m%_zvRqt`y`+OSA;eQOQp|aLl_sdYh+gp8`o`}B(kdTJ>`ONuNwx6lKq0V zX1={jhA6@R8z#d{02&qeQ9k;JC4I;~n=+meMq1kC?%WGlRnKwL=N0UkEqkCK-5Rc} zs0u0i0dE%PL~3Z)9D^ydu|NM#$kxHP}iqDm96=4FRj9o7}g6X(-a$ z)vf~{qvc`b*DFl8 zSmIFB&-*K?5hHJL2O*Lv5*EAJ)14Iz5Z^B*PQO+EG964NPJevPjPJMJrqvw;m6Eyd zFKmE#tDjZrIe?t>;0K7vu27-VB$D&tcX$JmU#U2{*yi0!yO26#j;wm^BD3_O7K;o=aB1^=nfRH z60&88YBRdPQv)JODqP;KmO^-TQyl^}_yZ&$zD$r^;46a|siwwW0P;|OSzVC&*$OwTCGr{wF;%GWahgmOqu%rfEua zP*2T+wa@#?b0*sY;l7l)FGB9?c zm~bu`RBnuZ71*eRRr4=B(?H+dUhWQnn+7#o}j-Qa6__ zzDv*7$b9a6hoK+aw-0k;L|snI&M$D>nDn1qJ+Vf7M`z4Dr68Oz3F%4QUT#-9)6v#+ z7Xw1!Jy@X89Ogbh*OE6=T+Vl00}#HMVMKH@z+;K^CR_RuNw+d%-f&{6W!`Kn)D5KB zMlyiWufgC=u#X+ML1d(l&v~i5bA-NO64YHksY0!~i?9VuWpcY@QmK@RND;9PPq7bKG9j0RZ-3qO zhuY;}_RX;tdd}h(o+SFha5t>7(n%HRj^BmQNEWw<27v4RJBYSZRmyHq-BQf~~Gj!yl)Ybo9Ns%;ZCiiImq8&O4 za}CCPh9{QKHd`+A1*iP^FrI&rn6AFg!l|zjA@#gvCkF(FXL@9LwgSAaE>47el~>9hK)7+lz%8-MJSF*F5}@E zaIU7yxUfn*2%o9k&pwAVDR;5Q7Ym<+P1Ka21Bfp0D9leGSXSU`)Y>*rLDqqLM?Dpw zvon%-vzdv$BP-6}x&}rm{Z67Tspq@#GN3(E7n*Zp{a&ponLGUN0uHEgSA%wqpay@Vn z;c^!S40VRi({1lSI^-kjqt_rK7@K|n7=kiA{QPN{fnDI%Bc|kLEE;W)%L}}fJNaqE z-YQNh0;(X^+AuBFTe-#@%Z!%oYxE{bb+$Fh|91acbpOCxp~&+5*X<&pSL4c4`j6T) z{O`k~=E@x}oh;e$gSLtmZ2x2ghYfL6r%zAu(=bl~60&|c3B>U5o9^1WQ2qRqh%w!u zHA-tA4)fAzy;YPV6anM9EV5T!R$bd*kaF)e0sFR2`HS4a#54BGFb6Mu!JR||dG-?H zdEh2*fiz(9uOp|M-{z`M-TUhbFU6G14Na9y(-~w2q+CQw7H*%;<^TCXa-XeSYDnLb znj&CDLv`*7P7dDg2}Hcp_(T$pm|am;O{A-gI3>4>Q-ioI8b5c@?E9a9Oq{3XA%% zY7EQ73qgHtaA~Y9?&0F9hYb_dMNpUWV{Ph&eS7tr0i&v#G<7gI(m$t`Qt@*14?|G9 zMO|&QpebPbt?*<$8`sS+D{cS*H2;zqm|c2@S?v5W2*C{oOjxl6gn_J(LZu7DYY~=I z(_~=%{R*G7dqb!&5c8x3ULTbQOrh@eWi0~ptVPfz!0fk=(iu9ZEoD2)zo5Xd5iYxj z7R%J(q<|9(5MxoV(0%<8WE$zey!rpTh!B&GwdrIoXCqRv+zh)qK^({(*Giuez@m#s z|7-#5>MFYEM-R7qS_pSCF5&47IRJ|?i0K6#p3_^4^3@Y1WFO5E*UDlJAQwK`U|P6c zEsMSe{W!0ZoF{EP7`sIreC=2a!7x2t8u7lNu;4e>C8(H|}3TCmITg zE}VCyFckF$|GiL?Qk7^`f0DnHTAR(oSgtFhZ~V@x*n{00=4GxNSlB~u_nRQRXqNjg zj&K#yuNAW=Lv`$w0pEp~Q8YW(!g#P4dboN5jw}WWnDI``l0$yZ4tZXGskKF&dZxPw=BHYk=I`l_k0O(50erxjL}z2{6I|Or?3jW+=gRbg%R_c-jqqB{8N&$o>ErUwL_0>ckvSoNcfxq{yW#m z#pT-J_F-q{9sL$%FzsVvymqDXFHF=1qZgl_b9!fjU5zHlJ<3mJgL+V~=2gv3CLXbn z@QCafb%xh`S=dT6v|eNaufVOYUhnC5p99GViy!!h&&B;0hSh9l7X>BKXrZ z#1n|tv#kKE3COGr88#qz4-D)zCL^)d^S-svqxJ4tG=0ieZNfPP_#P&l-l{)&pkG0C>{^!DRcvqY}w zRn=WZbooP-bI~6$uY(|o z?gqMM$fCc0<&MT~O)-R&3{Mh>FX1&;b!^c3aWK=#wqkAZDtk1KLg4J~8gvpA7l6ao zaCG^{Zt6ypG78pc_9N(jJxEWcBtI%ewao%D1>v`>^GzyWtU2QCFazBYBhUPYR7#*L z2~$UjX@#S~JX=kq$-I{1S>GAq8{TS-fm%W@MO640`12f4i3V*_;r=*{ja_|GVg*$F zbq%MmREiU5=F*K_WAulsY#_=ATEhc8A3^Eh{FgmU@(neFq$f zK0h`;6C|v)x{icsg3bX2nwq!_j3?ypFNPOg7xbg>F6%G&`}$)+R_S&3aBVkRq-XT% zJCT$pw6yzrYWexBzbmgm@U(;dK(fUoA&?L#=lC-!zGPV36m3)b!)8NATVTiYB`$%B zkf6~h3&xXRV!z;B`QfVuo*44I(2sP0+vh*kFUrQ6ew zka_zz+e+A`C_hR+XLQmuF)*{TykLPl1L-NmVkZ=WpvvLq&N9EfxSZ=3fkO+y7{%Yp zAz|~^m5Qv00jZa9Q>~9!-Hk16Z1X{tMkD18Bx1-YI$mDNsyUP$8SlM5funiU^#e2n z2m6U{jp8L3blw|3?RSZ1M?U!GLO5db6-0Cqc-0JGVtH_=42zvEzjUk6BPp|RbkE3)O3TvT1GCHYMjy;*Oa|4d3@9l;-Bf3c8G=>`i zPN#l8cN`;PA)Y+c^{VYlW3~riC#3Ed|GeV7fcTtHZ8&b0o~~!k?6t_VpYl0-a1WV2 zFTGG}F?u{u!>wxHJj`X(Cg}{qrCYWYRGYdxT6#h8ZFKZP1?#bs3y6CnW&Ma)L|kL7 zz3>E7iz%V&QBnU`>T~*c)9)e8;rlk)GEt{J0%(4IN7A1ccxLcY=LT5`w{p%UquMJb zw!(+kHCnWGBOE=sur9S$c*o?;y?3mjkQ^L)!66|~jl?+TRJm|ST3sI;pO17x z^3De18A_;!Y^g=2R}Y0s$@BGj=v34Ky&#U(@wRxP;#W`p1_p&^Q@pJTJn^8wOUD&x zG-E5mhAX%6DLSCr%-VoqzU`4jqgmxJ7B9wOl~5gTEg>~!LX&0mn(h;PUWSYlb-oT@ z?{v+1klJ`FR!~ls*guX)GDx@9`@~g>cgYf1N$jRXDI`fg-w- zZMA&DtyzetIOgwnawnz!K=D2$VlI}!rfa4iTZZP=e4S+*W19y(4l)G%>)~^rE}zaS zUC)M@yU7@BD#73{nV^YUA@KDjvBfy??&qrPrLxzYPDx%v;GIEP^A?Dyfp!Rz@k^Bgen^s~U>>{}v3tS|A@t6%{QNNDF8jBIw*!fc%zgN#lW{MAXErY1OHTdsoW0nVU)` z88|**zxx6U+;SRnzx=)cUF{T#Qb)9@jZV#bXK6)ivtYWJctQqYq|GdraKmj60D?|f zu^%Z0Kg_d_O%5&4E!bgsTH#Ds=6<*AH%MR0&NAke_T(vNn8rzw8)Ao1?!iKgAuP=3 z58`t|F(EYAeD}^=G#mq=Ed;k+x0P9;)?yf@Iw`!tLM;nL2i*IpU%t|~Yw^a_meT+8 zji@mX+&J!Q=tncE*+dRZ7&`kwOxUgA(}1Qz@M7HKf~(2pSr92IgrJ-oF5Jwc4xXvA zj&W@sUyRU`NzVT)u?%}$yN-6->&*iC`pK(dPq+*D}oj~zy*Lcp2 zSlE~{MUN!e(Q+_nAiik%aS@{WM`IU4vFxs7+pHqbYrfhz<4x{YG1CA>sG5@oQtr;^ zLY3xB>%!!!KxD$qjM++DJcvlesM2<3vm`~vl}Ue^t}?#rmH7s@_IPC)@Aa}?Tfl-K zBvj@o?NgOhkPs;~2yuM7rSxBWTU2C{35F5)X_Mdxf~WbxA#ZWMB~7BGLikZpLV3^` ziAjTptX73++0Sv2SRJpb(*Vzb3i>b~;Q5j_4D)gYcuTwSr@L@+0y1 zTY33>LUcSE#RMEFRu4B`x+FM3gwqj^@}`|ns(fAMT#OYdzkHo6zXNu)nJs(awC|8J z+M~gC#f~T~0x1tAOxOyuAvBw~-r0U}mW|{YC{3Al&O3)XH zdVok0Q&tlouMxb#SN3CoG-65WFI8$X9|Id>rKJ*KScVsd0N0U!r$vgH^ro%S0EUjw zyNNo_d@Wd#%$?F`V-Sen5?5Jwn-voYc>>6+HA~3c`5(cA!r9e>e}8?G@((v)_+oJ2 zuztqm!lc~*N2#stwSr3EStsYjq-WS~ijN%5pV=rRzJf5D8M17WBu6bR@K!0AaEQ>n z+{t`fQtgYbzZv}CtVCcc{VEGVp+SBsK#AZGnxY_!s;0Lw*k%o)>OkcPV;v`@1!fj6 zUQ;Zpy>N@7=XZ4WwM#I!bQ&kvTajkL1|b-zwn7-P(*kPrTQ>0j!^2MoVx-osoY2)P zP+=Pm|2BpUC~<3lRg?uvpg7x_ClwjM-X#rWc(kW4j96aL*4WCMN`r#QenzNs%lN(d zRB@{}Cpw1`xfWBFffvI2n1YYxsR;Q z;%x-g0C=(bY1WOFd;lWJ`4EgGy)TrXiEevv=eYmK5K-DRCfr zmxUB%BxkyYs4nZ7^xbM!e`e8aHNFJ?dew*Pzn_}KFk+*)x*ZOHE=mU?|7iMj8@b8R zVqXGqS9oF&zx@i-jn_q@VwBpM1T$nE7=IW~z+M#CYOk@06oE_Eu)U!7Le|VccuF`; zdvr!QZ%lS;OOZ2Ye76xc1%4IwN742^|2eKTFxCkS)L&wy0OKXO3O zM}Ws?@w>!UtE#p4$VU0qxFlGdY6!8jiKLt@eDZn5@#lPPJMJ;XS4bN==UNO3&@&+DzuB=a#L`ilnH7Gswn!od{9DbS>-hUqtRsXGu7zNv5@RA{{DLT`et?Uc_!!^ z(LcCu?uSM#urjsB!>;3?P%7qcZh9{h9wp!iW+%Kf@PxCwsw+E1N0@n2-(MK|4)7P{ z9wj(z;GMMMqz;&5t6^)A^L^yUj+A4A&cE6t>34M7&sO9Q1i*1pcGRttpa`c7qT1fB z1HNEgR<`DW@XZd@MWUJ%qLoFmOspz)6%hNu0C<_{Y1!4_GRxydN&kKmx(ZPylmoMG}#`2*dRy7leSxG~U z^DH6$>XO2C0jCP!?`AfxU}zXb)9A!rL7i%G!HKLUKL8eFhHV_bU5D3NonDu4_(C;M z-XDEZS_kl}&kDBth#Lum)wRHEwHvoYdLu94)OQ))P>}`-*sx^nJ-|k72^;hntjmtC zqZ&+A@Xdmg)`yK{qcEq32f=G2ub+_#-Z->QlF30ni&3~_3Jj%}v7=enaF7JZTQh`5 z)%Qau0O+FTme(?;h@o5^5YAvh~^Df^F%tXIth>@y_W>0rmmWbgMA1(iHELgeol zk~Hr(kgCy+I$^>5o~vN8J8DFeg-D7^ilxLA#*-~uE+(7z7d#&sF_}JFOdp9VfXUh? z7Z+DSe4{xmGCVS+`f3 z-hlw9aHfA~svaFLAoL1#j)hyjd(=h_+u|_XTP9Wie}wt8otRtO5rT9ec{0M@C}XP@(0m7VjPhWa{A454KQ&!_FqTq$3f7&q9cf|to{Q1r%L{>qE4mWc&_t7 z8U6N+;O*}|OlJN)ZX%WGhhq|;p)Se4IWEX7YYd04@-E(T?;IiXjN|MeskflnT<0Y~ zs%(24rp>mmPxm?H*6a&|N0db!@tp_fI(abz*>kAi)p2-?RSQK{bUn$ClstJ&{a<0=xd3QB9 zF@i6D&S4fWWcG=rK}ZN)uQ!1RS<}Cyfgj7>6|;{ZO(l(j4Ud$r92r9i+#o!&*Eyr# zQzN+as+oGF;&sj{OX7TS{96)DdBzyHUUCFuE`RSD(8*7)6bVNaJcH2(Y|(C^flRn> zSLma!d`OS2_q6MIc8ztNfM^T|dU^J>EqSH3h^)w*!o#m(iev78Alf8LQm=-T=`}S| zXk4wDJU&euR=zS?sjLPFzbTcD$k(DqPjM1wsi1_!WQT*zg zTX$qY$Sp>tj`W$WOlFIN#y)xUA^KuUr6qZRmWw7gZ)z9YHa0YkU!;zaH5=ok;ijJ< z#?H<+Rh6jVkrdc>J>Jy1-;CH`pX&KM2MJ?Ot(YmNA_0ZQTb%$@fIU4xB?uhcXTO$F1xr@k{$x;#WBP z1ee!o;AOY9?WOW)g`pv?6tK26e^=LlIE7s(?OC9yZ_yWL?_S^j`gt;zR%fP|Tnj}? zl&9I4VXHdQpdjx5)fZBpA%;yu^zsEXReMqG z5Fd*tCp272>A%}dIB2ErYXy#ltFM^Cng!9I9D&q@mh~nloD^eQU|Mz%@sLiR8sl+m zj1e#3<_oc*rZobUN-**PLL?32v)k&Bl2`3K3Rz@^W(~jJq*Qvp4*WUr+KZTG$6Y7H zP7UeD6kU{eI687M<~-@pODz5>2w_|GZJV2({|gV#(Q@2V0dDImIJ0W$n#a7P?q!im zlJ8Qpw{8ajss#mN2_=vZG8jKpGI~WHv<~Y<;Dx2efL^m#K(RY+j@9D3S2y`A9pInv zz9rZQ2&n^Hf^kFlE_#m9OsrMa3NLAxb?{{@^YegsCZYQfBux+`NsC7+lx{#37)NDd z7-qd?ecIZOlLyH~kqO{%?LQFn-5Txil3p2`yQMzljfSy~Qr;CTLH5PiXc0&15cdlm z>Oz-I$!<(13w|Vvzf(M#@%CR`!tv=J-U51Y;J6I@{Y6bBtMR;sgP*~w(f@H+adr|Z zwHFQl0qaWA)L(&&4F9;0@HG1!sSP4ShT8YWv^ARLCRiU9Ea>b|=>Y?u;fybRP>hbL zjmOVpn41s6_kTRY!xY{;Zn&vT7_gmS{VCvy?r783NV4t~hA@<)RW-yI!1DZ7S&^K1 zAlDHiY=qY(<%B^_U|pn?^~8Zk`y9HuL*n7Q0^3feIfe!q_m6Of0ov55FOf{(t(ST4 z71KAI!Qf9r1O~K(G8FI3R4IX z{BsQ^h67z9fOiH9W~Z%h)y=mIk@j?;VC0)QUD##l5;O7eF_yE)Szfz|x(B9@H&AP9 z6gfdv0(1Qe-><5!12S`IobTp2{;cCV#GgElmGB$@Z5?YJ$^$$TQzJC_!k>LBcM?T> z0&H?pI2%2sSR?^#y;hk_j<72!1tC0SQy{D|;4wb*Z_S0ijwXlKVxzDJQ}OwZbiY6O za98|$)ysG>u7@C7F?>DEKR%IjV0}JwrDBY!I~6ZbKS_=fEJU=t)Cc>Pc_VnEf`aG6 zp+x#y?xX(K@opnX!nW;Cy1ek1Hgip5=#8lOMFHH!u7)|V!sA&SS^ultYPqXTPxamL zMEi4rrf`RaSv50`D%5P`#h8&?k=s1FXPrn{cQvevFN(<(ck!4K5h+#MTow+4u<()# zx0r-juam5X7F;L~jDf^5$_&qqzF@p2O#l4;{3bNKJTm$@Sgz$GbHSh2=tQw3X0LxE z>GcCN9TsU~+Uc|15phHI*kPpr(Gb(+&!|Sz+RLYsYT0K??MWWaXFxs8v=lEy!bE58 z*gVWK4r*z5;~;`r0ibuEE5y$ z{cxENGbC1&=9PQ0Z?MrcV1(6#F{nyDGIQ4VZiEC_j24(hHh890_+E1Po&OY7jyar4 z9bI4+N(zbYLu7l&B}9AZG@+-=8?b&!8R{Hw5^fdkuOZ5B(d)rqCc_^8ZH_O&OIKAb z_qeBC(ir?buQUTRqi`+l3JCIxSHa&QA6}?aX7SaWL3>d`6kYOwO`YdHsUjAP$S0uA zp84{qqsjEe^Fu!kAAQfq7cVm^csdf5&{?zuIYI=W>Yjv_M|HkOA0Q8B;1WsC;|?pF zoqu_&AIavpOC*d@=tXqKkQ@jPoy?vdsz4JFMTa7?)j>4}MXT^j|FW+h5cYSvJdUJD z)OVLn?7r0yE`e)FsNqNjR29R%lsKfo$)tAX{Z5`8HB4IhpUaN2qRhO4WVRA%%70%l z(dgowM!_vC=k#JeuIsY}?eJjm=X#@`Qpg(;Z&@@V6|S7V4`Y(|ba`2FS@ARy~Zsp!X>G(pdPoY%4!B=07)vY~!F& zByv`Q%xH7?$mhl3$`aBhf;PW(QVaz>azJ*AS9=mtE6%Mie*&j1=*xT))a*lWUB0s2!sU%gHF)w!Or`WuOF#da?uWBN>`x%59I0 z;@iL%vqFq^!KUpNXj$7C*^qtFmi{q6sls-~7pb-b^SyilEJb1$FIGA2PtvoG;O@EB z^A*o$h#-^tjO|zT8Y!c5a0bY*s#x~H?+#Bxa)5Q^_B4Db)$T2}q>$?ISlq6=P58Xc z-ObTXcZ)0M2?6&$K2W}wg&EflK9DahE{5jCkoDM)mTt|<~bLE*Rfq1!&J=3ziZs8`zSoovDSsP+i6=XpCz zO|*SV6i19o0m1mVIxg{sKMzi;YiIG}liR-Zt{+^4yS59EcN)!2)lwM9vCrB4q7@qU zGBK~>J6QALOd8DQMMMr1q_F?=fs|IoPB8e_1+S=3f1@+N&uZ}FeGHSr^z>}HA#Wvm zp`{?j^aq2NWC&0S1g+v%Z2aLRo_UufCVRKx+s9b_mQ26t#jh2iT|KSDOLH^-Vf;O)9n0eg0zj9S zTZ6$w21844{VZ1-gI_(&f~Sd|A(6U4_SEogi#1Kx9E>2uX*u7#awLnLGac6hWGCc50)aG6)0DGx$^{8~Twfzw_&w}NJ+vR}jO?g*luK_b z1Y~vI%h*A7y*efT+-Icptx{6HVJiJLGdzFXY#0LK%GTDpaGqy~V^ZO}fxH9>lAY2w z(M=oHvAXN0dCl4$yjink!Vk06^Z{%hXN++ix2fMCV9v)71;adyjs;y^T>%4|)$O{qgSSP%>vR!gJ+M%81TD%k>xZI?nht2QQX{i!Z)det=Tt5#7h0V^Bj zh)WXKg~l)M5#1C{ZXKjh+#|SQ`^SApV31P>o1?SYI*d2PbkOy6^TJO!i?K@4Y^WDX zHsH`E&dk^qsy~k#d;nSE>fP*f_ry8OF8jX`j+4wgDmoHf9l>s>#qy~bVi#ti-S#&t z6qn$&L^$sN`Vxj9`&V482mv>$W0!GTSdH}QS&LF*LZCaV$B;H1lrdNyGpUjG^9{|$ z_vB_k3dPQ0TmO|p_ZV{Xz9{|hz=s~p-F74XVMYKc#yldVy72bG#y)^yBR?n<*5)AW z!e^wDC|aL5}0y$%?-~`M~36&;(R9lQYDzk!zNPs{aC88JSTnI#Vz3@no?IuGNYX z=tM)uV$wUDbu&=xI?J{HHfA9-IsN%9WXlt{QAF-2?;FX4Pvs3>l?N=l+P;g=nt0rTr8^;LmQ>Smg~b;H zDpq)a4iwwEWXhnd+DNoxQ+yRmR@t1&F^ybHZE4PF>U4M??{kg-$oOJnS{vu+u9}ID z&oC-n*b`Rk3z;gCUa`KBHawLPN+>wK(?o6O@&H}aZ1UNe+y+hwiMjJ` zO1k$#?4QH^CUN6Ng>PB*hwg^q9=J;tMUqb7&0Wd@88oLK{p#)7ytMA5XsshK58hv2 zRVJ__(AZR8Egjhsjks@LX;X1zP1C5xgjW9;xrHXR7jl+t&o4|!7)+TQT`l+kC0#t9^C zZNs;^vZ@LWPf;=LhbwMtvDtpcWZGb1A5XZ<@Q_LtIIM)EnEQb){xap?-VFE$1;(!I zVsnTwklIY7?#sHLG!8#7a_XZDxczPvK!otp`IfI0FmGt+K?YgT%%ww(P|i!$O%IvA@pjqL=um!BV+)P zRrEmsS3s!0hO_{!WWZJ8hu9$=7pn=eTVuMxECAhCo)ilzEBY)77{_5DRV#z&=7LWa zBLHfe<9ZLRUtA;5TNG_HnW9h%P8Tw}AZX4@Bn}$WK~2`IGYCZ)3H__C!<6hME*F(- zCal-T6AH|4n<8G9d+~-Oy`Db3Hf^{Ck&Q+dK9gduSFGqQ(cAfrP3{Z3Qdj_);gF(M zWo7(2FNWS&dJGo-wJNwC7(s?SvW$k4KC+SP~(I^6WR`#T+c^nkMzj28$vA*z*X8b(*ikhaxi3 zw0%g*CvR2|2yYy;6$b#CR%CGH{o|kvLRe^)ff19)(Q~pU5V-OU?`@OE)qIcnK4zp? zi4)Sv{FY+_Bv4RYDMs@w{niyTOPS?rpN-Z}e53Pr5bN);XQf3;rbg?6Nu&8bpUpQ= zfYA5e+V_j%F`42TLCMM*0YzvHtsBefs=6S2z9xu6db=`{)Kx8k=2BbFKrZm$O+#~( zN2qQd;?BmVMYh3GLxrb}pCH?X(3^394^z*)=N-(p&}12*mO)3%J#Pobs8ba}5o-Bm>3PmXE70&< zA2sXp8TWZrJgj83A5X&}+%ttzjMS2GLhbt=F}W@JO)j~@<@41=7{*jCzw&bT zy|tiB*;7h_M6NSxQqiCJSP@9~iz@}O6-iTlHNTr@jOS4s;Pr>&MtDxk$XS4New+&)#ybSTeDywl+O<(4Nji;+bY)&{OFh8GB9+% zE4~%Uz{Xn9!X98mX$QMkmM8sRp#a03&U9+Wb-75hFQ7vo4AJp>ii>uZCLbP_Q#dP6G~s3sTk8POr-1)W5PEN5RxUqB|C68I16M$Qct-seRE=A0VPRJHw2Mlk?40Cbej5C z7bx3Od#Q?x9iP##-DcGydxRD`S|qKPOENfil(Naki()Scj#b{?Y(SXh;)B`M_iV^5 zs(j*s%YgsoJ8o3dlj^Q`w-U^gs~R!$j;|<{w;Cr!eyOPjMQ=zv?=*KdxS@B#b~bs6 z%f|nzn56ipL1)^OjmB+@w)k&S9QR(6Rdgzp%i*RC@FR*@OBp=ck>`OwZ1Q=`6VV6C zt275dceoE3*HiM-$S~!47V>-^(sSb|1vhpYJ&X(1z`(w}eFq2N;UQ@jEHLEUJGAo$ zNajK5p60(P*h(~D%)oWmdj*@>Fzyppd+2#k?KnL9PuoJBh?^15o$)OQrIxt08f;dV z3g#SslPH-BPXfH4kPHg_b2+*6z7L75QDMLcGLKVEL8|=Eij%>TN+diRKeL z1rvb^D}G-C@3oPUvi5v9#AMFQwukcS9`5nyloXP;lcOHXR!)AK6C?3n5ZAf2ZQf*x z0^U$S#P6L>Gfu$@C}p?tzx6p;aUA34r#<^=sYqs^P*a~)Ix%dQ2Bzd8u0{1;=~SR7 zrdSl*#7vHug}qmHMsZKQG(}{qAOVo07A>D5f^J6ZzRm_+T`v%J)qhk=p@tAP#AD6p zss=^v)r=0=>Uf|gOR=N$*9e5{zw!E&B~I+$un~Hc9ESyL%oQxV*-_I%=GcQ0z+!*w zo6;FAGtp|e9Ig@5^jlLqsYE(raoQ%%!0Z#cv)h=GwvF4?U!6&9fgE0}x86`C%ZgRK zngO(FGLh8^X3A^S2-V9l6$C)>Yp9rxT#a{4{76v9=g|9@Gsxc_I+mQ|s^C5Ae6myZ zY2@6Q^6rY&%|eS5cB zWU-YBsbTClK8+#^!sbHmqqOkSclJJ?-z}h+szYw&Y5AVhR8!6Czxtrc&9lxh;(Q%n z!^**61$H;`>ihmtDjqdY%GwyZsjT>$9(*Bw!I1bKv-)joY+NL3(s{)OCxEzhS!?R` zNTR(tt4KI?U>C8c-dBj!A$RsxP)sxOEgu$q=wi;COG16Wx- zlF+-X&3#K|7J*l-kE!pHQdnQ>VmG3zPDdY^|?Xj$xnbr`Ah~*BniK6`pAl0aoI>tA2)Gx(SlaRjjZHjyuXJlpFxev<^%K7Rjkq45`|S||_jPviE++gfy@B?;7|Di>|a$WRV?W(^%r zSX!S~(l}$2POBVQF3!(}a(Y4ei*DT>q*I=(M zk;+Qz@CJ`m936$n0qd`5Re>$u(*ehsOF@tQIEXXtISf9EQl(^wJ-nfQegH(4kMjE# zgrS`_V2yQ9a)M-#uoZq24=!Zt4Z9Xxm|%gEA*?X_`rU4_hgiN_vA>STSGI9`+G#s{ zbWP*6Ni#t1hnf(evl~$h2#eJYgh`>T`P@t##%d(~#y1!IRG<5 z85zhrvLLRe3e{a`f?$%Acqpk(0N#X5JRO!K9g``Pe4l zhXPJ@ugl+p#d^p$C8&Y{!_?3C$|}9r=>!-vPg=l~AX@kLS^A~i9IoP%CNRx=d4IE| zCr>Tu$moIS)ISRP+S$&{E*>cd-PeNviR^7Qom%uJi^E;g70GJ(5M9!{hTf%zbG_|~ zaldiGccwlkm`^X4LW?Dly)SBW$ozPsNDyK55QyEeU(A-a&yUilF2+GTp^00&sRV|j zjw1!=Y}ok##7SrHxVK$E*^KPqfvJfVhEZd6!2&$hqSP&4Tb@XGTwe!QwiLs->?gSw zqK_8*OgaGO8VWAHQUCf(P9!<^J*&<+cri6@p?s79d~AY8gjxPmQWH^8$F1K+r(Cy) z_Ch?6%eO`cyRx2V>_0}Gq$^96&viduOOKPf+Rwg}1MPl zk(hSCG^tX4BLs|Zc&vob-2*8nRM4i?5}y|!^rVLK z$>#K1i;OQLieA=c&JQ_v9uZ@sI*&(6JpI50jK>K5RKi9VsGB6o^@>Tpdhon#|Jpt= z5x-@x4p)hqE$MZpSzhuhr&TbR>Z;g@H0DxFW^A>ot3Ds=VoH}be{>zQ`|Ht&GN8DiyyHPc zES~`_W6NkmMK}ftk0AIV000{y0iVj;hiQKR00RI3IKt#Fq8!3piwq9b{vuZVA%kZg z_glzySO|F(Kuk&W{u~(PfGtqzEFaa1d9?Ex9aZD0QGW`G=77$~Zsu8j?MwUoG19yV z9_MBezvAn?qiH!qV*GoTXtD#H6NV|xHx~p$c*|Nsy}LfR-+7J!ct%T4KSa18+>QU^pc-l`DG46RSJ+ zY{sx49!{~e2jaZU-Q7vAP>cg#`aGMDE7(pv0P{_EM$&!Y{lCI7gLA@7z4Zh1ll@-r zhmq}yLCz!dtT*t4#FP9wk1fPw4PS>GtxI{7Rh9Jp249QR8@zYBJH-4wnGmW#=NgWX z8IEzv3qi}9hqV*ldFOBlNr>HI-1dOXr)^mPW(X23@{X7=vZyby zaBGV&K*Zz(1`1`0_e>ecZO^{3$bm?HDs>jJ^W?{$*;dxxl8Ith!ujmzAzii(>AXil z5cFbMXIM|z-~hUEil-pvJ#vjY$nzdy3}XmrouY%&&GyL@ceGyvvyIA$U?$~i_Ti@N zkYz=Zv?(MVF=qL3wPe1QAPezUH@Q)@uFbAoU<0f2{O3Z_ztz#3=mJ)ApM1IDny0 zrBSB{97sO$#)@%EL4-3btbg}NX`#`HCegG@fxGsxe(A>(98zG+MgI>c`reDC2HyDt z?gkw1mn6LpEY^>1G>ohL@lAyTbdkod-cgY7e!adA{#lX2XBalndiM|dWsMJ&vsLeB z!x~J_T3B6P3d-W|c<&uSwxXwIt4wu#r`0(yTerkMK>dw^+bq^prL=-GzE#NoPqq3H znA13fUIGOJ)jT5Svcr{I!gqB0$$HOkV}B71rpu@h0X*^A6O+82Me>6&p-|jLn?X;T z;Rk4c50oiJLSx{l18k5-PS6(A`DY)(sW0AKBrEH8$XRpEc>+NON$X>GVH*)w05mu|@3ajZ+X zbR?iKtL!RFp_v;ls%M$j3@q0%?WSL^Zk~9}<5>zrSzb;FIRZ^`eafSZY&_ZcjCvT- zMKtBAbdm2vFee++?oEFBxr?!=&cPNm_lN1lqK)_+>9nQPLuNNqRtHTPKK`WjqMh~4=0H9QIY>{5mu!5nV9-2r{1#fI7z;CdT>j?)bl!(j zL&pLiYLS=)4k}&@<{n>{B$#q=fL%0aOU!7iJ`*LqW#r<<;1Uz?FZSAsDBfD~%t48Y z27}-|GKWS3$J?mt!1_W#t_}TOSrZH+4&RG5kqPC4m4Sv0p$mLT)zgZVF}#nIE^;c- zIMiYbT&C$u4U3k5MP=mJMF;D0+QM_Rg*UU1g{>RznWu?1Sk;@?BIcIElT2d26}=ha zxh&1U_i+X?gP{*OpO47TRa@sf9|Wkcf?`qal@U_Na9d8NyaVYO=zbTEX?op*a&{Rc zuy+kH58-@uOx6dIUuNg~+t|aN06dT-T@FWp#n%nN&HL*Y0GZ9ObRJ5GAOjhqy%Iw@ zd2*n)e!;)emjn!7PIqPNAI%`H`&2sFE9rrS%WFnYR5VWWZ^kZSYA|<7KIxsDKnbq} zOS(!~iB1e+;M0`XBoxOGyBgx5?~&qPrHR{G^)F8XV zE+d0&WOdH<6+Ao3nd>e@?+A9)|?0Xs1aI)_$vf&Azr%Mmh zOU_hl%$L}k1a=Y^coK`yKC zdAA}=%rcweSvhrIA<&EsyMXI(G0-9a%bO_|Mq_Q<;!}QH*TK#w1VNLwa$tYIB=RPVh?wLqvZwLgR0(B+PDEJU2-fair|{x=iVdu zW5aFybeeKJt27w0LsKZ}^jz%bcsOQd)uA+m6lFcaqodvlNm0r+*zAf~8jMq#xKlZl z>Y=nD0${<934(WEIZ6T!a)avSvfcUs8P70pT=aj;jOdXw-)SS}GmLo)(qWO&9mmh@ zq66S?*aD}3zWN=n_Rul9FoHNO%5 zQ+-|QLN6sxihSwJ2l2N=;hbFdWGKq5NR|yLt|lmk02k|Ok)Q3~Up`AK!?*y(qYDhd z*Eg8#qC4J=r(p`Hgn^6QD3@$D0qC2i7+v{NcGu3tu(?X982|^Pz)% zUmId#n*LOO-5hkq@c5X)V(&goQ+K-UR1d3&>^q{81*5WW}9G66eq3jK%{af@3kMC$UC zHuk9ajz6rl2=b+!Ezl%hjdj<018j@d1l0PC-rYngS?;l4z+h}p{j+yH91 z{8!71wQnem6FV2zR#SrRF^w}^CCt9$^0m@`et)VbZR|Zf9W|JAeOHVTsl_XEZ@9>@ z2q<-ka2**lu;cmACAY^#kbG$CWdju;Z_WjZY%$2C3xyevUGVXIm`ao|!4`u~9?kc(7a-6xj_O^!@3S)o&-z z6Qd{-HNBr_R*gq?jqZ61q`7t}=vA^SuMJ}=%QwwQKn|qwRg#2fHadB(ZwTWjsV))B zkS5NEYolyIulH`EYP3{+M!0MmQmgW{`+N`(tWZp1-jaB~5;F&M5bTBMAc_=>Fu?i@ z5Duw)nS_ne5TJGKPfd`se5D>YVD!AuL}URJoYXN$%oQYi+xNizOT`F@o^^89i^dXi z_;cbjM{aR8=!f}SlQ7|8(wf)t!&DED8*D2DXk6K{18rF$p!;1h(&0N931w&r>0Lvj zvqbK;1rRY!nVxx!1>nG%wU-i{-Ty&+iobVSDdtYi3%p4gn?x@Prcmu%R{)v5`7!gF zaP{A}yt;{2;J9-aqGo>R1qMqKj?3mh8P}6SB;eC zzF~LnX#OiDHo4DDR77=ea%Nvn05n4!wS^nM@75p=HPXqylxsTDp{0J|mWbDpNA=#j zQf|G!NH*dzSEz9B!9$8d0buHr7obvh!6f>1ZA|&^e$LvqaG)8vM#KC$2g77K+ z^PCdiX7etf`-nLuIQy9gd#Bl%458O;Hr8=bzt|PO@gdBCn;#^w3RnVvGw}UWY?PS{ z+R0Z9k|1kQ;^dqTUytzGEh^BnqRF%gq`1Ir7U(_K|6y zAuXD=!)zj8`8#gNPY5)pv_LWzC*2l&9wTcZ36$l%*|SA=KSSzM zr6E=pBd6^qITI9Sp{$3)&C)6z(xf3s3>9Z;7pvSWP$u3IaDV<(#v5 zn@yG&4UU^?#VH7`S?N74qLW_L_d@hTI~LZ1FgymhENE84Ly~}tc)*Ibd@_$dX)J4$ zO#6xgP7+@?xL)@>SNGqwY=Tzv<=433WD&bgL7R)sp%XuHK?7RDT45s11R70e4!1MV zJTh8gt|-F2H{oHeix~{~B*LT#a^%Fg1p^AXUwb3tp z0E;_hFu=vT6IqCYXbK-~!n$Qm#U5vKXz#AB0wNg`;iM{21}-_j+BCkYKTsk zmN}_D4ISqmSVMl)I%xGl4^R~Jq)Ou0LNFXt9=6u0uMelh=iztCn10?i7QAQH5oNDx z=Y6M2>ccG) zs0GMS$q$EQFm9tc!5RjTkS!z^v1byP3Nfd`EA0!FQL}qm;vs+JulPqW=*{6 za3?E%s|i6|&&C|o%S>*N&NE^Z=R%P&kzSvlsNZ})JTN{0!tcD#@~)^|qf&c(N67A9 zzvS@TB|u(#=a2Au0Mj!mupx(F*iy|tH?3EGuODxpu`z7GuD;~GEB35kU+&8-7DChX zs{{@Q9)v2RE^G-lt~S&^rb@JR-whgEo(qniOI6m7N+Widz+f`8cdxMIW2|cuAA4dUdny+A-bv{*cjTTWq3#MA z2)NTrd=W4)%~mwhWI9TIfH_-Q6v;D+a)G3J#J9OwOT_>nHMd2xnl(x>%n08d=DcXu zBPs=e#sIShoD;8+zuhpV*>7pBJS}9n{Vd#S zS>WXsljtoC7FMxBSPA?Nt9;t+{}&rI?=R+k#-bvMpZE?bqSej1Gv_DvFKwpQO}SALPF&Srf4g7 zE?!9RF7fArSA5}tuK=R>k@_KG)~uSK?y0txFY}y=5(hLWDMaLrM5i|H>p>M`wsnQrFXnMKaP=Hl{3a^AS>+|#nJ`)Xmk+1o0^~vbU?D zna1zBY_ECjeQAb)GiKub$p-c%)B`gxl-|nY)4?!)3&rt1a)s{MAfh08s#xXxDUim@ z9cev%?hzb(bt(OZ84O)_?$hNUq1{2b8c<*AJRv&8>Kp`%^&zlLXyMB5OI)H+K;>oD zzO%%BR(ArP6)3!0r<>Wc<2|;DEX>?fWI4hE0~wE9o>ax&S|}+y===#gf^~bdz|BXX zw|$ZL7`N$s7<<`7+f)s{q(vF}zJ-)gm-a9l=Vn_&-ji$n&)1j`!q2iD+__T-YHkg3 z#7k0ul&E(IPzadjN9au)l)|T7bbeQ^EPw)DqOSCvWir}93HQiK7qaeT=A&|X4sLo2|05@oT78l2)XIS0ZgPRW~ICj zqUEYkY;(eT?`Tb7>iexmd*P>^$M~M?Kh}i2>U^%`ZE6FV-sW&`KRN*^Q(bBXiq>3L z1>ku;DXnT*$bu{7fE(`~wf9PY_@?jB?yrXIY=vbiPv$-Lo!-1bf zWFIy;EiwRGp&~d&S-N_$YtgOsoMs%bwz41#lNXE<6;uavUdzbI(QNRCwhWjO8xh$8 z7X|SSOU}C5s7r>G^jc6f4|HN>6|b-zV#Ng2^-dvup%kK0$zW#T+>TyMxyOH$?ji)2 z=sqzyV!?J1g2jHlPSSkJg9N@nW`j*Ctfj9$%QqlZ3tYr@K)OZUyYRfkh|>Uret`gz zQNPDO))O9iW=qY@^{mj329Y{{F_ERU z8-PK-f|UEMpz9#!-UaabxpjL}%uv6&IDTN}lYXV@?FproPdyc;g7zHL~h|wnVpMU8o5$9=}l(I{@ zQM#d8VvqK5u=l^*-ppl!4dw)RdB^*i-V-ku7|imFvUwIyY&54A6Tzecy$WbCDXO+FkH9X)Gof_(=|EdIuCPGmp~32n2J121k9)_|N8b8DTH{f z4GZk%_AiLD{GHzw5)6+Ca zS@z&k(`qLkO&9+f5MlbC+eHyQ zJi5Opp;TyPXuneS8l}TgE(fs=P6<}`*Z$_=g2mXtJvK1d(Pn5 zizShCsQe;pZs3fY%Xr?6Fzm{(?#HWPUWGZa4)~tctMC=c$MWt}l5(wPwbrF(>lfakJb#7j~wZn{ImH7RWks-#lQ5 zH0JqUbJy2>@R+p|sA7=ThX1p{MW7l3llEA&FbzlsfXxaXtdxSILeyeW`&KvNF zaEpaMXBdSjmPN-sqBRDn4s`GWa%9_-M=z6?V`Co63`?>?k~q}Awn!BZ4tqYsmfj5E zzG8K!D(e;J3Y~a4W3f({PitFtciCTlE5$fN=j1M&1!-&cq(>fwBW{NNW-NW^Gz`~$ zEqAO;DX^t{#2M0yteNRSLCIax`1g39u``^9YcLL3MZcdM7?O`aadyor0a_SNXme^q zKHn*W9FUMi#U=>`I|(b{Tl711?`wjz*#d?3lTkw_cv#a0$&Wy_N4G8m*e$*8gOP&a zV}0VVrFuTe1*J`a{igmB2-qPO!zzL!4RDs~kSt#|wA0k7TfJy6Bcuvit1u1YTG7rx zXfkaH9RyLW0LS1@WPURV&y|8yM#z1o%;KXBQ5OK@c{^za>z`ik*Pw;=##ktvGSA*R zmbti6!%$Dx$>#xeOG)nNsg%rR!g@TP?ODs|(75_8xvqe1FDh_9>O}KR8VLT!+*d5D zTfch<@q4(j*XLrQ<{Y$xf#WYBq>3O`+X*Q4bX^*i4%x0&4#->8qi&+8Vdtp6V1<#6 zbUIo}MWJ~_WbE3}#MiA>>&4|uoMI;?YbvgQSUr%js3VY?L;6S&8R{3K0(X70bCHp) zfi3`4>>xPM@Q&e!O=x?!c@Ef>rdLZ!xUn6|-pA_{hJi+j7cv=cxfCZ)d#Y#+7X0ni z&AAkc1H0Bbnf>rGszp%7EDQi}^#iJL93KB=SSx9DlHjbj#{@|`;W_!svcWvI6EZg; zhxI|vx_8v?DPIa|iKX-^pz*>aaW7G^(a-ou+EI(i7?YfeqPFKV*Au3IYq!*y$l}+s zW3MvVVzpH$To{Y|#z3WVWmfAY;}u6Gy>^7JdX!_AtlyRY%mVy`vaZqKHQ%_ zq$j)SjM1Q3Xn*&DgCigRV4+V8NoW{?w*LO%K<5$biYXPp>PbQP?5REV;-%{bVPmXE z-wi%Eh(d?&v?eKtQ!qnQbui?prNHIJ3t&bK4=Y?u_4^P3lVZYr+VxrOXOd?XtSD-s zxQ0nPSi`=C3DvY*#$V7RRn2R`d|c=a|54lu|FB!+4Nh;^hf?p^HKc@Lu(e&F zE{i5^0OPswAFohJT4Xap+o@VM!Qlrc{umGGFNq2O4kO-UVF)#Ee{6P6VCB(Lu-jGu zeG0ePyWML`7Mw}wG#lfvyXlj307TrUVxV2sbxMa%N=3RW$6(?KK`BpIqdr(sw~8HR zYAxd;`*odW!2d;N{t8K=8+wu!s=~1wcA(|wE_ zY+@Db0Dav5^bYNYTeDZg49vkbxJ;&@n%U2ko-el^irZbQenuBFpMk!$z*yh>2(}|c zL^T1`IyRFr6PWwafSNa8tHYP5*V~(G1H2^=);r(QeWN_+o>LGx*u9Sh~imbM&J2Ma2)C57Fk zwS?UzKCixwp^;y`|GiZPa?BaBQ=?O=u`HK!HH+!Vo!>H>)w_oggh#8!IN}-V2;XhK z<3F!q9|z4l6g&CIP3^iddnf^>W_(d$9c*pOEObqBE?EocF3kBX5Y;88rkS{|-q>Y}6m2ewFC>KVfzW!Vp4x(8N&a#9s9$~C=${R_vbrgrF6VL_nYR0L-K z&`@vMCamW#uelXPxiAsp4_l3%FEycMev{5JiK`aRERfN5@v4(By-0r=?!kgbVHv^@ z%RLB)9Csc0o0t}h4f%d-2GSOgturKhq?l<~*XVGg|8$V7@cuulyv(c_93BC}H32p1 zghg(Yk@5IDr*}JTcN9w?06Bnhh34WQB&m71Rb1deRN~g588Cbh+v1o0pP%Oca5NCHL z@a|$yOBp%BC-G^V)&YZj0yvQyPS&@Hq)zp%3NigylgXnOaUG8qC)p9pJiMsqPhxu|P+H3%TVO>C*gB(_(RPBW%|8{QF0OF{Lq^RMx7 zD;s_9dgryCau+2A)n|70G=Wz3ebumqiwsD?TRMF!rxUL54JGm3xNORE+AT0{Mqkw_ zHJisKn#Gljp&CIm!QG7+n!o7eSq)rj{#uFm<|8@pr)P={#8JU!#&>Qlh#4RN;l{y- zfOk&EEhr*JjOc?#pU9NVRk-{ROW3%CNpH!zHbdh&uAYu|&Im!!SUq0AM<#d<=(+1X ziF5$N^g?6*z2}HJ69!F5>j^TJt$-hTO_d=7oSxmjTTp>RcENx@zxD~o1~xO~zH{+JSycv>T6Eltjp zJ(v@Wj+{03#vdS&n(pdp0BRnmPrD1(XQ<{aGt2huZRT5}^a;ZAj#hA;=e-r2Klb%! zB}puqE|83^heF<|A8G63V{@1J3Ay1f!bibi@I!$@Eb)ASE2Z?(v!)gbVa;fG<_iqE)BO;unhFAv?ms=RpBX?2nU+ex7ua8j#c+vdVrl+&+HdX zF2M^^&wJ;i=;w6CW5SKMX_YQGkRm9g8Nmcgnp3;v+3TLJp7UmafU_Mm!ldFpuZA02 z5yV~gT01#q6@%NJOht@tQxC>6JC;KbO2I}B}$z|QnK;YNX&MliMynO|bs zia!)FzTK{`am2Nd9h*$~HJ6YE^jH^M_0`9|m3z=NKvu?k90t#JGPWd=0sqO8f*R#f zC|_`WSVE^oSjj+HA1zXnEl+O|pIkg1N_xmpGViT8Pw!B<*;VcZcf3-ZSo6}uJl>1` zbm0Ox$j=3ZwqsV^jwsVAtR1U793kcJPTEW4>%9ihrt``d04ptvP@~ zt4)prFHCG^v4t|ul$h5(ji3N z>qfe9m{R+g6uC;X^z{D}%rAR}FDztcyV4zm|Ci8vcgH!)_+}eyKtc)K z(vXu_*|&2JaLH{M*!v3;_k?vCNO<1=o`Wv#N0`!KeAfyI95A5=y4eqw6;wwhWZd@> zO#EZY)POKAF{_v#X{am#@j-BT?*YWaZ#^(dZg>aSc_mUq3U}Q$LhmMmmcrTVk<2-Y z6gIXZU{wCeug{jhGt?Wuk&~5gyIqvhVx{;{NjtjP?=;9xQsk+?s9^c7sA;NZ?(}FY z&uz)@5htu1X03<^6warRq?=1H-FlN9OV68^0BqodZ&f|7CDr%H`)%U0QiWCU$SHS@ zLN~Cou`3yctrM5U4`}Am8VtY)$1Q2_6z4-dRf~Q{8&kqTV?75jk;{iMOC=Wz^whQ| z=OnOp{sd6*lTlN1*IppD4~@HGm&OWV&_=pPJ_0=S%fjS?0oG0B*HzbC_G=7vCNV_^IB2V)H{`vj%?X4m9=t6}5nDA+9 zUCX_Ik7}g)kNf6=GrN%@wG3cqv~HA7-B=4TdI8#_(HmhLtK=dfZ=4=0n}E4(pj3xG zP+J`|1f`|F^c)nCSZIyuDB&N=mP-;(VtW`Wtd;T*4(2ic!DLw=*U91#7&aavpX43P z#E^ceQ;k#+SqUq+{I6y3p6)DCW8!7dg1~eA1JxQZ4)`~Kvzw2x%iV7UN2a%vktgQC znnhQ^P`$3rv63hJYO22<_%7dW)auR)GW0JN65Gbonw5E8TqjsF^)U~djbMhP z$&wt1%91D*|9Dq4c|+gpP#JAbCEb$1tr+}u5knx~`fE4nJhklGc;fRAlr;7%S%yac z3S#iba6iN^I}ufv;oW2pK2`xFS}G&esdo44IK#bg;|mTbGOCGLb=W4|89ddKhbp?Q zXrT|`1dYRZE!Z(oAck^tKQqMmi7Zkh!d|CqU@uReI})Vup$mglE^lD#+@x=e`1F%Q+^iSD9~`Dl=kZ1iV~Q z+d&pP8#Wz#e7C$a=p@cT=4i5fTaYmg>uTmO=o(3!bn*zN_Om+i;09DKdL^0Q*&GB) zaL99UNmnaMz0%^CY?syTVL?UC3gZdl`2Ta_jQwn!y`2ia=J3s;#inC`Gleh z4B-@4w_QEpZxt`fuNop?H~cMtNz4Pt(NZg36w({p!{mf8_9X<xQq=gK&Nh@9MH>*iZ(w=O^!mlS~ORjtk~8#eW-;*64wP@R|#z z5=_bPgoK4aNk~X8K+yaNwpds^ASlFo*Ic^`0wf!7;f`h+&&(Zsy-aviuZ)y`#s_a53R!~HKh`$Qu zDEogXb}GLjvTqS0J~doF9gmV*=t0fIu6aQ>rFVH8*4UfWL%v-PMaK1(P)0#jpG6L* z^2!qpd#SIt!9s@CJx3b@?S$5(puTdFjiMf_#tzsr*Ao!;akp3nN;aKr}ucs!X~WS;WbYevM-il zXO+cDoz(Qu9r7^K`OY(L{%!McoW(vcsP{S9Yj}4$Tr{iLQJl%z}v@5!h^^@LUW2?AoI9yTSmc05HS_YGU!)ecuoV>nX{@ z%(#9;9&?wS!jfIR5mmV<9O|WhM>#V3S{6M`hupz0NRiEcqLHRFMNfI61loezvwO7XRMN(F! zt>t%wDsdgD9sG%iQOO-Uhy;F)pzTU)L6mb6m|h?7hI=xCaCFTFgq z?cnAp@r4*S;28Z^A|Q zVCM=#iqM38To1<2ps^$w)I4n3++ckUW;&@WB9H7!cY`-K8M-YaBa=Ckut3!C5o@*E z4hP(7MK^qYAL%3Z?E4vhVBc$@ghj-M1D%-)2hvO113N7UmE>6S&FlpX+&8#v7O~#n zzvjHiA!Xba5sxe?9&GVjYDhCS=*JBdrf?JdCn;jThD{J-@ z&xybBSBPFzxvm&{YTx57)0Hs~@(%9H+uT1&+b+$1>8Db|1+e(IlxK2CYLud;6(JI& z)Gd7t-kZ9KQEk|)9z~jkis;OHw%zK_~Tbe6iZLufNEoN-(95Rim zOWaw1cgz00KYuz$@J?g{XW6K2$2@TP(p)mCzKc+)jEv@JXO9Wh{|^~u_ulPEfem@E&4#lIC`=T7tIdW9j1!M|}LE3B`3 zHWn*^^Jc}OGfvXr1nnf-2tbAz<)2e5ml*nhlfAe+;P6T-#AeY$Tl$J`p3-Yi1x3oW zk3*z(9q~H^oeCVHc)AiM_%sjyv3dm4C zJWN2`*}bX%Nn((GBnb-RNoT74%GAjzP4hjfT3I>>n>ONMy>s}*koJSJo8F)dy`JU4 z5=2lw#D)Pwkt#IM#CYIEIlI9;zYi|xdvC*Ha#}y+b1=v?V#=Ibnp#Vw8P9gTCeh0| zp2@k%{4ARwX0dq%Tjoms+mf2Q+>Txz0bLbP>KnDUZfOKDcv~L_fJdGGXrSh6I<|#@ zwt{cFT%U)(p|=d5oyGi|E-w@O)qnj?Yofs=4N^Nlf%rPxlsz@)I7fNd3C5gHS|n8O z9l1B_X*o2yaFu>Goz&4|vgrLlQbCh#x-ChS_VhGrMHB63qfHEFb%(cS zoc&lnvV^I~#5mOw#AcF2VHAc3DdIA$ zGSDG&lbGJAAS|K2E;r^;6eV3H*e-&;BEyqW3T0^RGOg(N6?!bgs&wt1AV{a{mdKqRncGA7C9@_Vg1fvi#Uhgd^B*tqXBTR#QjR2b`!QX{s3 z-&dFQXrrr!LmjxevCrW{3)%Cj%~$B$ow1#PPPpR}FV;`f;MW-5}7hIgwNXfZIAE7E)ci zTkVA%EAJOEv6%)NM(lG}!7XL82A=~!;@x+g?yu3Zjd#!ia(hC-(OFHxT}p6*RBUv% z>0rDPTqO^hh~tJF7&g_c`6B_o{Gl{t3~31U=FaJ&-0wdb_fR|jUP6Z@0hd4Gh97cTP=%#^D?>a}e4Z1%Zc-=*wznQkI;D+92}jk1WI>d&of zT^c$P3?yb+fH;<=$IrN>(-(fx*g9dfAymzY;BqMUgoW?}3xXaV5gZoaqZ&Y5ao(uK z&4GhGZ*haIio6#9Og{U*KUegWiolxXBR{>#hA(4Zt)t%aVs7{aWR&fp_g*HEGwi9= zOmJZSsW^)7)z|^Gm(f418)BL2JTpZlk9YRb&xtvz8`eVVIv?(GH%hVdqYEz)0_QPi zGH6eBk?zOM9&#+ZGk2H`{Yp83&2vtRCS1?t5cyGWE62aw*6f6auz~~8%|r;?a$FQY z81!XX@nAC$cam6f$nv3M>I(Uf+6lLhGx+925ANfl5S%^k#M}$tvzZiGbjX@h}|I?4Vkjp#x#6;PfP zK2!9(5_h(!kW;Bq5Y^!~)4_f%6r_0iMc2YE(uxx`}n5BN3T(>YB^bBMQ@jRj0&^A#%ky{;gt?YCs6Vw}B zLBR|g?_-}p5p1YC2F;;yzCCHC)U-fhw|T^{1MLl^g0?zYQBbVtdG*ID$;_0^YPD&a zxK3d3{jYM0{lQgMek!E}i-gmLttZ!j_q8&of=f92G!!BIV1a-esp^(mJ9E(qScH@- z%BA}hMeT*Hjfc+--V6Ze55}=|2iGhUIyyPG(6fTtVNRLi`Z67bnA8{%AXR&PZ*MH} z>0Mt|k;nmRmaXn#p!d!jP+d$b``4Qdi{m5G1nPu|uPRjJJ7(CDVKy$i>FXVGRR3i5 zS#DYYK2jQ%!@*;pg(U)U7+EiY;3}`uOxzM`55@Y#c1meZlXuheNZO)z6SYf_Bia(b zi86p<2Qm#mSFN=s#Wzy3WVjCwfEb$3*{da3&1T6p2AI)WsLo1ZWlfJ%u{xa-)`?;A zf;h-daa|Eu-7{RmX;s`UkqoZC%}Zi`1&)h4XaTWjJn=MImI=^Wj;>%_va!3S@q+dL z)|LuhNH?BUw`PImD@X}f48|de*3^~h1L<$bl?COy<&Zu&7-s=bPlj|D&9GpF`Y)dKs$r28U*q00IKww=0A(ongh(@iCWb zX!%=oL!XNkuzj^fgC4^+*=}+;yVJN{MKEZ|4HgCf)I7O;{iE+7~jy2JKb16=m! zO&+D+)7~d&!)EHLKz}z;AS{3#$WKoFR%`M9W5Ol9w}W=4ZP(zB^H>T^TG7%`%I#?? zH-FxN`i}pww!Y~85m04vjl9iTnBSd}AOGcEy|!arFrWA6jGVDErM&JcyoHgjoPBV@ zP$R{BwaM{u(~-7i+4cNrZ`HO7o3G(l;W7txOAg_pW(MiZn$w$k&2y#Q5u)|Sr%l!i ziRbdXio@$NT~nfC4RxK&c2UW#I9E&{`Z*N)5e`glG4+gT6c~B(6Mt|b6L76hHs+Qc z@PcQ9Kf}}%HtQMv3YAbkL*?bAV$>VX3dsD8+JQLB2rd4ykaY*QruP} z+=z0PMg1G_;H`kM{7zVW?h8momF2uD3>7h(k0~P}ifDBtmiMT0MmSgThV4U%mG`gD zcdHgd;H5-PXhn6n)p+q*4c4hMGGh|X5T=zgsa}vLQ|Zt&$u}O_C9}v_N9Rz+VJ7%l zdW!{cv(W&4FswLFay~5cl9PKTQ#M_ooN>q*p!>!|-p+6=YXq>{OHh^VhszjFKN*#q zSoga&*OJ5*|D69n)njg~-@SHR*RCr8fO^G>JeyCa1FWCX%wau;hz5*QC)a*A|NA59 zDl^wu64=y?xgKVcQu5CXQicGM{rje-Ps^NL4Lu&wd}eV^D}~iiAou<9wQ$gT`iPI-dkDD8 zfJCd6bQsZF&)(_Y-47-D8vTLJ)&QIe76|d>Db!qj!5iyuVVkbBTvHmLXwY*Qu zWqsME4#*aa`*_`>o0;PoanN#iALq&5B`s0{_Qvpn-f=hE2NRCs2*q z7SWSK23}BlmnvJ-;2{&ht)+OPvhD!4jOQZVwrglaiZ{7I_m)a>x{op|9=L0HH<61y z8TYJqW?f>(3OW^W2WGt1UPZ7@Rj$L0_)vg+#P#4|;eu0ZIAOnmIk_CT2j7{^oqv%! zS$n!hfRIKg1Dh%9F}j61ZKK{x`lFciB{?}{`#i2`)8uS7(CM-?JewLHHcabA)S@ZB zleeUQ!7Hbf1bSe=`)3+jQB3O$#`+o~d5yr}G{b0agqN@0H5y{|lxhw?>Mmh_w>bVg z&^s)#dIin8B(}wd^ax6i0z%+@gMN#8u8_Jtl#c^@5Yn8Hgv!~_1y(<`Abakl4wQZG zL+-g{^ZtLUvFhd|Zj3=EJO!!uvPe(o&8-xhfgQ$NzauG8nu3eR9mkgDY;P=Vw;N69 zBc}Cv%lXXfE^iHom|pW=DD-!rEg$GTETi z62=bBg}e%KIY?J$NxhDZF=qG*10wX0g0j8q0~NlD&S`Fqw1|QlJWH@7u`ZRX8wtgo ztAS`id@naOtcdcbI_Yf$_8SEHqPu?iR=mm8^IgdFu38@NHfbBE%~mNlw7ZJ3Qd?9= z#1dDm(^zfLBWQWw+(x+_$oXLdDDTlDaEO^&SEt$!q+o_Mai=uG3V>NUQpcTaM-{?( z34#$4%av*7Gh&@r!P)R#=UV~-gmE?Ix7+P~bQTP;GUQqr-;x{X z>8RsL!W4-?zlN8yx&p?LH&r;DN|dm!`m)q)e4i zMs{-z<$ECJWpfZ?IW1g)-*pZ^j6E?Efzg?`YMxOdjqfqw6fv1ob`JSFB>OHVJ;AJ~ zLaLC(`EkjMkkt?fL2U7&(pLB}0O^9<^%uI-Ile0R9}zK-VAPOiGm)1#kGyjYtc!Q z!yNc+PN|w{35dX*yrl7a)HwJA8M27~Y2(cLCUrlgP^8kL4^(n^r^gSPRwKxWBFdX4 ziq#QKfczkw5+ar}jjVJ%?+)Fr5hg>HC>dgdWaqf@vyUZ6ZTS3tj==JU0?!$&6TLBH zM$2-m)rjcNrI3`ZEz{#2x33Q0KD@ zP?WV+~E^xbS>CxnG z?^4B`kORgI`TB#j7_Ik_n7TcC8{2!C!IniLP^BySbguE8MvCz?7Qlqrx@~3vWd40Z zTD*M|Q+TVKnGAez-v_~9Q*aKQjM#m~h#!93ZrJ5J>?Xv2d@?Dz{q497{(9b>+)OWN1YP?r|$CD5-2)75DY-NM437a4bER#u0=L zh2KqW8kF>rg#Fg+Ktg+js$l|@XRTNe9MV(6QvmY}7c;^4?A?STofSyA_1gIP$Qxz= zZU-?B%vuA(^tDgx`*nrs)u2rGb@?U6d7f9AQn_i8hnw2Nc)j}Vitwg6@ZFFw#ZK^o zlOQDY+CoaLD|8)ATtufF@2`9@{o&DA5!MU8(2@lFIM!qAi!-dKSa;2+eRY=uO14%p z1aVJ0Mm6za#B?h3NbqynG%cg3ENkTb;LSsXC9uQq*cK~3wfu{!_7AAv^QsjBLubK2 zI?Pg3Vuygh!b~{=ywu3grE_daR@4UaJfb4aW4E-Pkm2L10^w9g4f!qsf=~t>y=h5R zd?Uk~2_;kc<7dkW@_P0Rp$q5Q8`i8`;QVVwF=+Wr%SG#w<@*NYnX!LDGq)3`NEv61 zd;Xxo85{W>4r!_AU*PZmwDO|iCco&Al1NgTjo6q+cW^yGW;HlbW5@p+fIizPBj3J9 zUSA4(z8I>WW+a!`qE#vPOU=akq0S0de z;yo#o&}w%^-$QRtJ)ip=ebySTmjbI7Dh)q_A+*L_+OY&N z5mk%J!f=dDM&w@GS*AB=z+F{(pionHgY{C@wLsLoQW({px%&qQ)K)17{ngrOefie* zaU5EJK=k8`|71@lA#4EgWL|P<2qa}6%je>OhJVM)vo4#0i*5gd3V+IQalV!Ix+^GW z)oiaS-bhEU_5yjypk*3$JugHk|4T(GF14l~ALgi9IzV<$;d`Sogj*gW5hJPVYTOg@x`uQvBDkOYM%p~+j4`DAT zOosY+vFn*U-SY)bu)Yf62-B6pOrTwumP7k6kJHg!a z`VntaawiA+CL&PD&X=)r4#K1x#4S^iv|o`XH!z>xJZW!NmYnPblssfJT!W*lH-Ya* z6nR{tSOg0?y=$p@v@avdyUE*#jHN9*VfBoqwO6UUV8J;LTEIMPVDOL-EmA=C6jpv+ zqa_j@T~nv8psV1E>Thi&mdD{VCAb|U!u4<~X@4uBbnc{jWD-WF3a29!QuW$*b@bzt zVx8A;QwX9Pdc$cuJwMzOko8GXX5%<2za^ER(b_7=-O#9PQd8PxpZHDKdKDtXY1X!C z`JVbuTakD4S5_3EMsLDV0CfR+y|&jCii<>v9+=S}DJE<|K!E1o>pEa6AO4GyjOt@U zj%;U%6xHAVV|%G~1aPOpGOMJsZ}B=_e^v%Tta1`9B0i41;Su#U`gM$`Xp^XaA+f6G zlQ6l~PqewCE1$`n2TAkSze+nVwUK?6Bh)_e;E<!>xNu zCX$gU`)k5M`l>os{{nvijkKv56O)U7SrH^0H8D!tly-qvNP5$UUZ)uFZ!K;j_D#kj z_U+0PqxvHX0=t%EBo-ahvzU&}Os%5o9@6z{45f(fB9Gp3ah}7c1Cp_*1k!^xF7w5x z=|bF>cSvt^%glz)JJ6I`5kkt$u->4q&P>d zp@p*uJ$9Loj_r%^@E^n-!`&)~XW3Cq*{5DFVNt_gOq-X+RGouHVuoh}?Tw`=cG{D| zoz+I`(eif~9-{vrT#zHSF|R9W6h%D}BpOKagy~WOzGesnFUP#e!Wz}QGv}39h6>91 z!GI($@(o&pJ$-W?>H92eE_@Z-zy9jOe&KDoJ#Re$z@Az1uD#i+vcutPZidpq^8|d= z$dFh*P5>o_^>U=Fswjwoqt)sq4!d5WhBorFc;q=+ttPXMGlqokuu0qsvyGrYI-|Z zbFm)5a`nO1Sqv$}ELENh{^qY&eoa5&^S3Mn<%k9r%TSDXgK$Zax^q-Rydy>U&l~-K zIs`!ZC5J7yTRH?v3&_NjN?b7XhpnlTPVozBKFB}87yB0>T?8gp4@fWh{z*7~wp=0Q zgl^86KG!9lP{9TC5j!Bk7Gc3k^qS(m8aOTG>G+!Ucd6>S1-M1UEGtdmh%lN-8nI}51>$|b zj@q?gI7RBRAL!*DM3&Hp<%ux6*>_w4CSV>kt~~IX-DK4|S8~*(T`0W{i7oB^tPpCa z{DT#D^3hU{Xf?Ov-HU0_eP5{O{KHIkhPC74V`%N@f#PL^Wm<$%Se4xCRz}Q#SoT~H ze$GKmS%n#s8p;VDawF&xOvQG(2biv-Hj!r=rshMU4q^(DN$EgBsSL3q<(3@B1P34* zyzDA%NC^@lV+LMM&4IrL-Xms^>5Xh8^7hl?z-PX8Y#68DQyS)`sAMAt7LJ5I54E2b zu@Q}PJl0boFb(3aXA6U?uLfaWx1hrDwl#PMuYJJrT)JwE7@A3jPyAV!t!r$z9O%yK z^GR=-TQPKJZU6lekSvt%i}D9=%SqZe{H3QNoZxQw*yD|r(o%3bsfoH4nY}x{Va^fa zym1v~jtne)oQ%i-mCo9lwhImkr72VU*4QyoFxvkC0FV+oyn0KS#-_VF_DnfWdnd5q zRr9r;yo4JWB+~lBs90M_%*udT3U{Z+k~4aSQD%BVHWzP68>pjtXaYRAlfi=_#Qe5w zdA;FW#!5W+=cWD12UKom4~2w0}Gv=~tAS?TQ3qq3fx*lM0FwV3~ES4Ib;o&@DqdYx#W6c|WnIPb0b4Nj+pJhCARV8$>O? zPTR2Ru^0RtA-0uoOVRwM5P5skl{>0WA0R-5N8`!DTXHfcaWMgfaq8_I%%iG7d((uw z7-acDKy3XiSXC32P!y7*96jdg9E6MkpkU|Io#iZF>qQJ$2ucZOuYYbbsJS~eG z=#3?HZg^+#I6Ogs`&iDDMEDZw^(I^35Ze9eA&{+usLB$G`(m<#qV1lFsg;jI$gNkf zDTzKVxA0O4DLBa4HPw{gUyz;?cV4#r@@=^$+7^mhi_Z5&bH_mazOs%A4Mr0uxJ0uI1(i`(z}yc^xlKIwrmoyRi9!8od4)R>Uu)D*z$MKo7UQG(N%>Cu8U`Zh{hr04toqOP zuC(&=QCLt)WYZFyOXS%8d-0RnH@}9k6FwFtG>sfvvl`}A6T|d>h&|^%o1d_Y{j_9q zV0e{q*cE~uOUnK7VrXT&0sU?;*w|%nMEb7z`Ccol=#UBAgB&WNW=n#i$ZVZHu$Q04 z$}?}OX5t$Hmeh78f5(rIxi1&ZTJ)Jt@2rrL&(VrP!~a%`dpDqEoGIqL;gXE7=kU+q^>y>z zEXK_pjY-=nnT-^~2OyV0*O~i7Z^s@Mn1M15aW`J1P(ZuqS|%HR2Q>CI=-NAV{yRR} zPsP4|R`U*o2*xyPLd1^4OO?ilZNwXR(1a*t;MTQHaEmNEbSC1;l3QQp`|KyVI`$+v z8GpjX9p$~g%5jf+q`P%mzlFZR;vw+wB>(5%5Kk)edB< zl96t(`v?nGaygcTcO7Ip4$J1ZO)*vmvXaoS=3FL=ow#0HF0+3W)__ORLyxY69TP!b@?m+HZ&= zER{QD+5i6C&HmTGU4!z;fYLU*u$I2gBY>xc%=eT>NevhEYXbII5SeuILMUE(zj2II zvv7G=g16|I1TFJ;^#XAO3{+xyRHG>7zfn%BmHclrjqpM%Xn*O(|BqrXi3IYmd^b`9JTV zyXVy(koCxN62pKOz{J7L3#E$qtcf)yNEYhk z*@A;1RVOZ+-Sqle$8JC>&nnU!g~T$@UNYCHZFaW*Q(afGO6n_{Oapj2F(wY3H*iBr z86S;sD}|Vw9n)%&Pru=zvR9(C@gy8xWwbwA{i07W%I{6w>7ITAj*mzia`Cn_@T^a3 zoBl9~up-(XKyT84J(xt)$eRC0&FgmQKWhmL<`eE1h};|rD0g4AW##~FUtyh!u;(|J zbCwf02a5{Ktvv^tm^_rs#%Ep#!NZik+!lDj(YhBPg3ee*9O^{up#c=_g(Z5Te- zm9%T=#P;(?ty|{`IfMfi)TMwS$-3MpkAE1pE^vRyjXD(g#F=X$a?&*g_+-X`WfxxxK8Bex%!GWsA$d5KCdf(mqQnZ(a_7!@L1{=?d%O^v z!gn4rxPIQW^Myp>jY7?tqWUH)FT`n0-}Su|OS`Co8JWGs0nsxe%%?uRObFNwuc3R# zxdKWDh)Sol>3>%R+aXO5*UdE#6(i!^nWwb(ix{=)+Un~7c1<+XQjLeNO^gLzOj1Co;AHm{H+pRFjwYl3 z{cp8?;Wu3JR#h-n>mjEX)ok&;v3EeHkpOPQBGPobQ><;b)~%IHby#>Y;!S zA>G?V12VLuk<76P|8PCj3TaC^{&wWd2r8KCmiUO{Efyv-j^+SOcc)yVDujf!#XXKV zG`6-u&~VB2@b33!SqO57{G;Wwl=r4nsDp=w$0+RxB<#!2w7#t!jCmf)g;VexwE)UR zMfX&hM;NZ@dI959crXO(C4nZh2{PQB4Pr{+iDau2{W5taB>ccIwVmn6Zugp>Z}BtH zZ|mw#duVxays6WT*?yB^e#qt*!Cge(nHU}yj8h68MTWVlVvIw#cJj>_vc49xC`6o2 zxL~OiBN~a=7KYBko3a2DbYIiBWNaC2rzW^SiMT>Kxp^Tz`Lm8m6?S#;4U}_9RFw$H z=9Q-b+^lHexw5K24^2VpwxP{(A=>GCb%cnk zqdAR9#1s_@D~LlxIlQXhZl?6pUi(b(*^>B)Tsr@xm?ees`3b2mY3cFBr|a(x%KbWP zlzfO#as9EP-%_^aEi~9!5=MWa93`tfacX;ZBEz5~pZ@^wWpHy$Y8NL?Lb~Du#OwUi zr&rxeu$Cg`KnSB&@n5T{s`e;Go~R!>XAa2QidGptSy4bIai3=!6K+DS{sim0_n~CV z$hU{}WPvbhI6{o0z;09JWhh8~)5%l15A$z)tjwxUU)XZ*wVG+<1WV2Gf2GSWdrY6FETun~hid*ZbPYhreHGAnuV zU$IBiyXT9Y5Q?Lp8tN;Opih|_fYgrN64)sfcZDdPUJdY1AfLlWO+AQi3+z+k0s%=6 zI3&F4jfAeGw$NcgT_$H4=}QL*AL)evOvAE`k>1DhT-vot-1n`>%Sc*#>+}*mT}+y& zwCk_)cQ|Xy6ABd7RaY=+)S+C2op2d$2&R*WVK(K1YN`BVS^K;Vk2^adFk`2C&7=`= zG5+KHBY7^f1_AzflY+{lgH`GnHerKsgCMemIkY#0A>M4@%-zqg9_UsBgkfy<5j*tP zl|Bx^ID>K#$!Xj$Mc6QjV_;@M%8uQaK`{mz>1ff4)Ap`AxR#zOD3%P>@YL>#A@L;0OcNAzoYW#{j?tf?i?0$$+48VYrn!Vsy5O zN;!#1PNjMV=COB=%9)#`PJqZ0{JOe)z{ z`FP)dy4Ya#9GQH~*Y6Z9Ky3Q`?@tm!N*%xZ`Hxw8z6HIP)WgmBgF4P9O`j@Pc@<7p z7k&G=^(8H(LP|0cH)GpWNbC}Xo*rdj(i@Oa5y2rpaAVjmn9Nv3>kMQJ zCoIyQN)5{Q4jI*Fu&O;-SdI-ThWR2Wi9BnOOcETt9DcH3fD6w%4Ik-;X>CM%lY-ne zX;whXCT-f&ggVuemctwG+xe<;nz!F&GuLOwE|#)#Q-}9@Mf2Y$u)2p~g?~-gRBkZB zXVNi7e3Uj1&(f;fY&y`3I@VpJc*n6-9q5(PEfagsv0yAGf(Wu(2xx*q>>g+%2?slP zbVgfj`RpV8mY*)YjR|e*qZ$zRgEo7Z;wngy71{o1@t>t(-MJg=Nw$dJDnFrOG5ZPp z&Chg=N;?v?5EW*~*C|A8YH=+q{o4|cMe?oacozZTr_m0&D@(UwwVToi9K1dVLxD)pC>0ief3vsExxy*GT=2OK< zEPQ86?aU2hEuexwn~pXYz!gh$!&3|S z2)2;z(*2Y8NKgNr?oJ>`R$gLD_p$Mu%`>}o?!wC?ci40(xxT*mz)_Wp!s}wwK*oZh zQ~4R-2-!ukoS+XwW~%NEa*dD;QeylDbF?up-R<ryJ0?SRQT2`17eQ5$%zDlZJ7R&;b09jcZq`k>o$DK!jL1?fg= z%jvJA0_O5JR-nWGiwv!^R%WC}p}M1Aq;f;KpXT50%=vK$<8!&!?F{nj5Xp0Bj8QEh|qlafcMm;D6yGIgE!9&kHy>#wyORn;nRqD<;Q&FdcO5-b?mZRhm`- zN3{Aq7$1Hk)xhfa0+AWD+jlFp?V`ph}jy zps<6p>@UpOTf*X$QfCOSr5fT7z&<~Zy~7Yxtz~L%@+@%8)&rGfEAis9XFgnF8|MSH z5F)d4<4%IQ6=2@@j~Brg^1fEy`mmnOmr7pU07+P>MERjW1}ot|8LZKUtFpPbC^t6c z==|1Fp>%%3JS(dhppBHZh$x}PuCAcoXTS05oV1@(k=zA83J{{(K!DdNd;<`#!XJBC z7S3@>AHS03l)um}JdfM4-uD7H-eFjtEf>Wt=DoE6PGW2SO{TJQt!y=I2txG~9j>mI z-+g=J-`0MtEA_Qj!B7Cy8SGBDn(OAPr8GD_|9XF?h8_;XaYIe_#=*14{Ee|PKaTqr@7QSB84x8Wnn zaNexoe|OoQADfns%)jGR+nz2KU#|QiG_bBF3i{$g$|aG_Bsl*jW874n z6jk@$bMOxvY?aJT7WnmER|uQt6!|l!aVlicc&#n{&5_%=e ze1`W4+4s*FbI3k()x$5Os=(gu8Hz{N#co%Vk~phM>U)0HfFc}MZPVrraKVjSS8B86DY>+L6*aBT_eGRw5fj4mC z+UK+-Hx+EMLLV|jES;>zEvS+Yl-4UMe`XPem^;wK721o45O-5r2gr+it*jIVN_@3- z68qr{RrUnQ5O#kBLjP4x!;Fir-%{+flQ>=cZe7<(8nAz2uYKaB*oSrB$r?6aQzBxQ-UYHH0dycd&k9#YJ*);@WW8aE3HGbzkB%lay$zxLIeVL9 zjfIOQy&)cqINR$)}xUtXT2@5zo<^;Gk0o#0pyKT_XnPi zEv97;R-N~iRw?YphD4VP!{arAnB;-ra7s`nxSHH!Zt?Jx-1A(^OlD1*a6JZco^4+J zwM3fC(RB_be-fPsHLzYv@dEv1D-l8WP4q78wccN!8lKC{yZ<})sjiR3KGbtdfUE0D z_PXz`t9S!3bDO>mPwBmrO(`Im)B6Nqd0X-3b>DHYnH9I#0HJ2u7_0NjO*Y7ya|oS` z!e&pP+qHyAeV}jr{>Jw`D;rvel`bsr&{7fEqQKg6!A*Ssyh zPdbw-Vz-aLBR}}Th+O9#NRQhubBVQzdzrdkNI#Y=(igfbSmgAns`B;`NBg@kn9UVy zhj-3O0$bwav*9f=T%Y_UQ=vEV6tnEA@o%8{PHN+pY|SG=+?`~1{mG%0<(y6ZMsPBG z0H%lp*DlLZFzaYN5SiMvj_KkHz_ZiWY#Z>Oz_Tk~7t`}qip1+Ta!U?P<)1j`MR0CI zcOjzHU5p|EXj9|?&9%>LGWAx+!LlP9wyADg4E4hwDby!3V*@0~A+55KNiED)*+HJt zcCUmzBZHTVhnB^Kk3`sNqH5FtF+k40sne_XK>^!G%hawa9$LYYBDA!hVBrD?(ueLe zIz+Jrm!;+Ix5jH0R9^J1YA6HJBL1*Kl@=*lp=G_cySC5tKb`VQkLNe)E-T=)$=nOz zPN`t>Lfl#bIH3gs%SHvz6D2J>dQ*Qzen=Y$S6dBWZI`B&^4~uC?Zb9XkJ=0?^f40P zerq4zn;$rb`Ns9>Aje8^g<-0)d%hjyt_#Uqx*G0b;4o4*E|rVurM%huI{`(h2Pn2F zbxx-i3oCgdJFf)H)_klk^=trphBOYIwZv=zvVL6e+;S&;3jJ3t@CGFkRrwRzlunVQ3ca? z^cxdn+0qRa-+wz1wEf|DK&3V?H5G_e4jLyhE&rf;yh;>?1XAmxt?01-vj zQfTYIc2M``ye}WYa-NmUJ3+9mdwmrLQs9%ZWm${Wab6{}Hp~Cy8y?vgD~x(|QE_LT z@?NOK{_|^L`Zug6BO;f1T4b)MShQdo^aq?19YzzGNJsiPiE0^l(1!qs2% zv-|bh?Z62($3 z4rQg4j)*PqL5Mpf++vNI3oXvKMxU3?6OPSl@ZVq0ks3mimSa&L6RFe^(?@`D$a zTYr-tQuCy`rJr-!6-UXZIx0Gz{=J@ZjBH>fXtNnd>w0(W{HK%^UxC18CK=n#S6O?q zLhM+SPePE!LIt0^Sl&E+!~?+DcjDwOp^OzA(rhAHD)MEGlp8*1%S12iXA&E2=(G~~LN?Dr>PE{M%ebc^A ze91VDF?S66VT1YHA=y1;SX8Tm^^pY+zk5pBsXCQ#=&IJkULpYv$DPK&uDw2T!8vE$ zeF|e9%zy&-G-{+p%Nx#G{`O;)Ip>oSdgMoNj3}l@{b?FjhQdJ_-L0Ic&{^736NU9! zH08niSK5E+t*TNNt+B-3U0t#?eS=3Aw35)NSL7q&*eb*|zYPN5YZ5~fJyjjB2ih@c z`(MF$QxdO^z|jUABm?p8Ot^Y1pjnwrkZ5~yA+gC_NM{$W`jx2K2QmP6!*2B|7@EV4 z&9feG)f!!D)^S}aY?>9de^mQF_GFK7f5#7rI$^Ai=#>|D57Bp1q}hB)7IzMEBs6i) zqb7fxSKYlGDO=`4*toVEGejb~VIZ~=9gP4?jjTdHUp`uB)?OjJuNgXMusO(u|%fF;o zk*l{>=hM~%Kyu}gPAy)Lg>P5d$qyNf(lsy(f#pM)dC1r$6t>UL-%nG%O**(7v;DeL zqvG?%UY>UP<_0D&xQHI73XzIYYTV*0pt$9TjQY2AFICH9~pP z?*ZmGJ4OW$2G{EyZlb9Ap+GuB|J_Q*$>*jbCZ>*c12C^Vc`D@jg{iI;QIEIYGD=nS z%g2IMKSgBkr_!lG1IAh4D1sA4#QxK=cU!gBZ%Kg6iBu+yXGIufbY;L>=hW-73?tA< z05wVf+theW8^6rVQ)iAK&bn718D~JAfVC?u_0F3)Ay=X|_c)!!4k|p5$#Dhy@f(2m z!TwQ{lJNJoU|oq~ool7cqs3X#&e8o?Arx$*gah$~)=gRsTvyBQZQ4j5PX3)AQBFx&qdn6jWnE=`gMZ+KW>?cx{#D=4 zi(;OH6=#7-l8s9W| zocKWO*voGUEiZO{ZNrPH8oNM6`#Svw=82b_yaoXQ-Vp_4_orQbq& z59F#0Da_S%`pJnNJgCix9rFK3HAu2EvvoA&phOR}VZ^o>cJb3AMjswfq(S+3GpFJp zYfJ!3i2_K~WDf9LvL^Wp#TRuxYcA!J5tmknK``FGvrmeY}(mcIbN5jzakQGJ5+(lZeA ze$}kPnN)>YS9wXx;s%GmrHDQh?S0g7qt-mBS8?=axjkkBW%#H1tedtsl^1*OO*9*G zg@B)Y#OmB;+YV@#%=EnI&7)KWcT(B66)`8V+$38XC}%S~2X>*7c%7>8Ke?^9&CNCK z59xVmD$`<&zI$b8xdAiDMVfkm={~tRf`knR!&IP)50_S7!oBQ$N*iK+j*ZR(*qYI8 z5R$LQUK?TfX^{8_P&$!)7x@4e*$Y+5RX^4FbT~e~Koa3@0V83Bh%7HKp`ZscA7qrc zIkA4{!xdk|ZkU|6c_n;D<5nGO@^bHc0#g;|`SD*DZKMh7S!vXOvSt?OSVo{&oh>Y) zvc^~RGD8sPL}|T@(=;Q2cu<7pv^A9wymTKO_H=Hbf;6RI6+F3X9*vHj(4ktNDJdGg zBlmmr0x6q({JUo!O4T1HD<3fz9_r1F!2rq>0&Zxz$|@&^%JCtCXk6;R-kEDzm(&6K z0WOMLX4;rt&fF=WSwgn+e`P&P30G%XkM(W-hs}K*|7L5GS*T%V_Pt6-sxS=D?XMmS zgwj%(otFbp6#?L2L94RhRgQ&*1miOFD4}q_7sXB#RN7B$e+&uYg=c84x*!eHy=;*w z!1m3ac{_O7@-XI}V?3mCY8MI4a+e&QF99+oFd#yh1jQ7)J7+)uT^aSfB%@|_xT>Yf z?CepP=yjFpz;dm){*d;!){Mz$Dlf4ULqv z))L_26(kdIGpOw&l4SU1cgqdXoJ0oU5*v3+D;-j$E59c$wJ?3m;zwAv5>fvP5lBR; z5Vbv%9lW+8otV|An0pB5W9C*_VkW?66K_p8EpIPO%n?a6Y$Pyy9 zU7t-(Sc)Bgtv;buPYdjr&J7pR=s@DqoicdD@X=j1O-%ao@WyX`3U@v9LNA8%xGJckw8m~s{YPGkc>hpA zz^%`8zXmO=U-|#S@)(RnV8Uo2xN0`_Cn8DDxKFLb4Vc13e9Nc!jHpn2F(DEn@m3o4 zL?6rwPC0oTxuVVUzuJEwHu%WYVkX%%uX+N&yj{NnV3IM`|KPri0JGuURf3b-S(<_+ zIl67J@N-kJyX_!DQZxWcD5)^`s2G5dLdZfUmrG&1G8<&s z5m^lua1HR8-LG?ur-f#=yiAjw*`mGiV8pj66b7}~JjxOiP~HBToyou#=k$*2>YWlv zd(dIx@VbNNPYfDA=SrDMrE3;BY_X2qt*#X=7Mr~$)9pc*g}JrL5^3xqYU^ZE3HbnD zH-{kf80Bo1-H{g;1@N2Sy~J9Z@`X?j{?szn51{P`0kaO3(TI#vY}6oVp?R)dZHr>o z3bJ=jTVlg2)xdB##~w}}J(k(g+)npqy5a|nE6V_n`4pR=M5-kphwr9TiAh&fw~awb zJIle;B+9`ojVdVBrlZxAFi5#D1o(e(A-yTq4*rItOS1Gm60RD_Q;%$mlqW?g{Kff(ov-AQz7czp)J7eeXE2X^Hd>NSpY=w@&3PPWWow*hNW};XFr= zk?b0<|q{7(c2{d@&LMDeUG*=E;un_;r8rzx&&_<9S9 za||!qu{UxzL{?qiL~?wU+SDQ=zCRdk?@0eochG@uZ|qhgTT!Qum#9$uT;LjrGWZ}F zw$ElxDeV%#_+6OXE>r#%K#+jT3P6#pOgrM2 zzq-I9k#Px8ipoE`6!qKFEvvg7vN9YQ-<*&!|FPxVi7(t-@`5|E07TwJJ(BpR=tW|2 zvhx4`+-ISLM!2?KhYLYENvh@I0~0mw`mMB}W97!pAWudh*8<`(bOX(61rm}c?}3yV zjj9FcGBFD7df=IVqbg`(fLsD^LbVv?YnP)6hn{#*;av2pIV``w0^h8DDybFGuyhYt z6z2aJ|B0;1>@uGOWiyczhxVMWe(Z9^&aAuBdODIEBE7{T&utpcXnF-C4!6VjdHqRsEl>@w`^cGoED zp1wl8(}1ub=8Pk5Os5zm*hcu$E3&O*=p(YPc$RtQoXbVJg`NU zmPyF$PKL1=+a}^l=ORpht}dh8ouf0w&YqoH+cE*v#bX9o$4hzCGXOEUU1KxIxrsm zg5DE7VlJQFNNIX(mwOgm{zWv;YeX>%`iEOH+eEWjcrYe8F?<)C)!^TTQh_wk)I;~X zHNsv$GX!yzTZUPxdi@ru@w6g-Jo$piN&~^*S&~AY{@(>L`2h z4e5jmwp2OLlr}c}uZ!Voh9ak=+ccd=8&5mMC1yk{0Du*Q_wunaE&{c*imeT1G>W&t z8+tN_ngK>$0M+QaZq|Ja*G;n~VS49}9f=CN+Rl%6;16V%OV_IS0YJA&8Owag+iX5$ z_eI>s7vd%WP3Rn-he5g9L08-t5VbJil60;*_Ac1XHw){9@V+G~Ibw-Oha2)3Tp5SZ=^4Hn zh4>~`+D6ua+WxP;x*X8`&Wcz(YX_Ks^ua5FKQ3swF9R-7Y)n?{g&9pI@(`8~C4OEA z#}4TyTjXNppO$P8=g?|8Eg(0S4`Ib#!T$-81j+LCz@yo|IhY?gf!RU?9zJPs2-a(CokBF9fW$~&`(^FycNP5%Y7K<>XrF$J7scb zW%1~qYNT+9-N6)SHB7p2CxGW)Gn9f+`S|b%x$7-+$;mQ3XJzV99A@)ZuCI(8q!T>L zP^n&KmsqEQ6cnU)@e0GjR_VTA{#ngn|9H*K#4JZ`fgsPKM#J)WM!|n>Uz>T=iBlmF zuBtH+$m#y;-4I0e)-g;&?GZ$6T<&(mH1Sp!^`V6F?o90hT%%T*L#ZZ+PHuWJ+WXiS z`$cRxSHq4`@d3SJPdO}T19^dZmte{nb_bfBmjRU;P6bDZHlN=Oo@Ep{DWXNSVSlK^ z-3u0LAe>SIa`;T5#I&>>2l}qA%xB|#CIJpg7(*jhQ&bw9R->lk|B=xxXWs-|ngDvB zY3)h|=8rC^T=E2pbcbYh1`tCX$UFh@Y`O7)O(N>UVq2oAh#}I$$z3ht2?(6Dm~avi zELepJ48w?k9~4Ie-Q@9Ck*Tllyo5KFhCIO~Lf=F;{*BH-t#73D=bPGAZTlXA!ja3T zEMYFqy@KjAe0ZM)YSS+p+;?c+~-HoWqdj@XihwuyQ0{R7aTPBOTNywfp^q`;Q1di8vu0&t;OC2LQE`_U|U z&9dk^e}#QmM!}o>vR8u%~0&yOF;LMUxQGKtBd-5y))=RE|x5 zS8B@28a%EWF0!urn*hw~#PXlKO3#`*L{j<7*~6f>4`8Zi)5hIaf3U)bW2Tf!T?4fft@Jt%d#HYsF7_8*T79v+w z$l4hUemJ@|l8;aI!*}f0dDmrukTRS0P+rCB8BCFuxgBJwzVq*!y{#Lv(6E)yNA5R0 z0+WaX@mvKoGo5&_bp+xG+>$GX)s-6-_o}PlZ5|BaX1rVC2~557Wn2)sG8M6QkQp%k zv~K;zpGvhsUC*fvFD|87Z*vIu*TqZM?MCSMh;3PGYJTrg9~P=oI%x?G-(xRNI9bH7 zRSCa=as~NB4_3Psr&{Q}&X8!l%O(1ru_6zX-44_;F8Xzb2Y<}=#249`Qq}o*u-5x| z#?;*5!mim}qz(rzD+cd%zr8BJZ4UQnBiO7FyyKF7ZF$?SD#$s`6<6RE5<1%|9o}71 zG59{Lu=$qHBppS&id9GA*zf4f7NRv{XW>d)&4k-kp^#iRdJL6PoOq2-x-Z-ZU&tHV zzBQ|licUm|U#^RbtoAv`38NHrMD2qZVB|0{)v}w*H-0923e*7w$(gA;uSLti&CG9!#DYHMwR^@qL!d$&ixSQE)$0pbh=B^c?kchI?7*CvPdqoiX%;`xy5K{ zAe02>!pDsuYNe^5^GZC$yMd{a_D??95wy~#k;uFUZWzyoWa|&l2Kxo-cdeT=Js{eS z0njjw2JBZId@`v5KGzC#?Xa+Vvn3es-9snRmJBqR4|6qR#>Am|D({AoqsAF_tXB*1 z>FhivHSPt)4o#E4w5mXweSvADChdwv>II~Ofs{gLQ71p8fv$>{t zv*jj|mG@ke!}?JS2*ToRt$QXt%%SRSD5wv%M&ESb30D08?Cn!0s`kgC9dH{$)2XN( zm5e4BXqnjIPU|XeDG~KPl0vN`#`9}y{Q*lw3D5lH!}iN$upF$so@mFGS6syfp08RO zp?Tg)!HBIVU8{5WQcZAANk*A9UV@kKzGt{cwXodW0Z967lD`z=Gq_!;+kiDsuw0h7 zmu}3v^6kL^T_Yi_0@Ae_bV|45#jVBgEAuXoN31rIUd0O$rp%fkU%I~ z5z8vmnEDm(7+nVv+v$#WnOx=P+;1dwr!wH7J&z%p*X4(DZaA3l6(2rsjL*HYn>Xif zXTlLU&+R1G*(*w*X-@BKrDXa)le`KOwE0hkwXxbq)eXgIFw?16wjHpDA0kj2=%T)cCb&VZ*OP}F!YZ#=G+svwhFiaZ1(y@G;H2Hxra!)W&Fbzw%%03%DK zSp50kkzC&u0zDcd`L)3E^?XK9-PBlx`F*@#3llDgK15vNK{es9 zr>R~a(Czw2ajRygW-vYr|El1dY-@U|1IsgQAH|=2_ zi9w6}!%sms^RjmJZTxLEP-mR)I+i{XEO;%jR#hjb@N;dljI?ZtmNKQ64xrI}E(ceW z4C27cw*MY1BnkUYj_ad%(fH9_T{lB8TpeJwRSj2B0KO${6?P zK;38w77c{-hJ1QSvJ4YY$wx{$!O#uo_KE2Q`?;04;(l+%TAikRt1mTyttv&u?(cIgTJMvy3J@a4E*QjsOwvo{QQo*Mr$GbwR z{mwj+UlOKNOGQ#o{+C+?-emhR)Kwv!Ev4tWJOsE~3~RV-7Wsc-g}-VoZQT9+@Dc0M zm!E6+XeS%G5C_G>wJGXTedvoGN(HhR&YI9cxDF3ky>hq+1E6axB%K;N0&H~%2__w8 zCR}>wBo;?>RXX_=2`^%{Z%`JIo35I|jNT#E4Y+8_g+=o2Hijmqh@`^ z$3FB~5{i1d%W&FcTnfWbAFQ89l#%^7XfIBEx84}+U)A>k(#X=MuK&`IDiCSX{CPwy z#xFxpy9U}8L@xhC6>6}hLO_Mp>4ITx}*;Y!sFD}+R%hacZ?u@J3A_>NX2#OJ_ zv?H7+ZJM0&mwX)tU!5rjkCVFw+z0HlcK zvh=h%H^kP$fsh7BkKK1Sq|Vk|R_j(T{EG9p3byl5TEP#*I5(z*qQ4$G+)qNHvcYQ8 zS%1l7JM{n)E*S!>s1abUy9P(GjPXm7!Mu;ZbTEX27y4x?xd%qg3+N`hasP>M0LS!T zHH0aLFs@Mlq<4Ps+FB4R7vw5G5B8e9N!viP=62~}2BcbuNJ3_pUrcM_v8FEDiYx-+ zv_iN3A4&9o6dd(BD1cG_x_?s~7Sn>$wt`0IOmvF2k6F}t*%2&ldF04RV)(X28f%NE z4uZ2r-h7Rz#rBHkiwJJfshjZP=Xp@e@LShUKbC<7EzaGmz-FW)C!8vB;i)(t65R3W z+9k{k9_SBx4J@rleO%6rAJ01Hv z6acs;h9sV)2X~FZSb-P`%{*wL+`yXFne=Dip|4Ifo-z`-JMF5= zv4d^Az|GYs-L5vY-=eUeW})9rL~=3`^XG&Ol5~kif~tLYb$7J2U2(i$l1^+JDLm(? z6(|~UwN{W+%{6$Ep*F7Ox2tDYkQ_GYd83Jutm>JkSrq(fAyEB*G#XrkfvM+XICdQ} z+UBgXoHf6DcyEYBVg<2lz(cr`W*{8IEFd`= z0%X_+iF6k%!Vi5x%CM`eCMdt*ml?bo(U`X!S**K3(V4Tl^;Wr?5$5&+8} zVmfL3;7{urQ}2IT?`g`m9IBYJdvPw z-77UJu=4#B*CdXmYaQMidER zhnM23rq0(buA;Y7lkNe&f}${8x5{$N!Cm0E4v>oU+%w|7?LfgL%d z_n;*jmU`=mK^O)o1Mx@-e+Qiz2r9-mcQ+{E?VVjU;!NLb&s|tLZ5Inm*|>5c{p|c3 zB>qJpZOa62!2|v}jO8uU%=gfnNLKfy>rtus0ccUMbwwZ|y~?+6ww z@go1aj~3NqQ!aoL0cf~C1OF5*mloY3Z9&DCt#V^O!!HtuadE*Pb~KAf7)H1vwTItC zv@}4zKf~L48T@tFRPp%1IZv(rrT}tGtnHFO+T)hKb$ZUF`n^BUG@`%J^BKlg&7_Wf zvoEAdH^Nmz!Q~&vf=9n3j}|WPE0&6i^2-dc=rV4yJtPo?;#MZgBYQryXf*JQp5g`X z$GKJp;^ts&T;&Sr#lYufkHeA<#ZwC9YRNBp8ZuhH6{&BqYcZ#tvzMDu8WcPFE06<7 zXYoC26&{h{1#`_^x0dZP<^1;@p7t6u*4Yb@kij!z1^oa#&SQ_WtCx~p2Af4+*>``P zjgJ(1(fbDaiIy$8Ya^GHUJ`K0c147G%5lKIPFbPWf^@v)HanKnAM^84>r*mD2~q3t z2G=2*oG3$DY!uL9p&dL)A+k>6xI_ZoQSt-0N~Nw?pwBE3)yjQbvLK!|U|fkV$RdEX z0Mprjdlq9sw(npZX*KRwcg{xy27eW(s6lbZf45Q>knW+4YMerpwmhkG>i1M^EfTEM zvGSbc;t;|e=CU;#!Fy3%ueaKRD&9)NP>ltVg23Zvp-$s4Tuc9r+RyEWuzS+HM8g_w z+rES(%C;g>Q?EZpkNf%(Ewmt^&b&)tP3rnZGg&uf6WYe?rVOrDqB7hgX%t+1TG3sV z@~tYwI&K>0lbX+|rReLD)9No=y496T*|SDr2h@H#JP0!o2Cz>{tolM+-=4Jp@2zY8 zu|Ct#Af{G0j~Kf(;fOiut-hAfqubv#;=tHt_I!)d=@W;_`n(K2BT(q&mA+jl`B!5P z-SbP6yRSFei_n)z8p4%X%aQJC7EvmIr6L{PkHfH{8Mn6<;xmHkx7@ zLTUHehF#s~ep@+8SpJ9#@_yk2J5*y_$j#Arn&}5=3fGasgDVg;SFt^__Oxs_}RVNK{ zXIK=~<&P#Z-=SYghyzEDy79JHdz9>oLbhP?gHo~L{Bm1Y$j~)5SjmihO%Zrn(n_MP;qBvM3qVi7c z(^)G%X;Vikmt#yBy7&{qB+&mBxg!V*#4)+{oMWOdJijXG180rn)1V{*!GRs%bBks8L{KlwE;yPiH7DNk;P-ZIbPv>XGrk0a+pyr&OBqvaT^t`z*ldbDn)yt zl5h5(Hy`R|SW}{=6VBP&U(Ct8IHQE4K(j{w*7)7Leei(kw274AXw)5HDp~kfCxbT7)K1-Zgx}K}i5jt3C~+L~WyP zWnbW6`uPe~h(&Ig{DSW2Z%HwWT^zevT)Xq5;mmZtWAvWu}h!e%N%(ZwP zLg$+azijvv7jSRxL=YPA`Q)@3p6ph;r8WHnByOUC&<1Ufe_XzrA3?Ua(s@d>wf>M_ z>StVZSLKROwR8uOYYd7Ug*}xY-;7MCjJP$7Wa`BqoHhy*=oJ=+i4)Ta3I@>1zLnZD>9&O`b^ji1u$E}PuZXMLxGo9feV1|r(^mb|be*0fx`e!DQ4 z+RCs??3CPkI}VEhkIZck*rNyCQ+lUv_tr9?j!53ywWc;6UERjr7Fr*1se*})r0z7X z;0hO~x$uD{@kL9T&hcoCX$mIEA|X`yfHL) zvAk7*=$9vfQ-)!)^t z>;bZ2LCpxD(JBH{-_&e5WilfZF=Jx)|7E<*!-WbI^1rh_-TzO)%wpGSFP#74} zeR3yK2o3i5aS;9$FB{)X*4l+;lYBMt6K%>>?sQXHoH^&rK z^vP{Vkr9C`e_0hYdHjfrcgnRcm4V`#WiN&8#?c%$4<94p4?r?8aq})+86H*Z)o^8g zrk@rq6umw{-aL_F#$vUQ-w3=BPUxv}IGpozGoLWuVS@oq2(i@I#X?eA?M66|GhdP_)*}cm=LI~Bud{{%$n1q zUtM<+Aj_Kvty{{*3_$g(m-(Nxh=R?|r|NBr=J09R$6NKIs%LT~Xu&fu7bX?~cCYLu zw-^Ft*KiJ8W_25r!_W|4Vn`yT&k4+V&l~H^N>z?kO7D9P`WXp_*P^Lg5_g*Kc@C!# z!Qotl*@#&&p=$}=9FOh-2GM}Y?YK#$3He02-vb*7* zVtNDffzV(IW#w8VLeOa_&AOnDFLF#Ji04V-J3Z9iFohSvc@i*mP>20?`+*cmc$$g( z8zA}ZpC`pYHB4`b!_xf~I*8eRlg$ACxvAFb&i0y?H2n z0mGaRkOOB}6C5u1!{2^13c_ zhwBumn#mj(6C0nu`$@)Rv;315*Qh+Ar#nKvRpS)!h22R6PYODa>gsVQ|zsM2VovRJo7x~Wh^l)B|keBN|r*jI~RZFenrLrOYD_*H7ZP;?L ziaaX1JwsBB!SxBOgCrbX+%#p@^mmG}EsFb%=DE?>u1xNTrR)=_pcD?|f9VhZo(n?( z@Et-48rR8_uwvKnovBWn^zY`96|{D};Uf;q+4G`T3`@X2!Qy-mg1S|%x;>AdHMHC3 z)Hw0OCC34tcPaxD)RU*4vMP!R-7P?-riBOqw&3%|?e5;4EJWh2HImC2A~h;~^2f$|6Jo2c2zK9PS-)7`!8n)uRp_Ip{7&jX zf*@q<+p<12vEL8HVV+ZNrs>#F#R6HL)X_2s0IK)p^Pylt4T25Kf=`aOrnRh z$pj*I0DJOY1vm*tH0$_b?Lc=+%KG9k_20S(VTv);yhDv{-YI8_8pd?nzkIME7dDBA1%*SC zk%1bzWXBgRegH$?>3tgzWb5j$!jH z$57i@wL7P{cWpQIzBeBv!RuSCpA94Y+jCXw=hHo-P6>N-U3Zi687EP?Y{?PFk$Pq} ziV1T!c<7C!S+#?1^|A@e6t@j5qShm?F;inE@>*dZsfwq4ln|Tf}h;^;IFl%;OA+T9A8qhSvtVTTu_EaDTL z76Sy81IGKg4i3&D1J_T**$0k0_+xN@Oa3p9zgSHzo3=#DY>qWX-g}v+FvqW$$c$LK> zY}=j*cZzTjZV={T#q@b?7_ff(cm2(j`(cHk8sR`3UHap#d&pnJ&Z_?coF9(1{TL-~ z51p}F%2j+YRlkgfi}IyKh2SfR_gk4i0lK#<{7dPInG}*OB&Mrn@wWuZk^Gg9K{zVn z_+o3M1|!m9f|XYQMQv%vQ0j|O2s0-ohU{lAQs5A_S>=TCZ)pLqjhy=d6;=*s7^6MM z3j+sO5Mk3SR+XE5Glo`5{EW@@T&P277i6sbzHbI)_Rl)%tlY zeiu)ps%Ks%NqHRUM4fH;!x6cQr}&7e=ZF2E?YrFB*CfQo3wLLGa%F@*sElp&s7S1FWG%6 zbG~=l?2%f^0==zd_diORZ1cm8GSabk`*~fbvh@`gIVGD>E`cZxi0tIS=bfhuPCS%Grhlcn0uTmGB1SXhw zqfz@4MOvEkR%}I(D5Tse)o4;ufVw0v6dr zSbm%-$p8?gmm)kl+k8EcEpJ{3<=Gmo{6vwRAtrZmP#1;qUO_pQFtj2ZvUT8$nlU1THTz$`=Dy z`F1l7D)e5FgCk1lbyJ`8L9{k^wROH^s{OXjISX)QlwpG_^G>0r`He}Ve}w|27gumE zEr?j%Vd)a~-xk<`z>8MVoTwXle*t}24^x251p3lLYbD!uUo{lH(Ys_%3kyS<+_9GG z)<0j)8SZ&uWu{1`ctK=y`b@sjoarRm2XKQ&^psBy9eHLC@hF3oq4VnRi$5MNwm09O z)T)LmrnfJhynGy@B*@tu5sKq~Qltb1e>(zU>=A6HP2s0+rYA4Zi;`Y6B0)3z72CO7 z#Bjo6a?VVb0<%1(vjnKdMRk=B3Z4l~OJPXm&Lb=T(4r+^KrnXx?YE8fcep0~!;KE% zlJx(licwM*^N(hJG=vK~iJ_YBN)jeDte0nzX!fEl`(3V=4$lHB7Hw9=Fq+#fq;9i> zwIt+4V+pHhNpe!3Git2`@OuAI`=v_A6uck6*mrJ;!HZ|`>~Oq}JBdwZajRyJ0lcDFW$`Y=-3hC znP>Xa(%MbZeHAie6iFz<-@jEY9(vtI?|qwM!c&v)_1q7rM0qhMh|+CAI&a={OljwaNFf97cWw=3S zhzb!ASM}ouv@3#o=(pZp7)|^Yc;J=Ikbwt<&6%EK)rRpT{D?u!l*nY-Yu|j!)Sml{xEmx$0SK_)sP4+;`NN8 zXX&tY(epJh6R0vz$ID;E*5J+s3rz~?iMN0QBYf8*2H;Z&g9lg2*2OXpn0IRim6UpukZ9&LQ?IqfVKvpw4tYUX;#is0Tr;MKH^lG+qNG=6{&@Biv60Jd(*tV>H+F#w=3hrLP0YhCZ+ zS)NLpOI=Z+B2Z{lXOyb`I3p=T070;ba_QxS4Ijyo+gqUD1AhivruO-S`%J!PE^*!! zw8{kkkfZS$-mcTtXpamN(nl#8z9$>n9p5ldX*g1bk$o+sBpVi8!G%69fBGvxWz#jK z8fd+D9z|s5ysx8pzb##|ei7)Jpg}^S1&z>@>Md^k1XXPpW1k(_L_o_hD@{Sp;xHg= z$s(atF+<7>riH`%Mi#wx?wp1)m-zIkXKy$ehV%K>=TDS1GbsfIpxoi+v(fSSS`SlP z*S|no)F9IU-Cn2F5MDPxbo_-Hd`ZSRoh9|)$VBt%l``Xg-0PiCI|L%eHzH4d>bRf* z6J;9KC&h{iBMR-;%;DZK1UWV`H767pWFK4pgf|rqkkx_59$_Q%d%y>Q8cUDX#1>Js z2Wto*WQ6uLN^Js`qF#vfS$AY?q0CUYz`tWv61va|$Pcf%fAp@sCh(2M4>6V&h*;XeEhHU#@vSeb97=Gh~|mE!?~6H@%giXc8f7f{Kr zfsVO2eHkd%A}rohAL!urt*5})=Q$NAMymZ%N=m24deM0$={MtmIQNmzNcEl5!_)+6 zPE~lfE*rW9%bjj*po2s+CPBqjU1^*gNn~Z*yVF0FU`+N}Y zy3F};FFiX`WQE#eA5=_w3Zv8cIqhkg_)8gLP|bRN|K2K9bU+x1cp31=`nU96vkxx3 zBRb;>!IE9}ba-&N9bk|ZMf8RN%mQ$qKx(_IMrY9v_0fItsn$G|{k#I?hQ9XNXwVbq zYXlcUMf?FXqE}JLebt*CVTb~ftGr&Rv!|u|=qNe*MIS6gRX%CK$o+5<;F^j(8wR^* z!m+z6KT)9g$pM(p)#=B2$YC zY3CWo#{A9G$=y%V8c~~Y!WyDCj8S-RUe}_iSpE59>1Ea zMM{*{66*+e!*W2|GLQ|xJHZL0d^h6EVmAWL<#me8n#WhDUfS&#O<6X=~= zyoM+yM6Qfp+TBW#d!%O#dp%;nV0iL`{k4tc?b8N%64564SEYa3CG;RP(QG%cxGf}s4+Rl!iEHQPU}Ec zcH$0{-^vJSNF);wry0XFD1wUZM8)i$@`0K7lCGSkU|e+5R%~*AhVnQqQx|~m7Mt3nbjVf)S{1DYlfqvv+ZLJ0MPYQV+e<- z(07Gc2y@*jIn{!S^e zD)jNKeW%&-Y{EuN!U9rxB?A0}s`0hQ}83UEu;NWb9MTEf>+kdAF@%yzFRLTY{v*9F?bI9L@G=tq;Q)&XVg%MOyNXg2@7Mf}^8>#@K_o!+ z?(QXQH2apRI?xy57M)`pT9ad-sn91eeXTyS=WXKr{kI>XT|X#ScpdHK5H`vS*1lk zdus*H?we(UvZ3J7qfsUADh)K+8_UWEYm|It=*;!yX*Sz31}ODURQ5sK9Uyv9J~94? z+}2WP`~}>ae>(8oAkT@lu6L9 zg-zNjiRw7Z9}Ui_lRauaBw{ZwIF;3@OaD%&Bee6Wv18nb&NkGH{X5D|h_fr~&>x!v z0r#-7YNbqMAiK1>IX?5q?etaw{ybV{Iz47v5e{(mY=)|R6;NVBD0ZiM1@{W_lpA@? z*(f%OUJUxIj6txupln%$S-kh%*KPm2zdpjWEKMW5ud#;jPHw1P425Gh>5kW`o@saC z>rSnlk}jIh zFs}Sg)FN8OFuz?)*$)t8+!e8+2-dLK_4tm3+a&M~hJxvmZ=V1S-4ag-5_;Z)7$Ua~ z^rE$8tqywMqBh7L&ZtzojX^5r19rDL8H6ww09z_xG>q^jz-wC z4y&m>E>AYD z(QwDoPDF;~x#f{$Yo`f+$?-LHG{^PA>5vCpc;(-K&rb288(rD6+cxM za628u@-IctUY|vQSO+OXZy06EU_Ds`Dd5Ka}17{1KO)d)jenM#I>)*f`g;rJbyFMVsJVlRCQ0^ z%7pbH-2A)Of>gKyBTBb_Z@3YN+I%f1Z|tL~`$`NpmU7NyPR@FhBbaeeoepYJ7!||x z@x!zC{RV778;AdJAp@yMIWykU6yTXyj~5+>57zo7zD&ok8?R!2(jsnu81T(KYSG%c znDTZhuX*-j^+cl^n)BlW`s8~Yw)Ej1(ZL(QJXE!D-&M7zO_u~UBor{d3^DhA!`V!l za6r#pNnrKecRNd8$8DG&GhZcv;V-`SO5bPJvz-=z!+d+t2pVNL>|2H!8DszRd&j-4 z>VJpE2!io;F<84#{g5=SWD;GHz19q(Qu>#A5b!mkar~z8!iIX-vUCE{Cd&X+mY(^J z@QRX~sW6{8t?{u%Ud~Ycc=p;6GL9`{LYbrOhG*3`R;|&>DK@yZ0&rGm!%)L67;8QC z^dl)G^SBx%to;sRmEk@>%jkz^``@bgd6>2(MbMmMCextbG?LuA#_#rjh(6qhO~zXT zMh)}H7wHf;JZKeV-lShG94Dz%*4CNIq%Z6vM8H#V-tg{H4v`>fZ#V)q3BYc!)QMFE3MPfS-@%f*mQwwBb+T; z0J@G6%QGyzfMv|x)VQn`2y7Fy45mU&tXyY?uB-9{(R`Hp0V%#4x2^M)GQ4r_-`I%{ zeMS)}*W-0XctvtF#bgL5pJM2p0IkZ`?%7ppu888cJWH~)vI#xyUQ`T!Rh8^5>9L}| z#O~}ge_GMVgVtFWJjjBWDU$xyoh;cmVbw6!f(*N%8hP`*W+MoBmjrhs2d4Avr^>C7 zjT__p%XqGTL+pL2(5V~U~GO(|39f3}_8m*W!wEvJs=tFL+ zvvq$l(W+!!J7WBSJ?#{vO=Vk|@wysNROe;hwONNA&=pfkK*ZsT)8f*Tk7I^sv;woq z$(k&ozk%mSHr}uEF{@RF%c$|e;R#gUcd&|0k1$~+G){ZZ2;^E>f?y=6OPshw+FO0Vh0f;j&AWzJ!^|^j!cohgx_UVL~r|C&drBaFNF6a zH(s1}>gV7kEu_bZ{G?pSMlJ>lZ5-AF=WMqTUbf8p69=@dA#If!u8ilhRwB@pvZ~Jn zR;VtH<%a60hXVvtUh3OV-3|S{$mPWqH{}t7#g>@7AEPDN)8Nh&YE|2F+4ukkaX~h2 zyVJDT2Hl4A-@$>RAE!yrhjUGr%sJxewnzrf-Xb}<#vl>89zn7i2{s`U2)kzMwP^pU z1H(|uH^*xx2`5g3Lq%@|@UCYK`?gMQioJvH6asR*$+R6s?+e91C|KGQP+5|3e5F|? z9SJiXnn5JAoxlJD26C)ZcYXKF5A=MYR+-_L=QKP$*^b}SeI3EeYXxOQeqCuCb?I>s zhnzu0UL)sI&?dIDAatr6J&Ye<4AcGh2=?d{JQ-=Ktt@X<3iB%d>0io9QK^_MGpUtq zNERv5SYzUE{@&F=TK)^_0z#fCQIDc`xSq@rz>29^G`GmGv%w0~Wp7EUnBI5qNwi{B z=f*?{mjU?Ifv|eWGW1HFlvr3cdLcf27@%T`5qQH4_k*?l{K|@w~ZP8JOO{OW2`zA|4oH zoh3=Z84V{685N43ZlO1P&XVwCiro-FMr{*Fd7zar@|{V674znojI-M#WgR;DdaKSZ zRFi@@4y$xRmr`SbWBqhNmrQ+3i|=F=*A;9(9}6MMx7YvRg-NqGu6EYdx&mKDuWAqU zsSa;F)?5xx#!S=nHtrRw+SM;?aL8ECO;B$#^>uAB8i=s~4efLLVnBil6@XK96PbT1 zoZfilUq_O?(QLZc<&?fXoyS+U8a@)Ysv)phTqtKm%}#+lk0yW`N@6D_BS^4b36j1L zJ+3Tl4UU~p*8WP<@RNDuzo#U|LwXuQd%m^X?Zaf5aB9e^(aL*nBDkZK@=_Y zLEk-Q&rwn7-m+@8L(SaqDKFLuvj?qF1O1t@Bw zD-@(*=kUS+4-GGlgK$(mGduOt=sZw-eGBqd_b2&NxaIJ)tKAm9ZF6#nI21peBc5(& z)}q>mGw$0A9?c{ri@z87$I{Wsr`YpEkvmF6fn5kvjab%aiEZM|bQ^v1b@A|a>E~EN z$0x5!w+k!gY-gR^m(;R#%5Y0%>PJjm9bWc+7D(4WIK>-)p%`p0z~Hn!Pv}&+g8QgO<3)s z&Jr1fc*Or*c$j7J310}`f{NkW#L}}v4?;>#jzKs7;r)=rW1B_Wq!MRtlapvBWyPyW z+GfNhlnWCP3ukr)(Ok+HCt5)M@D2c$2f5+2d-tH7DMHF>;a3i;J4K+g_OEM*j93O- z*6er19-lV6J;X#3YHEXIEJ-k0Q&=0@2~C$f6=FVSLbb7j0~NUNziMOtlVkM|Hs<6n z?5vPVRQ#4v-s2bx8sUhbk*5f$IXhhTkBuBft|_D`qG=zY)8Usn3ML#fgXxgt8P{FfrgYFCn9HV$0#@BOEfx@+0X|US zTTGRLuegAyu~^)f2Sazcu&dswEeCOkM=U|)fJ&3TU_RZk$MmBYhz+W2{9iJ~Lyis? zb9ahDPOiccFWwjz1R=A(cZlpwhhne~E8B{F%LJ;51!H!vjZVZ7n<++*7jk2Lr83i} zdne)hnk`bW@!az7LAG(`h4~MW7%kG@fiQ0bC(G-JcGOO*-%p4>4|;>{16Gp$w}CCOf3hfupx^W~+fdcA5yW zOc+CwK0{Lr!G3RZO;*_eX1BxKP}4vLQ^Etu%OtUTdZQ?#VOf|L#ZB(TRT4kbu<47#XPx%xBkM307@J0*28~S(9r9VA;*!eZAOC zq4U&dM6!H&Dms!X703$%SCi!4BCmNp1GoHF?QFJ~Grq#iNiZrElTm$;oLo|LVd0c( zN`(It z>NcFgmpPv|#zXS*H!B7o`9aTW4;hp!Xkg=X-G~2FrcC?C8j^0`jjb*v*6Ap?a1=ip=$Fd_rfNR&qP}9qah4$&*%HkYr8Ip| zOi8I?m(N5Ll`x_OQedUYM7hkVARp%jZ4Hw7&*qsht=o%9gUO-(kGTlPCD*Zg@{uII`Q>reZ= zp}k!UjfWr*S-s;3z>SZ;Rv#-zNH#PH({m&=4&0wEaq>I+{Ra?AOGlc6AUs#(f)=%m z(L~2$+B8u9=yN1!gsv=X>yNbyf+fxd0c8jLMA`MmQ&$#y>j|y!I-cI}^p^%KA!Zs} zxUv1X$}q)1*`91qjYP{D9Z>c#QsO&xrzAsB15ad0ADaM4O}AJ3!N(msS@7&jQjzyc z(MEkK&03dP`oOMQ{X}h%`qt19=Gro=MSG9wKKR|(G$Z4Mis(|<8Vj(4V#m?-1vEM# z`;)#tAeuV`4oTzY!4~6MDBN>+t6i%lS7C28Pjg?4dZ$o0n6qEcrtjr|N5%gn~Jb~mLzSB>2u5t??flGWAK zUMMyeH~$;&NYE&lH8hXNEF)foh8-1o^EeBu4eI;s zsnj|;$y#lW8pT(Z5`pbY>99mK1DofHfCLDt!v?aLVN50+h~)7tl?y0D;vR4+Z_h>& zV%n}L@7~O{G_~y7B_vN+KypaA08jzM!Yd&XkGpIpPBG&6xsVYrp&~&keM4<}J(a}L zZ^_%ENMQSmz`bLyrat4pXnp*#yV$9L1Teq%uz0*oC3O zpHQiXX#q76yu!;|QA1Kgf;5C@5y+7uJCLMxydV-%d08uj?hlEW*fV?Y3%mZ416KCl z{NT@TvBf%5b@b0afx!KYAY`fu+uS7pH#5BE;&1bQ~w|n#kOXslM^DMQQYy zo4QWvbw;*2+OpYW zG{c#IR#X~w{OVV9h)bgd+(D1X{TMo9SE9VqkX;=_BIT5f(>^V}1we%BYFky%^OK0|N5z#W@i7tj!LWrfy8FzLYt=m!Jw0n%**s6Tf4~N2}qO zz9Z_Lp3SlXe+;J|IqidEqUGQ|2J+V)t+AGm67~}Uj1Mrhjxl@TBBbM6bp`Ga%y+h= z?j3?Jr$SD5&#VZId{%JaOaw~CH0h!2bCTj858ueVH!6H&D5{~%RQ2BN_3_)Z?4#$t z#sdz0VHX6yF`0rry~H|P^V+HDBm2#Dzpk~Y*Y!(FoVVies+Wqdt_d}civXs^vJo4^wApKx-aA9IJ=n%l@06R3lQe>As>(wa#ZZs6rIgmOw z)_({k;5II8f_((3<72*y`%HS){08({FqT=xW;Qs~YgCyJd)}99W@Svn zdc$g5ft0hQW{x?0bzd)vL2iS%;oMx+%sGEXCXr5O%`0pIx;Xqe6KH)*z zr>}X{)dUBVXiq&&@OQX_^deSOsF}c-m@$JgCp^`%*D7>qE1`Md0(!hzpDhRgYNvVX z;2U@N1?{kYl>h-3oMby<Q*4?YM!{31FX0@2qww8mO zyUuk~&JU#79<|e^5nI>>oL9lCzpA&kaEEG33j#7e`AHDU=-@iRTIrfJ>n$+(bcON+ zvx%6?Sz@?xW^>?7|z!3m?fE)mmn8r$yZCi)%<~+A&Z&-UFB4HcL<#;?FhbA|(=JlJldzV&vKy4)=;vHCS{Y zUB?eo6TG93DBykV9+<=8JV+6!%g&0$N`aWSdV;4kZsI%Q08sl0_=Io9Ls9MSX%Fx! ztOg|pfQerJKX`tLLByokh|$Dzwq2k{OnG!d*<-pKDyQjmosbt4nwS4yXR97Cv*mn7 z%@3cGHk>(9>dI?i31VpSa=``ynPe+6!N=LyI2xIJP)+}xN9&Xb7#G2bawE_q-P$Gl zmWg@F#?{VyAKMfj$a&K{tF1 zXiX?9G-WFrsUn|AxO24H28?J}PD)w8HT@hG~7e`~rauQQ^;+vNZ3 zp6-}XkSD>D{U&KjbEx{k)q+5nW3d_*^qSR{rjL*n#qV#hAiV~+P84{FS^D(#Cz!+1 zMo#g=Ko3vmRSRYZ9Nb-?c5_w1oSg;0A1_ZBS*)wgx*!O2&We|Q6#=a}VINFsMtoy> zewD1w2Hr_rO*_gB^+YGj(SIg>v!OZX(wTwshB6uQFBlRkhLr zJ!R=7rA@=AL<(bxEA-DGyxD0n(wzj0okVPl>GLaG--?TXbraii?{4=>1GxspN3RWA?;m)=n1LDv3Z>{auXn4 za68q2qmn1}6% zU6xy{yjl$|n>mKUYNE@geh@pwZJsVc2)kb<` z1cKY`0LXjVT<3X7WBuj~%^bCpS8rH)0|I6V3X-ml(U`M~8#iGQP5)LW-PRX6y}VH; z=AY0sIA6zViy+I&4kCwJp|yHBUGL&FebHT^N8S5JlAvK_+1>hrK8fc0H}9O{g*Sa~ zILoVwLw6EzSEJ|3l$I4~fQa%JmeRSc!N-N=Y3nM`MQaF-Jo~Vc_avTE${dYxQr(Nh zM{6{O?}?|dV_%niFe^1dYhz#70(4U8Uv;LcyC-wAl6ltYBjH#IuJ7pOH=xFYkn#bh z_e!a}V1B@aX=I<$l?~nK>beF0l7Uy|ls#83f=K+AOnNyX*e+tux)_$tPI#7kir5N) zp{EVoI4*YYuZaqUg?!@lc9j1ThllYhrFbQ?Fv5nz)O0fpZQYU;M}wrUZX_B&ZYj$n zSiqI)p`(oAot_&G&?W+%E&!iWA-LnrL9**jH_41OFfme9cslbA^a-X@R@9tUL5bP% zUukkA{Ap2vV7yG=-B6;OAWA12abV`w;jXv|PE3F1_1Axc2}$7{s5+7B&CNhgA&#$} zX^XKWyWFZA(kddt=~ZgL6lz4aq(`x==|=DBuFvhAGr!n-|6&52%=v>rs#!{l4T zI*t+5xtu4CqWg!;n<%5P?+^n*u0f~$6&a^$lJLWCX}SC}#3gwtsO~SE%MV(Q$tl|X zNY0LfXB*9S{N!OSmYDTcqJkQuhG7f9#1|K_q1&nE2}2NMN682aaJKqg4K&;yr;uap zE1?Fi7lCE4g;i@;BW|QxOby?Ihrr#x-grfXfl~8}oDaY)Osd0b@Z(@p zqjNlu3sG;ugFd(=pL+2GoE~*a+ymmkBUF6!^b(Xe1})IVe-nCSQu@340>B3fM6z~Nyn zmsSs-UJWT6OlhkRx%aiWZVq9n{G$jhg7ell?%u#t?O`EU>_>A^?3_vpaZo|q?sK(( z@6&}WhBq=~SzqGyrJih#$*zDq@X`6h$Ob$4?=X2~Zjgy29Y zx6VH$QsQsW*S-xL!Qr zh+VnzLIL(-JEk3wq44iFxpyp7Z&6<+jnXE7Al1C=T1_j9kM>wsqq(GB8*czd7{N*) z|0Z@KfI~*S@0+l#kMbj$SJU}lvBAtz?0~VcBZVDv1@KC(&#ylTCWNF!Azuc93aTw z6)Oa-aFKD-)abzH7qdz>*``7)V=upu(gI6S3)E{j_eVctOM2!SpZY_kY3<`h)A=em z4v(^AgO*dXVgml)s(dkOLb!SGXQ%6a21`80mi=h4w;6Bws=z~wAczNOKI)x_{`PD! z=vV%>p?wxVfOki#(%^AJYV^>T|NdS0Lm&!nhTP4tM0kVSS)6UI5je9pK=M@+C5??TA$hI5$H9B=M>$ zK1($GLj&~DM?55R#>6182`7)fs(k!`;mp2F(n7K zz9Ez-6?9w8}ZVoNP ztRW>vN3gy!Sy&k^Lo~>Rx2(>=;)syZ^N9a)T%`XSUvqx52Pp;5&j$q0)~#fw0ix%r zFRJRdbNqkfOc;7o;Ad`9!ESyVmA3A@Wh;#h08vvCH40~aluqJ@#LIHp1iI(9xLsG=MT5E*Jgol65#3}Si>Cu8=tcs9;yuQbwIYC5x%AyS%I-* zc=T9xV!ZpL9GZ&vHb-@{@x>O^oU=f#@_Y*Az8v{=`Kjl%I5?TqlBV3RrkCy9d4CE$ z5NRymXWt9i3a4B*2zt@)%@9!50$HT6K9Kz3rHtp7DUNky99C%L$r$?Y)qNqdgs6^_ zL)&uH702v}IUmg9%JVb3%{|8UQsqWP3o*xDv#fz}egg;aY9CFO5h@i+JmB}VAFdth zs)}|q_S905AC69$nMDIMzAW}eCP5Gz#b1QEl z=(w}tu=UM2swmBoXgH;y$J-~BeLimoJ19lD_ZrACCZ)EboUq)hhsOGPb42V(xA8PGD_@z$dXU%GOB8Ah|V2 zVbeeIiHC#nB%yEbt)ar+iuM6Ik zV+mI40{%&EURhMLAJQdsGyT0{W>1yfT`a&X?^{T{Abz=;wxbBu1xrS;CFswq++WJ& zch(0|fHU9_4i9n2jb4jlUF=K-CP#G4HhH~ j}F3@QfC9lAV;V7=;B`*B_Sk5y+P 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 QM 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

F519qZrKLBrO z!)rp_>XrJE496h)xhP=nm;@m?%vMSF>VZucn~sM`a9dZ zf_p#_(nN(N9L0+4)VSDUz`?d$`Tl3Y>#IBnz3&fSqV5SoiuDK)!?QYW?wsj#B6>aK z{PPh*1EC|gaLda*(7OJ|>8YBrSoR2PU_9$QFqO`Z1@H0*h;i_MILP4PYR*jWLL^f@ zr+4IMj6(_4X^QliqzA8`1{zUb_q?xECKk==u()g_+&Zk{MqIfBXf6XrBdo+TEdPD6 zf1<;!vceZrT_PnTiDq(Q89btrIqn`ABh^(Llx=Ta;kI^3WKU*b2XeYW>Q686wrh1M zc@Q~h8I$wtOHVEi7)>EMd=6=Pc&jl5&994Dg9~lTD)lFGdO5PPi9r-m?{tg3s6^8X z!yqpDr1BC7X*YkPMd3!`_;&a^Bdz>(OMRns59EoccswG?#^>PJ+M@|KPg7G$CdD5c zd-oQfnbCX-8yw&+$w_GBvv1q+l%?0gxnFvklxM_TOF&Y@AhTM%1#t}ot?flF$4%;1 zjXbNd>_&;R_!XVbQ1o&J^F;Q4r!M933q!$G#5YMQ`ie{mAbvUfzfdAQ z&Gn-&!S$*2!lF`n_E~PGc7RdHnFAh{ZoOML1|OqRgU11BgjsaaISo(8;zw4}L3VfW zgbIWb;I=7f!rFv8iSGMA>|dUYLU}XgxAJ@=m+Y&Q5<_ztq^4zn?{R7feqyU_`-^zV z7;@19b`X8{KP~OgS(ro{yBQ~*osfq{6BO{di$A&Qh%?8-m=E8l z6Fy6WHov(bxt1t;&V!+)6F8o(R1)_^0L2GIM_i_l99(|N1t>ttx6h9H7JvPba?Ou( z7EanrchfZ@+Y<4SX`8evKrg;~$K-)E|C{^=(V8A&+{4ulO@@TWfQY`7y)$>ShX<5M$dH5sN?h z!Jk-4S}+G|S_b4I34cp`$TAbQDF~^C4qG2BJ@RyH#!?)`jNf)>xz?}}QtAr|119c> zsa=!|4CmDLx5A+gxbqmTlaH(RFgybx4pdCvgY!?O$Ag3)nz7>IOLUH)B>k5AT97o~ zxh8eSm2G0C7Qb?2`h8mcT7ya^N(o}4R&tzyzy)6|owXObg}p@?uMVwM6o!yMEBz3z zMTYyx)Di!J6U;^bus; zp}Q3ooO}6D4`alrvz0E}ZCg_0XSV`VKmBViEz=n+0D3^R$8g!f>Vv&Ij#EHwb_n{! z2ccT;fwz$iLw7&rPJBd@VI|1wci~<1akAr7@tr0D#7h6>f7}_AG3fCiKlmVGz+<$X z1k`$EdS>p5J`O7`hE-K z6L|h07bol=f?lLli|> zufa?{H!eiy2L78+nfV^XJOa;fMhzWyGS^kBRz4ROZt1qqQS*pl4`7-EFshmEf4D#l z=~{$7`OoR3>(_b|q09LFI2#SyzGmI$?;J*=oWB&zaDzgBK;--6Wx{ttkGl!TsEE?1 zWQZq5Y$Ue`o#UF%sju(FD$q}sX^2}mgks--jD)KuZ2zlv3{UH#L2x@-c)fBMq?n#f zUp}fzI)tBfxnlA^i)!^GTXgxPPKG{X1}AscltVCqlD1O3M{LUcccdV%+Oz+5rly2l zN+?8r`t|%;Z<*|iiW;)TCsV^VElcnvp~|!<%_91VPaMt)qRVlrQuXTiy^lkTw|$+( zRT59p5cfRM3LgT7WPVI(R(D&pQH$A#oZ3FTtg?^fo{)1UCm zT;VN2(!u1#-{&EEielko6$!Glmx9=+N3g+tay;kCJDSLQ)vEVz zHOjg`6_1oaba63Nz)S?L6A1AVEV>ZovWlNGkc)RW5*jlX<*|VSTYZoVSYX`#g5@xP zF6Wwu4mMbXkAgye1CXb+Fi8DEeO@XTeFv!k9WY;LnqzGphk2HXgX_1X4A0jnrX5a; zEy2pv!$&~gj{P$lmRPAD%aFK+HndBVPAd<`JsF|q9flO;4*NAM}u5LoHftT>{i~JXyPgkc5|&#Ww&h z{bj+O*9dy79=`uSyi;d=A_iAqht};#l)RH?Uv40ta-^_Gqml&Tu6d?IbqJ-t$fSo| zLcCpGt)xLQ3G>I$=iJeKYG{(`I1Kpbm@SAZg60`2(3+KRC#l^#R7;GdpJtsSimFKE zeY5|cex~{x1d72TrV`DXz+n3TT;EBgb2;3aPx}v*=rXV5V46hr{S<2>ptrVzXE8Up zxuQV>vtuC+6`#SztATo2P=O#<{&Dg)-PSGj&2;qoK=Zzq;@5yEa$yJ;>&7BHB;j7g zWvz!{FZ~00{^Pv85eM%g0=u%H5v^9cW4~so7HOF7)BN?y6imlwf|-j7AqA3%&D>sq zHiK|dOz*~p)jgMxoHSHDSFkl_8h71jg_Z~^{K7D6D%9mj97y1M0ciC_&v#nZxeTgQ zOTT9O#dM=61nWPM2?I+|oIPmuDxq#iXAJV{^H#VMa(-KU6)+*!2RV?=;!8gEh+=L(i>3~tUyoa%@gIYew6P{qpIAPxy?&(4FNddbaQP${WO^ms5+LiO1)XY9XW@&MsL z@Yty*%P>YC_l`<^PE$;BGxm1;*sbWNuEHZFHL;tg5WNL)FR&W0T|bF4e*OHOQ1AO6 z34A7+H8Kc6XvTnkzf}pm|{yx@%{Yjr0W3bx@FTbZiOObv@U#vA*up?IHGHcSS-< zK8{d00dEt-xK^0P^24FI26o1O?s1dUF4QcY9c?U*=Zfd%GoTaHEP5ZhbW@6GDSHD_ zl$NtO>=%d(^4hjq0zRu@!Uc446t@t|SVTY(MTQp=OlAAY%LE-KS0efBxZSq-K&mt$ zV_H#cPgHsiP|hXO0|al4@btnL;v2X;+90X(7y(1Vh`5PXl2mTLFgxZ;mjIpGgF%$} znR`RLut~U-Whf;HIIhDDf0jL}1)Uu&4>1bfNWAy7^(<})8oK*$>rFcn0@wogH{?Mg zyni8e$?Y@q9BZm!>evgD>YuD~+(`D=C@JI8a7_2|8d{qjF>TQ~kg{U31=PkcKYYFp zlrOF~MB~fUtNnq)FLN9cB57s-Rl|_^j#5PDrS_=9>q`S&Z8vy-3Vx%%QamDOpI11y z4I5%O2l45W-#ZRQb3NU_#2oyLO&a&%bl&|u{w*5_P%Hb=s-I9W8axw~v*!z@qAlXi zghsOl>Zyy(zU63GKAS#hkj&Wh@Kf5%qGBm1y*8_!HL72N1gIr5r?`9BLSeE?K()ev z?5iUot_;80!5M!Gv24?pIU{H|OI{_sVt`d-n^FxNfbTPe(8i-Yw*Q7JcNHds6?n88 z%g?k({n=ONt`LkUZ9xT`Ealg3LBqwKJKCu#rI?LH&-2YEl8VVIJzdrLzBz*GW@Dz^ z&>)KO)b!@H?k3ku%CEtuLP~xHuv_|g+(0Vz^>Oamw14YY?h@xmppAk5i)VywUFjNs)R z^@MhA%py2K+ z{L|$FCe^)_D-cBu;m0O{@CJ-B!0rT7aq}jJ!EmVdv}5yO zsArO?olB6iPC`~4;t5i&TkH?$PbM^&yk|9A@-?{de3xZ)7F$n(Sp!VdaCxwPXEHn( z1{*scio}SsSdi!?yRJ8@S8ossXedwe!v6M;SXsaQ&C}n-0#98$I2KQ%`B-hY!GA29iYu*o!?R(-I6R8criF{~X$f&_^pi=$7@lv-Rxd@Xa;v)W4|(BiXOQV5 z1GX7&d@SjuwSs1?cMb zB+0MCFw_8ofPbVJnkS56Qd64Ge3a-(#uK~=N@mKd%ZnW*0h~nci>LvarQAt!?B1dN zWrg<_4fX2QE~7?4q!@e)*NS-%{V_ zF|p_Z%tx9dd~>AMRt!)+vYN7Ox* zEk>ko^rqWiM;y|r@#55pqgRJmEuc?Bodm3L z%z5E<&vqY}Y55oA7*m(ugAD99)eimBo)8Ti5h@L^+jdbWRcfG-ipJH2#XnTf`VVlt zKmW51QG^&MRVpd2Hn6fu5IEyb-iLu6CGaU8#-t0IiCh)D&$UaWo~S#A6d(1uW5C`4 znWu;xRap?AFfCg49xO^TtIxip6wnNlndWu>f~jV;wB?ajfn?UU?NN6kneQ%pysZB0 zo*Mk-ZeZUcYB&jK?PWq99~LG&b0QzjViO&-5~U<*MGKv=zCiW0t0&Q)A|)xWw5REA zjl9$v|0m^oT_!2SU0lY@a-632zJbqS>w?mpG%eMssA_}pE34WG@3gSTzZQ;)DmTx` z%`4e+0jJxU+=1gPvy<4uxn+M9U;+v$wuwb%gZDIk*!`` z-YNp75~?}QWG#6Cgf~t&i5FRz#^&zVa)SKzn_@CA64zjcnsNcT+YHs}`RMni7ln=4 zbBX-uu>^aFyBoK2QDE?TCiG_N>Gk!9_tX`p!E_ztt`}?0h%lM`>UkqsnP*5W636Gz zCffHnqb6konnK%~R@_W?CN^>IZTom4 zTp<}#l{QCp76jKVnT=$Iq_JR-HK9v~6_%kQwmi^_H4v7)vfN=);Ghr=uzHZ~HT zCc}SOu$qg9~ z*uTB(x=Sia@O#!8guqtv;rEqJJZ8{-xA%{3Li)ncrb@>5o8#pyMhPpvLHYV0oEC!P z?KytdWZzG$XM{mf%_)x<_A15&0c6ClUc#z*skj-kaH+}}!F~^=Gk}$bXcsncXEidG zBbwjnImIntMHZsULDBRq6)ht7)v&?o4b+$K-^eQ#dTLb2#rP+RZtfer5TR*`*7xu9 zzFlul(f2)s6@J6V_LX+d_bKJ@y(1V`Al656xDP_(MMFP;MOZ~PC(6-G^J`H&=zoAq zPP%|eE+By23ml^LDnuicDbJ!JtMSewxCRXK+bymexMx3l`(?!zwD=Ep-hL61Fdr~0 z7~fb6v&}TD*3~oIp=9^uOD05N_g=SQ zVjd57TM&PoQ)G&g#vtb=r0%@b#HRI(?MPaGaVT+gf564R;Az|$6~JFl=gnn9 zEAV9!A)N6ow|IR5WRI%>AOpuwxe$r?t?u=D2o=CTnn|Ec(=p0Ua$j8NjBSnwb6^Q98~samObQicCM zorF!K&|Ni^l}hDNuHpK~&5|?PiM(fU%x%DGSNI-psO05@A1WSG3_Ym9Ye&{ zV%vBu;J%K#aRy#G1zFg0M6+cUsmTA4evQcD9gFQLWYq@HOB=m9pd|_@5?dr;@9J@P znIdH)7V;p3J{{_Qv6ocJOoCt;rSNMn(KgUFZ0G-pCqsT*za<;eig-?u#_<6Vh$W^- z#&Oh5)^4cKMO4u)PKUCZ`s}L^T^cCXM^#v(tr!w;Hp9@6}ts!!k=_87OPo-Uf?^i39a$-N6(M!^zYK{)Ec-U0Mmj zqLr2%lL3c(4&8>VHPV6yL@LXaw+E0jBm=vrYl#8wSj9t(nl+wc+$SNUx{XKsW1M(t zYD{wj0w_P=3g`ywj`DXmEdRAWzkG)eQ$F~cT!c`IFGHVQ8UUxEFIhmABk=CPMgdYY zDq={xU!gFC7|-R#Sj5vv@8uZP>~!F4&#jDfRI0HA z{Df02-NqMgIDrlUn?K5ZKEp0v1dDJ}GTDaq5%R7ZH~jn%`~c6fx8de)Zj-0T?fnTi z#a$>EfLii!QQ=E~Sr0VZ@d=AfR3PM_K2>}t<%h3=M6+&_aQWENL+Vmb0HM@TEaGYU;NTd$tm3k} zkI|F1(Fj#-;TzL5a`v`GK!(mj1$C8LBCt|bLshhNw~!5fU-TuONmp*i5&uSuwd5M0 z^P1qIHAYbTSI&A&%Hho_Do?5e68mTY@xMi}GcH=7Qhezp)!uDf9t>l2oUDQ+5`Pj+ z4#oxb#G3Y&X+Qz*oRWtaO)Bh3NOKIXh?i#g><&t_ii>&hEEgOhai@;o+vhCXVf0rqQSDQh?&`Q;3%r45Q zsKV9uJY5CtCfzJAj+M%W0EfRfA@wlWHP^S4d-*m*1cnSLQ!M9S_25gpQ4`!-1vp$i zF8XNcq{*rb`^2uj-8G7K2&SGl-G7e=%M@$(X#4TZSP`CjBM5|dSd|{*M>I6A$g(!C z+Z}giR`o6C4@&v-)*DO%>KUhouvu$DK3p^TR)quf@m8S_<)Qr@05w3$zl5Tw>_pZ8 z)NM)3ndJ%SCH2ZZ$s|G7e%~Htqx8xFvoh8#8NDZDxr@BBDv%UT{6dYce=ZYlcv|+U zm_8Pl11|6%eM=<|Zj0$XeY0YO_oAWf)`gn506hkgVmKU$XgQ&=+<<<>tLNYs2sffB zT9BOsWil=x?$Wn+%Zjh@c(`PPOR!(QYmwEi4GkqGmn> z9aBoypsE>d+^BwZ(Xb;&BQVA2gY&dOU6|=avU!b%!bWRV>qaab)ly{5%F>*}i~-k~ZGh9h3|zuv8~e-)P!lTv{QhMa^sAWLlhIVmA0`8z0Gf+qnLa@3fyeCl&(*kT4q0y2M1gr>Qbx9HTLSTRFtJkGFtd+5 z7NkVK$T>r#CprQXdwwSSqpI>V2?O*Ihx9NzPR6lKalF9;-AUXoXkSmfX-RuLWYH_5 zC$Qm=v=~P0%viJFWmrn`Yi8FzdqRmwTSYm+d92(1uwYQ^{|jr#B33#{m<`HiWrHL;!g{=EnOMdxtaYSBGa1Ubs1d( z4v`N;f{a1sM7)p*^Tqkl-*qA4e>R-{rGwYZ8LZ@I#R28?L0tyh`hpZ7p!E8+nTVXU zesz!`?GX*WFF%BZ^i6fBwmhRJNCerO@2-E#*Q`ej0O;E;Y+zwiY~ex;iKLC-C7YGv zXhQObe=AxVKOVQgt|)nd72@?`KFx0m@czidwap|TJw7AoN%E8Uobt+GcpaV@*7oWD zW~6Qz7V!PZ>ERgu+JTD^^?D<%tO}O5J{xeAzoqR zU=2ozsvAm^I$1bkYS!Y^C5Hl`AlQ=XVqz7`B57O6Y#?fjs|AQ7Avz+Tw0UM^g)*3@ zJjI6`%tH(`p>-q*l3brVFf%mJ1^I6dKP7ea#_da*obSP|YGdxULgKP7iPi_eq|zmSd)eYT1MInGLSib%VIGB!oT!<~v*zF5VvAuYMQGP#rsEE*lhv|J zcq2W~{OZDd@LIa|bO(MN!q4Td2iP~J^_y(_@Y#HR46_kSEW}8r%hw1rBU0kLe>CMJ zzp-uAmO{Zm8HZ}lV10X52|9nBD|}7&#dehT>OEkqB)jW!hF*G+`6b+jh+cj!V-Q}! zCXdG)_Xp#0J_PEv<}-mH#nq<-GF>_LL{)KvxPsD*$rFnmko;w8H0q=L0-bE?PI1*C z57%&-E#L#v&aH+Y1R-;B|N$^56C0bOZm^y*dD zeM~;Gd-T%Z(IMhRZDH*{T-XtX2og`96$!c z*1WHe!+noi;=Nf>5ip2uzl$Ly%293Y77l>d>z>T;7?D0jd;fD2p;@{(lNVLHnga}Ilx_nW&N#3X+6wTqB+XW&@| zyy>wzlS!$H=|1)MFNNa?>%=9Q!(1%WJ-^KlmX4du4^rte9-%;B<()BHr2X6W@8~hc zRDg!_PMzdQm>tK33Jr$_)8*rb8b-J;1xY;n_hW{H4hk=CCS;qPmx#H^Fc*XSpS(c7fNL+H{cN!-5@p%s=>cZ;sMpy_i0`-~&Z~lpK=-^@P zD32t4Co=f58Nh~!VpY|04c0$Br$6~m`ec%`x<`1E zdM`okZ{=eVr)xl!I$R?xYky-g@rf%}x&j~H2w)V+&_hBK@ta$<6EE3*F%NT6g((!6 z4K*|bk~2kd!goRXR#0P59nK+_)Qa%k#K38{(TozOAN}gIe4YDn8+OJ~>{X<>Vv)vL zgKqS!x05Qx>-pptPjY-+0sX$Da~TV36fk5yOwTC4x_mWgkyOi&JRezY`nmcBjA5%L|q04+UAusRNJ zxKcS_1HOVm4YsY3J_uC(%wM?q#+!LhwL2ahXn9yocKIAfYFficvV-h4tN+ z?V0DKcBDI`!Up4#9$+mcvjyL~4%kTdmfMpCtnvi~_0&k6gfu^@i1&ni=|I3EsVF2% zzY%p7`L&Q~?6$y62V3e~S+?)lnO!nOjU?;j@4pu@#c=pPoA}!rUw$Bo1!C%(d5!ZV($OhyJZj?CpXRN%#BJ^JkNJ+2wbe(sZP9o4@gy9PSmh~Nx%A}{Q@zaF!B7k+i_H`DFG8~FV z{9R%QWj$sY2Sj*#j6t8Q7;vr+bi;Vwm+G}1V$F3~kDnQ2VeyA)?h zs;<4L`^jgcu-(BdlUkp>>H3t6b1<{_ujS|@=tXap`4Px&zgC*}$)-|HLdFD1#+hIh zJoOCN!%X=>0Tw8E>1_Yac~!Dc;}dk)DPjc{1A?sQt)3sx-9N2Su#%s)XW!hOrJwsq zRim@*#fw||4e?rquSY%jPcP^61m%J2#$i$gX~y@qiAqy!_w@p~=efm2fKRrrvhFE- zhAWJ2HGM4k3GSd-M?}2+=0c0$4sHyCC0MyfdziW6V&vXN%Z1ioZ8gid-PB&Ymjp&^ zqGi8P16f({ZDwb zWORSuBlM}s`gTZ0co*ZcM{<7yz}asOU52DdgcU=^jO7H&4~UjaQrL+1z%n0Pri<^x zaEx#&e`5|&%SQ*>4B)=P0|Qvh4E=WtbXIhKS0OjXiPp?pQ4DX|f=cmVIN)10Pm!=2 zTOJb%SfgjSvXe=Cdm^y9U7S2 zzRdsvI+k8P9KAY>O#kn2zO+!((dtZ(UH9!q1;LY(&caZnjiqM9^y8ms@%g=#{2!J^M zoZRijNQMOhe61#7(Z)?O)s&!;l`%zISri=~154L2%VgQ0VCv*}UQthh4Q2J3F%d=L zwJHtoC4De1N(Zw}NTL7+L%FcRC$Wn@R3~LQJj)tseb56$+M+li@&&5xoJF5Ef8&9sES}%yeZc+bcAqCjzC(fnJNhb)&aG20R8Xe;b5WOXR z9)EnR5J>h9ll$RvZu}nY|5Nw2la?C^<4kkDz0f5UJ^|!@KC0JDxINjI-^}f(dY^8f zNGXyXv5kPl5L^HJy(?EdM<(sxdu*{?8#~vfivIbubmk*XSm1I=xBALSMN@P@-vLo{GAJX}aBDka#P9uzA-v3d zLHd6hF7&KY{J~K~8do8kv*sZ)dig&WuNt_HPYZc?TkyeWd)2Avv!Dnel)f!n@_QE!7bM@^9WwLL3UHB!s9Sd&~y-La=-~4uTJp6G4E2hWO zyJPLDtznW%o?KR%WGlv9s{M)m9LA_OTx>j}2qA;zPeC_woV(v*ey{m*vJWa9#i0}W z_OJiN;t;;Dolc`(t8kU7WCEIj07n{-MM%hNb-w=PC_^hu2EBBe;nhKOVWdKeRWj}$ zTKb?5TgLTj2bp*Na4krlq4duA+vKD5+OwwZ4LvG&%@X~uH8Kb$SW=?Ba?jN#z+c~T zDd}NpFwc5d=OzAXt)`zAYaD7XTx?*_Mu}k7htRV90b+`@G|i&1^o&q*wNK{VBwjOw zoE}x*(tZ`gEfP5wQui$Ikg#vFw55bm63NlQF7Ki? zp|S)D;cW{)ARRCe^p2P;%^VJCX{&9<0J3>&Uh$&7h#N4x(_ed00>uWzDL!?$vvjFe z3u(l1!dMqLG?+Np>`?<=_JnqCLuxA#nERex1{}OD;8Q1Z3^;cQS68pn^Y^VK%AY+T zj51ww0e7_JUm|K3UeOjQF}N(BBt!KMeJ4Ds@q@{s>#dNyx@U`3-rnH1YFlytDXKm|gcV`F@V`#=S;*CWO#Z+mH#qkDi@7&#g1mri$?$8oGfho#Iq8&OJrbHkL{#u$nRZMD zSiPlYk=1?1uobTIbSz+Q-pB|kf*uvkG;5RN`^!KMleCIArh#<}+HT%>MH$SZq*|FC zP3zL0S>sxQ7aB=8?{RZ~3&0dQC_RVnM6$`q##WfO!P}#db;TpwajV&L1@DxifyAgm z$f|GXo-3#YJIN>xD`eu@NoBD2XSrf&h6ag)02gNjy}$26Pnz_WxX3nb2E0H zRi4z&6f|5YR9VUZEYXH+;h12M@*}ctPkh=9+gk`j#zgwe9S2gRjruBe9u73W3!66c z9?cOjMym)(*r{x;`lE)i`j*(mPeiivn=K1(#qP#?Jv=p|oqF2J40H0o16vqDX-qF%tb5434zt9dWXy{Jq@*^+N*%Jmon1_fG;&IP;d-Z|%p zHmSNHS_i)8S;7Lf_?U;s4kZf{J2;^5tKGp;e~}2H?G+689?C=9!2n&!Z{0_A48dB9 zReUj8q2v#gjj2G5$c;;=yjif2;6%t#*b_{g$PKGLcxO8qT_g-B3#iPt=ejuQ37UxnW-XsNKZA^`3^e5!2vC?la##6Nlwj;ArI3Frs0ZS3*a>ZIZf()!CMqm}8OlT; zc6r6?AFB&7joi**VvToy!bQ$R5O@W0u0L8c{5nIDay+cc(DR;EyGKTzD!p_VC?l=w zXyDD6oyzgbD0q;{IL=V4$!Sgj51MxG0-N@+)jgL74gG(>14Uk&rGwyY639*wzS!QbjRbY!3^q$CJp``=k)VbsLX`Zl(q}n3)ZHJ&BIdP7I^sl=x zfPEkwdH@jN^pt+yhJ_^ij1?X)WT5C!M!h>&Hh0*f)u%!0eATnbCHZYqN~-MA;4yNg z(F{*V?#hmG4UQ-CgLH?OJM)yJACf%C+ktr3N1GPVZB)rx{MsweO=87l0u$S&cGgXL zXTkcx5pj00W3V0-?BADg3ykwf|F|DtF^qVqATM3TquZi04=LWL=ld5*9E;pFp>=4# zTc**$#|L>ElI~hZ8tF_&*q0ziZA11Tg%CP(&$n_{$pL&CfIhPyLU#8rZj3KbE+r?- z7+CVyAh{w{Pxa~}Plf^GxK|MBROAmKl67*Ns8?@oeV+q5EKdF*m^qHykhm*DNN>%< zTN$WcWF(hwb$|0A6l`KYw^YD;f(^yU=d0FuBgK05gmi&b5hEd09~uF(MfZF;y~;62 zGcuvv&+wNG4Ij96&esrTO~jCG+k0ttvv*uIAS$Cn`^s7-B-rBU14_HI+a3Go{oV8< zwpswyVY6k40AG{=t7b30%yg4LZiS&>^x48S{RFJx=89}K!_4;^EavL9E68nt?jEo@ zyZnD70|eMv$+p!AIewZy{Gf9}#M^OREzEq^%8KkB@`fis)cq85@b2HnGH4a3t#(~m zA1VOu_?uQeR~-2Ct?gltpH5t63UXxj*u2qfZ4NDuVYC>+D;7HfQyCQ4hJP4WYV-cd zsam0N08qz11R;E1clU*_yDesm|Cqq3fdK32{(ne=EyxFo)JRtF_6;^vE(e- z`9DYEU{VC5Suz(iE>BYn{vaTiQpztTei-&Gh}& zQxvtO{k2FKuDkmnLjYGfnXHH0=(_CvVvooaSWv;)w{2?C^f|8=5Pd8klFmM>a=I97 zkAO4DGzaC{Lbtz>oK>OT!%y(XjO_rurQ|9<4rnTFIK`#NS=N959gsppywter%9 z&|d(NOYk7&AgX6xOn!G!RfLm>&={Z_jAZ>_!1dO$N|0AH|G>j(_K^ii5J$Fdca)2~ zR4h!kp`q*!lJ&K3^;zDI?tdZCRaKQh+&I`Ls_E zLR%FLJ}_8aD z?Q?~{^<#y+jB*xc#f4kU=+i?xGNQkyugtcPJPY_LH~YPLL4fi4pw0AO&>3{#4|nvv zi-}D(B(=DE;L6Q8;@>|(To==P2+SYh-W9229`ytXh;`>>JB(Wl>KUObOEcz8IA#8t z(seEld?%d}3#Y?LB;#o4A#7Z_a+GWtdw&nivZITAUSM-cvF3v6Tz_FEU~7DEiCj(` z(WS))i&yWLc$IJCyn4hT0|yd-8oa5BVB0l7tmg1#{prAQf>P?!B1~`e13<1+l_CsJ z=4+yT)bJw@1S>=DH*jNiu^@#=StL(bWiu6pOREf0(e(Pb>e88?X#h94D~1NNcsH4g z%(rG#7$xos#1|gvcC%!_2BbJ^M9Fkk0piAS4P}5^BrcBvmGkc7KZb|k_k_@Si_&kd zPTQrx?O9u=tb+x$R4e6V+H)~^L%4@az6IeQNM;@T`<4KaKzjRsO^}wH^X65~-bF8j zqg*=OZ1bsrNsOzU8Cl{YYyv;qs|2?`nj2t%Coo8`k$O^cFKCQj1y!=mL($k;?u&UM z`lft%wH@J=)Kf`_-S0#$WFz^$;=S&b3Q)+hVe9WRCmphM3$6p5mNhW>V-lrRZrlV{ zk6bSCzlpYeQ(;;u3q`Oyar7;Cr_9FKj2wy2zMg#?a6jvMaD4s1zqjmLFUElr$eRiF>Ww#uF~?LcQhW$%VW5zMa&QqZfx*WIN}o)1uEqTGu_xdIjffK zI*5v*eBSXJke5tp%gg9+2`m98R7RWPoa%_Rqj0w5Vnfj<9J&Riof8~3NWE^}o?hna zmW%RHTn5XhsQqFsfF<@ znMpRr(vWQvA)BIGXXp7J@o$P10BJLE>Bq45M5_WG!c~RQmMsS0lDcvG9} z`4*d0%fRA0l3@BMuBe#0k_Rzvq67^UHHQeocxSm)fEbvx&@hC^C?qUFE==q1-ek8C z|0`pgt2ua-(*e(@?Un63mo7JFe}^LRbNE7PzkfVHmVe2Z_a;-9P0b}UGkwf$muUZM zfdSYOILaDc7^F~E9-oj!*x zWR&3VjltO!r~!qbS2fKl`R9}}R|0xamp1PoYYG@QWPHu9Z_i`;L1gaE9GU!!6m}vc zrqXBHGkOYYT4$pn#f>`*r?%HeOM*x^7O=RcByUcPZLIsJxDuV2PEqzy=z7gCii8Z;4kmeWy+HIr21w5cNXy<-Zz$a(-e#W2oHU;((_y8X4mwH( zPLW7q(?ZT!SOIY82buX{QFveySPPH+UjVk9DEQfOzFB!|x=#<5i}mMggw$f=d>Kw! z5^tu4hR`oZnh3MgzR4nf3T$HbGQv-&<{5WKqv_}q-NHE-Fi!0iY;ea#>~#|{f_Jie zbEAzcngPTU%ZBxUfb^GKF;0xaso|g~l^KBjn^KQz@w@m{3=nxerqZNr0`n%U4HO_^N zS{S>2p>2tuKWw{F*6Vdm@rtzNr9ku8nZEh!M<#x@DK$&6nKg;dz%@LIe(L2*xBJu4 za#muja{J(frG|CzQFg?qH~NBIw>M=S&4bx2V|r@BJ|j5uU{y?0)>7Qb0Q6fuR~P6* zb?dJ~BM+Q|T>B(uT0vRx&I1hhBS*8J+o8PHd}r9D5Mr+VejK z33uz&58^YZ&mtl6(%ZuO!r%GQo#di1L-YARTR^t*;s%;>M|0zb*!gfz&8oGNP~ z&LVc=CYdEcx-Zg}r`edm+?i5sUY@ptH54oUhi?8}cz(Zxw_E&?7KlDV95N-mSmJOE zH%>p={O~l?z3rI#RB5`?ac&i>cAhd4yEAi@e?>`EW{U4!eDiy6(TvaIPscY3JY}%$ zif0XC0vZR;pwR>VE8sJAYqB9T%L(i!Fdo9sTG8rV^2*Z$Acuxu{_G}I4jQE2|5vK_ z-7uY1af_~^a1X;l`?hD%DH~?h0KA`^s4DSv`tmtyNh)K6u0!$m@RU>DyHo$}yH~(` zpD;Yj>*s^Cb|UHVuTi&jhjaYL1_(&&-$xa7_%;abJ0x@5{lj5Kp$m3bqmM9_A{&im zPjvZ1D9`&v1(K$W53Y!FNah)ydiA=E@zNQ?lr_(aHJ#3f*SSn}D%|<=G&8V4D%>pw zc5Wn6a90PNeCZ2*VD_dsg{8Y zmT7Ijoe)TrviwEAh%oQf-<==_4L?jm7%S-|R7QChh@a*jYLxjS63?iR(PKbG@x4xt z#aFOF6#IGGj;(>dE-iyoN_VZ;7^0Ucpf;YMpy9(1 zkm4Sy=48=iLT7E(yBr?=aS6ZlU~5-AY7b+BtzF3N@aE?4&Y?po5>=Yd8LN@FK$QNK zat&O2N%G??wh{S+&$58jkp*SSymfd&I2T&uAIcCYM|>RjF9*MZY~F(6eRul%eIn)L1#N%f;($q!v!`HPOTm+z2imMDFcaXwiCyo zyi5_xd$G*oKO<=yyZt*!RDfQi6(N!4E4fKzoMiz_Plde^I@-+M1@d5Ja^=`~sfc?jisJAk`=*XzU2-(7LlSK|fI8h!Uj>KiLt|JX0+lE3-+^_@|T&T7#P zPTHS3ahT{&c}gW|p1B6Ha1mlmG`z?zl9f2{BXRnh0N=w{&V!9GhP8;cue8Lk(Ws9hRFUnKXshb%ry<4y;9y!w|@ zHi0=!B-G;R%c@xbR z7?CZ$zArAG@dzwMx^V6Y{}Y*?%WVFx4WXwTdX{d-0B+mqAa~|By97MqYpWI$?CvwC zO>mVWB=d#gS@w_0V`k0}ts*bQQ2bwirJuG|apRTHwC4YvoXlRUlC zsqD4rbj*}luUwBuIS27@grCf+oGi^S6SS=Y@R~hm<)$+VGRYPkt=c6&j zj`w+U229#S>A*i60Gqun7BW3`4TtG}+a{9Za}VvGBhb~89c}jUMDrT!PnJee>t`TK ztr)q_n54XvjZGUJ^(x#}{dXZ+g(iX$f&e@#a5w0S&-#2DdJJ$0)M;}=6%FmtLqK-6 zk!$q+^G5-)=o6RX)0n0X3<0~@QA0Spj{d%UNqAd%(5D4%?+{Qqyp^p*2AQOa3&2K8 z7%)NVt*dnEV6NOjMA&@dfq3OzVJv|8epwXiV%nR-jA*ehIdWH=?O392aKAnc0Hl54 z{t6-v^fboZNf1fECAmUTW08SN6&;vH-_S#S8XJUSVcHyeVe7TzImAg#7Nar5lZ>5e*ts~g zzqr$7;`krD%rM2lJxG3%;Z`BkP%d@ff+jBg#NVC1gSX8p7zTdINU)d+<5a*OfZjDM z<28~ZvUYy|CAwQhQ0G>*#BBNy6?XF7hNSX;Y((5a;fYM1#Q%O4dL7>kAO>V8WwJ6C zZj`my5W9NixV*d0yXPI`|5vzI8j&K{km8r&g=7ipmYYDjcT+XWcqnS`aqhVVl$eF(;82?rfIjFe=^}gdmqHsG%w5AJVq;#D4Qd8-VPL`35H4Uf|@A+aR(Wb}jb6+0PEA-37|8bdVSYoD`Ew{Oo)^QIshGNhBpOS=&1 z3{4VpbEKgW9($HXNY09;npc2lDpEhSRSgLRMtW1~_W8jXkiG_r;~pJ}rzg8LdS*cb zHPd&=pVz!e>H>!7zHb#l3%>*-t45f*U*i-iy&vg^98_yLyZt7N)^z3NoUUPgl(B!R zeH?FLJwG1Oa?RTT^JTRWe+R+$vU5P`HOdiCgf%I+5Qf##+}W$bnQq+lf`9cPC`Is3pFi&)9|sFKSlhweQAhfn{3@g|Uz*<(<2i%Hm!O58TR zY|CL&v;qA16cmidg<@7)#`Ll=vA>^cDMvM)_4H9 z$zS3_G}tdCrYuY;C1?Qf?C$tptZN&3`VVajyo{pjD2X6Se?Ok?I4nP2C2J-njY5I3 z%$2-O%a<80zz#Yw4;dq3Qt%6bY@6Ta_Y;Cd-7hSk2L_k<)*E6s|yOM&}a#xbZEIN%_m(U_;WB$4r>1_^4{B=4{&Snrw$4VWz1eoe zKR|OKUT9guy;0}Rnmd8h2b5p{VxX;f5gblsXrV9#t~MZYz?Rw9#f{1UI`kNh{d;^0 zcdYUCF7TBQt6O#7Z9_t4g&&tgOMa8ai`eM)z+IFfj z16i@&UJ0pOCoYv+s)CyS10y%SDR?ct8Rw}HRfeF2dDm3UaAwREVoGmeJi=;R7%1N( zc`2n}V`k(MSkle{gf??hR89vixgwbx>6b#alu=8OGU<;-vnr<>7S>B_sbkj@U{PN{ zVeMmk|1gwU<6Q9$T0Lgu>CVYuO6r7o3RHIvvp>4Fl^>btV)6ItE=Yeqx6@ zrj`>;^?6fKw+4GPd5bq?b(#CbWf>FS2P10}^BL z;`CU(Xwm(p=1mO|=OGTH|Enrq6YqIAVgeG(G9A5&M1pN+_x~QplN(XgZ&I7MAJ0Sg0fx6E)`CSM5z86`0E>LC8tO zl4ZkmE%GrJRI>2p2}m4Z8WQNekj*~e!idLFeu?yvJh_I+@qMNP7X(%FER)E=5MTbJ z3!M7c_K{)U^3&F2X3PP~Xcc41y19w~V4&Po+D|D(E%CN#=z|j{GgN-MjxFbd`%krO zXW2U$-&;Z~KchAqu)ni>Kzz zGV2C|Db?(-5Ob6ydqygh;pki&1ft=O553AAY2c(RNC4Hi4;nJBku-ProL9&YA`!%e~>XNZgW<* zXxIETi%m>(^(u;#^C2DOAL1}6KK?QP|LNVWs*MShL}{QZvQ+FM`R+e9YG`GWyz;U; zl~Bo9^0IhfI>M45y3yFq2fW2Cipym;F%^(y%(Ze8BwhSQt%ldoi+r-|G(0Z7+r9zJ z-r6H;qbXcK4)s3aVoPZ}XR@P?Ak3~W_JU8p8M~88eQe|44CSNn_QC?!IaRkR!HN{! zKA=ZcG6^s_B zU3}v6H~HBU}H1^bVF0n>I(RKAUn`-kMY)awar<%6Msk$G>{D zM;DS?C~gc0)>Kd*RR<24(cn5fhbSh?clcj4piliIUv&j)Z*Ab#)zdrOv|u zxiTuJ%{Awz2?N)l19+YdMg3=7-T?qb>vnOf?o{MKbHzK6Hiyg$K(*$aRcgOCX^<7a zCVHXL?nrUiK>FU;4ylc<86Or-C|VCN^#Zb4c&#%*mUVb_Nrd56ec!@K79qyb)jXDv zuuL4Uhuxf8^gmyozm3nZ9OAl#;L3tvsq9y$_DJ&YLV3q^sHIrOykk4g3c+_2L6WL zbvj8Uqq}KkC@^jcSIv0J_)zLIlMM3c`7_y*L~~oogeo3GiL!q1Z2P<*H>I_}k9G=eI7JlT07Kbo06wNnJ6p=syj(8)d~JFbvhPs8IG1(k-PouHFA_UA!<(sptb(c zvfdKmoBf$UYm{fKuI8v<1^<+=C$1zJb)B(rRv-1A_z5L`r?{&T`sy=*ZbA_a|Ua#k}k>WQIn2 za7G27Rbuc?6M*mmR@FW}Q+c89fwVE|Pjz~(lo4ZyL&+bnl!}*v=MP-eD+EWMWtg8a zC>f-Cjp&nc8K@osZbv%v$ZqDm2tN+4*$=8CAL|;hqAQSZmF4fINV67-@hmQ@puV;@ zRLS{^*o2(!gk-8XR9ChEfj=WHk1}+vJOs$j-&<@q(>GMJFG&V}cc7FB&cK_*?|m_4 zV=i-PLepM@4=_aI&{Ck@^?7wN$Hfx8W2jcNSq56;HfQDa9k)aeY-EcP;`MfJlCHE9 zZ$^LS%0o1^9PicZD=&ulmYh9*3*s@RzEGE$?Tq~ySBDeV+UpD#RfT64Se36?(Ea7C zDk7>Yx{67UCEj&lT5D!r*-{dg&j&c_PfHV6FxTa|`M1-5 zC;QCDCLmMGm!0xj`Xm3C5aPIh4<}h1eIS|#WLLa#7`8qyHVpHurG>o7?l=uk*XDuF z)K&|U@}q!?z-V#!jidf1ReSk8IcwK``D8@NLQD2?MW-0a?+;oNUsVC7fTdWU3Iku zwS<9T?3o%B@;fWlZW9h&mkp;wh|-j@3B&dMa#LhT!r4V7vgr$k6&WqgI&6YKzerBv z@k*0fp%tX8XA5RNoe^E^4NV1|v!amzKv7U*`bcJ^XVG7dd=L71)s8abQxP4QQYrYa zb)aX23ebi~v}&`kuEUPNE(R93zKTs$tdhplnZkOATeepu6JQl_0iXQB`?bDjSWxV| zi#>0&xrnwHbuo?s#hT*A_A_0AlRvrlsgxC;2i^-g$DGqTDMmg%jQpFm02)@ z^0~_TB8>Gj)-@zMLVC38HGv-)d=e!o)nxn4Qv6Lxpd6&Chv4mWX^zDQky87awyHVx zE7y)!f39NkG_3b5?4aRcR4oTz0=*s)AL8;SNL<($9{BpT)#t??QcE(yjgxaZsgCmk zu{Jtj*);9{3QDbf%J$j8$mi(V%>CK=+9N8qvh;$T`MQjn{Z$zH@;EmJ(6CI5-5(V{ z;C5P`oPwyX1fQ6P#Nf5vLT=Xh&Fr`5Xk%Y#-~oIQmx@oZ7e_4ID_v&5SbRDK^y-v~ zf}2W9Z-9cXI96Jt;q>hC#^JwakWM`c=EA!`iuU3g30E41zF;c6?7msW(K^}-A6`gj zu=hmLD*E)tmVQkOu#C-T9j&nzMIz~ivxi;zEdO3MM2?)mZ}V>W zC0U-TEiPjp64Wv4Mk%y}zWX`9WgLo`BQNtuNDj?6(xe+NwqC82*RhxW#DfKm96ZM3 z&F@EQ@a9OSOG3tg*X~TWht!O8Ov|HY{r37D1pjLf!|oU4AsGgKF%^@a;x%4&W*1*s z(ZM}-@YebK&HtW=0?z&lWFDJxBgb_xs=iXWtf2pDsNfJ^o68atKcM&E+vVrpw7=RJ zdsRdZoS7l8NR@S?sQuhiUnM0d5YM-VN--yQ&>P>7mY$W;Nc}`zApjL=hnFO?NVF#3 zw9NqM*|<;XY8vPDv3me8LRAD-OQIEiB`5n0^=oHv;do{3)qNcoo2Qr1Q_$)T(uOkp zC=)$8w?8#3+(+SUE3v4!o6&Mh#PG(RcVTcVdm@nGl{hVFZ;nM*y|p96orvl|8)!Ji zhx&ayaQIIp`R(^FPrY5#oqX5h9R6Foo&g0(Qvy#Eb(*kxn3;PDZ%lE(?d3mb02HW$ z3RXr7Y>-X!O%-(bFk93q(4Kq=B)1e(A_1;%j7*qIZA`HM z=|B-C><9&7?#25c&8H}M0?Ly~>(^p0N;#G+1|}aNOchOv!=}xswO#@w4A*-g6-$ci zv#(9-oaX1PFm5StTD=o|=Wk5@|2*lOHMwusLrNNetky^FbJI9jpf%%5oyp2T8|b~P z&Sx>u`S}c082OA+NCa2H6xx98Cl|hf@NjZ%eof&?U{3vLvlh$0C`Z?CDfGHT9K}&E z0YRBD2J1L##1JSkdJEcB$f-~aP)2>;m6!O*v48Ynx{N%gM}Be|xA_v%OY!Do9;Pw)1%)i}Nw{OZ&o0GJDJc9@ zB96@3o3Z6W4K>O&0=0w02zXeP>I&L3 zuxXrmk@hQM33I;Ipi=Ymtd9i*hH6PvoF{`{z57Er|LGa=D1Kv+MLiq-UA&1M#z;rXYv%0ux->#Fq;C+f8rY&7FgvLK& zb*E2zUSals2LW0vQP1CAL~Ybj4^{0I-*DAPr6MU)kNeET%MHSJ{FOobvKiDE>Mn!7 zzSrfC*j!M}cTSvDGWy^RR_S-ML>=jE0pxa$>Js8fh! zdG?I1v1l`6swY7yRwnZ=lolJOh{g2df+=hR32TBnu~xJ#`P zvSghm-?>l8kWbon2=UxzBYWjhL5ZAdaXa?;J8o8t3JzUz7k&l5{r9dc|KN3in_q!( zfH5TDX)tvxx{|;g47l*9bG;!wY@gNdzTyx20QvMzzuyXx0>^<=9_9MA6H}q*3)w~$yevH{~X-7)APjjD)=}WUex^i0JpCQ4y zW1n?Xr7wMT==mdw-;x7!bowh&Ps8hY6#=l5dS4gB0d|qV0L*$~>b6d+CZ!8B#+7Iw ze7tZjT{+vCUakPL$L5)sSS#i~wxBw}c3>bmk&_OAeYMxLY1Rv4(W*<1hw+8Pl70Nk4cYP5yW6{?jyb zQio6-(&&nkDm&z(jf_$XM{3&QX;lf!=a5EFR(ZB_6_%!zH}VOT-78I73ijj6C*+vA z#I`JW(MZ!@qt}@y^t94&7X#jX=-YOlk)d|@EICRyuKSDc_wZKCGwZ~_J9 z#`@@NbHFPM$bbS($VZ#^QNH=9IkrMyNb?iyJ)uN@tA`5%`SZki@j8FPxYF^c*eEEg z^7@yOky$4~$xk%~yZ+&(^B^(Cnkt<@9Qvguo$MDUf|B{w|61V^yE*4clG~RC)foY^ zxGCQhY9Tag3$uqWO+irIX(b~o@z9dctCX(zocPV~evKpvpzx5fQMt9Z@V+;SeqWOV zb61Qnj013!Zk#i>spt~|Bnby>mF>Y;iP+}Z``p@oXYxGHw2WsLA7Sq-8e|7LEYVm+ zRJhYSgh*4zPlz3d2~NeM$rEzsAfLN(Oo;jMspSp~OneQKy3A_U9e}^*SsPro?+w9x z8KUaPa;*3lIN20Hq=vMl02TP>Q=wl)jC+-p`kx_%EX9!?Qu9mmT-Ss_&azf50>di-^wE#vOgn zM~A%TXt=)8i26r@FH4Dcx_^Eu*J4top$#~#9JBXz^t{rF&k2Pv2=ekJ?J<$x`m`cR zI{6^tr@cIl$GTI`;cQ|C-;h15I2?AmfuGb@;PidLYfnhX7_VoNe?9(AcBResL7TOo zgF=oKEsSi4m@g+xxNl=K@D;fRImy1fFtfwk;)(bfbuh%;Q;bD1WvwCbYG^g=4bL)%1@hg!IbM;N~w{o&ir4ID|;#3_kM&_V|d$v333Vf z#(}w4(dn;e^Jg7QGy*tauf^Tsn`x7=hUE<7Rh!xMIyh>m=nw!xGJdDFfRRk7mP!5V zQKN_Ss?Is%Hu7R2k}qNEw`WT$APc`Q0{F$!GQKM;LcH3^hl3;O#B6?L%_b|0bPliyx)*S7%2e(_%WW22%R=R$gmj6iX`>CqLb)LU%j_D^y2IpAF0@*v@a`^ zqzn{M>*8lG^an2XsOr({_%K1eZu3e;t6>HQt@h6ls0wE{s*PJ;e*y%7kKcZ}Z;J3- z-QE%3CY{8vZ`MR_#!+;|dd)wy$J}HJ|8;DcBYdSZ8c-EamU272G=73GzoYk0X<|7o^RND-3EP7vStv4;cMRkvU z3N2_C~;;j?vHW44OblNuMMoAY-A;+L4-mR~#ZRo~y96Ms}VLBjt(gyBd zCrrMICneb$@e^*`!}mjw7fl+FXf0`LM;ic)N@$Z#ZYS^RNjfNkyd(ffNZ61=86k%y zVFNkG=(jc3n_c)#Zt?|mXVf33*GkdsRN8X>CEbEPKV}}AkE@Vi47QL$%v$RUeF+Nr zUVsj1gAh`ukjweL=KUD{IT6t(#fq7Ck*c$k9}86J2%pVdQMESRO<^Y|$lplE#96*@ zr?hWQO^%YrquXWEE~I6)%zWiXIjWSoQp}StjvZG)Y7~H-J|p$s@$K!+TY&!ret&9b z-_bm-!eZsnPdpia3i!LD_MBa)@b4R`m*_Mirku`!K{e|950Lu5JT}vE4!7DOQXV@i z7ktV6um8fO`ZXO2BU_G3tDFF}Rp#s)#`;E79Z|0g^MEAwXq}IaC(x|P28mcwn9eDX z+RDSPj0m;{RAnBZ(c&4(8!D->x-2T(1Vl&7n1!j&j)vL;w-A5?_s$ zAs^rL{94Kc^9cS=6;`R!Xo0231xKvIe}}QL4(DvnXd-pq#Pd(zAtF`lMcUiqU-*pmy?8ShZtT=YIu10AGS_urd+$rEAKe-Rmrr)5%fb}t z2}*diU9YhW$0dShfUuMb=5d^sM3HeWh^*8%g7hi$=NTFT$0#(SX!ZNJ5brp{|y>in-TmO%ZORUsL<{JP1P$=pP1DCWZ0RW^U`&0gv_jhcgC!pa%I z!RgRT#(14J)tO>j%I4}oYw@*3VQ{t9OHfgfmX5KZWlv~udMBEOT;tnQl>{7ZpBadv z1Y(1{#|(i4Q%cP)HwDJc>~n6jIP|T9ErCXj?}6O23#7n@T#|41k!j^XuEtVmmvFu1 zfxJ{kmV_bKiy(HD={lwTHCENxz*|$)fMIM2kJmUGAs%X;Qid?lI_6(dbJWv76rWpdgJ z#0$$gT;=>~KtInd4;dtCfh%p4<(oAC`Y_?z!6hxBq(`vW2h%v5M&EHJ<@%S9gL$6j z$8vB_P!iLyY02~mi2YI1V_Dr@xCP62n8mmYnENz_zs(4Y@nQcyQBRbgf319H zm$P5B8w^+@tl(jI7rkzyDay^)mk=g4P$T7g6e0zA$mfJgJD$=W*QT~ z5y@4O8@H+)J>KmMtVkOiVotJ{Slg5Xg{cIYSE?TiF}2VezeA)>i6AXD1QEqjH2p;F zG!ci_Duzg0$3B4|s)iT|DAy*pCo+1m>W%9tn3l1>PAvqdgqGovu1p`2r2d;qbun{K zD$vT*v(=soH<&Dku>YopM=y#%ps}p;9f@V%=K?>8HuK2!C^kV$4)R4rUY-=Ee= z&n*X^t}g~AB*|(#I#*@-(s!p;5RGTuSGE;7Eiv#S-;nB0^Z`HTNC||ru5ub#?RrRE zu}0ZNpL}qZrEZgCx&nXSTF*L~*;aA62XAj_C5A(XEb9}09!0F1@{_v-Mw#xXBKkK= zHHj7ERg>aAJSu|AFey-vBr*DIE9ZQ-E`FBZBkvt282RYS4*wN@F`4`pNtJRJMrVGv z>*_FWR=X%=HD~c67N5OmbOpNLg{l_?+)b-8T*b@mPBePIWST6T-sB&-o0I7fs6!S({tj2$JFsEjZU`cm{MGJ#g0aCj|gejn|6)nJ`agEuVMF622e`LgKJx( z9GGQ%WuvtrzmwYifn(|so#>tFs3h^c4i>tGAX?xXH$tmt3jXPYHaNeF&MUP7K&(E# zONQc|=yytsYfs30GgAbkuv^9?!$NQ%qUATVp zISXXuZFP<&0Z*$&2Vy1n!$o%)QoHU8P(*Jn`^Bt;-^<;fu##8`>;Sh{Xmw@Ji8ZcQm!)|6BkoH(*JMf?&!HQm*KV+>kDmGgqXoMGhBN~L(hJOv3eg3^*tGBqw4>nJPoE`Vc*N{!S?NH2n-V)VnGa*W#lhKuoG%q-r@u;x+su zedT9<7yXx?B9B2m(bz|xLh&B! zVj%k3pH%B%&}@t+9o>*f=UB!sRTiv32WBD0!(9GANJS|JobZn58V%-E^P;qw+ASM4 zMK7FZ2kBYt%l=Kr69+N)t5>wCgcd6@oRDT@eX&Zv+iX4akpzn!pv)~Qz5!8q3w*UY zo(%UMyqRkL)Uo{M`u3wRyl8Smc0sIq{I)|${^2L`+z(JGypuNwrt;}o(iJE@qg*L~ zq{pL0tHFum(iT(?MILh76!39?;3TB-i7{c%HfQ9<*Yd8FMuM%NB(_BN`Ji<2@#t&= ziuY)AuymZQNYbtt3aN^zq2sL(h7* zQ1&*`ZBum%TB$C768?j`XG$uJb?v%r@?YvLMmHHGH$eG1V=#XLx1a#+X_8Izl-(Oz z(bQyEY3cVVtl;(p>aTqYNErNMcHI4~u~SHp4(H#YM48Ho zR&{uek?jd92Ggx=K`sZN2dT_aNQO_dV~tz3AA5AA;AVjnusol+?G8|UeVI3$V@32S z7A@KkiB&7XwO8=VuXi2u23Pu90G%G4Uq`w8ij+_WvQvfA^0+7Feo9Wwa8#qz~)kVl!cFGxC3)h#ie`s|aJ!+lC3VlqGOpc^x^$U(ST_DWUxs4fKdpLRm>hznx{5JGifu?0)%zM`%#r&Pn@ zXRRS3WG!FfN&$v-_DEA?+_vxXdjM=?YB~f7jLZpH4VZ(pIB8^VhevKp28un8BrSV1 zf`NN8p$9t+N#`C?0lQZUpr*fhzXmurJ>3)~Uhw_x-%YI+>TP8BH7WlYY1$ zs1D~Hkg{^jQXW`GoE!}OLq&?gz`skJblt&cNeo77BULqfpelzm-E4m6{?Rtj*fxiF zaRUIoE|U%7k`XY227dIs;1yI!!6;($+F?m^((ooRIwe}{C4@+n$kBNlFSk!}y~_7~7Q!>;m9ux1yO zKMQmsH~TAdh@R^`f7-n`={GZWuW%0TTyw7|<5&^z_Wg3QpV%tbl*1rgqI z`279Xf4pt+(Q@ZexOs!yzC^;4G*_X5(8#ghMjC(!sP#OB-j;Ei`^G+Us)Rlk5P)}B zZ8f3-s{D~!k9>tkd*G_#W5FqyiQ3ZjeV@*r&q7nqJ-Bpg>kf%@xn~vxoa8wAXPg}Z zM)jSzgVG=7M;56J8eCHsJ8!Dd^mIVe3;|Gb1&7DTxNMk^Cf|)Xoh_&JRJyIG<@{IC zwE{a})+@M0bYf2Frk4g-_c%|?zWcB#%{W&cv&`+d_{kYj9JQOym44nd!^@5a-N0&) z6D&6EUl3=B1J*Lz9D?vI+Pv{qE!K*Zqd>6X z8c!b0rnuK0yi4RAg#nIpn>=7I4^Z5f1Q=b0DNA_7adcyKIu*Lje*XHyDQ3JEgOx6Y z>1KjWDGJuHE{hd$Szre`SL&&ADEO{~tu+sEITUj*lZR3a^YkiyD3Dj~nTA^YNJj4R zt)z5t%@Y77t68*_MzcpfY)S*yhY;0tU8`7Eli!mE(SUDgG9w7K3%zx9<=w6E`k)o% zz~YfGq4;Q{UHb{I8O1n?=Dc4e>MVd2LY@&pw;fg%SsZ7!AUovTAF!=?O}EuTDzaR0 zcQZ!TGnfu)YS5sI{BqQJ@YbSaFxGak8Udq$<6s-ew&5G`-mX$Amq%zQ@Ks|0PqLQ_ z7@}9zT)5_uU8ak1qIoMRPXGkFEZ0@r@|y)0TD5thCMF^lmpFFWQ9HVkKH6|e>Un&3 z?xaqX8Lk#nzv=0la(}gwIGiLJQm&=~4cz!n>W86p?jjh@WhBX=YDkD8 z-+*lG#MjXZ3vXaK6i&Q!Pa4amM2FJmR3HC?C8`{L`029s>z!Y5W@r-O^ywMDa!idL z$}Wk-jdv|34ABF|v=S~E!2=$X8_SI;Z>`}?MsRJtEP*rkuC2fycC`#UnI8M)0|5Bb z&K-!7lAg)prgflVz_z?VG04;<6y)CVjXH(NCZAddO{l3ICKB3|OvI4~@NGuI4=7tY z5nI)KrbsL4*odmAD8-}tk{B)|E>ZWW4;Yy6<4V%1hkzxF@>3@4iHo)yMp*hR+h=ie z+Oq#IVj0`}PS(}We`m4AZ3;hkg78YcIhw6z4+TI+;(w_0^_)a&G6i&!UGO-VOjV%O z|7ji1_(V{yBxplubr_GaBNkZPiz36pjLd3MzAET0NWut8mELQEgV$U(fW2Og{c2eb z1e*sn5>pK*!?3j`5YLyn7uvV&As54}K40bQGCyUn#1PZ&>d`QSOoic!|11>MYjQ}_ z=gngJOXi>J7cNOp{B)=CiB(8nKKYJoE0&S~__8okTgYTpD1B9Xx_HReg){vCwbm2L z7{2$e?!xKvJx`s0)9aKpG2E8PC!ruxNC~0NHB~w&6@!OJ@1`CBlns1}c`5+bmj{T> zFxkd7XVdDF`M)la_J~!RN6S6b(lh`IW|MvubJ%p)=2sO@ktS=7JgUsYdVrZr{&-hrL#GqD<$X>qbk=*5-90)h zdbqcor@8j&wb%Ib6pe8JNLL7g6}G`JH=~_$n=m5ZE?r%Ioe83CQi*17*lOJC=C13L ziw(9O!%Ym#2U_baNujn{lGNk0=)oQtJ}9LOrt8Ptd7fg>C+y<*=(3j|J@)3Y~G{7Qy4)Mzy}ATTKXaej3?yEK>CmY)2XZw>F^Z z!-4RW-Hh`&uBYo~6gHBFp|X7)4OvdcMi4EJcK1poR|0(Ugr)kp_zSUDOga455v;K? z6|#!+%Sf`m3KupmTY8O3f=i>}2>uw=wcYJ24z*7QN#V&g0L!T{V9X9HAEZzkA@J+Y zJ3`^tpba>$IUthEVWy|dgA!G-f|LMOpJfH1k|1*Y@|FtBTt(?obbLemkX?sC6rVaE zjZcOvgLVi1Bin;#P1W?AQl9S2)}XlDN2#nk>Tp8VxG|)F2+0Od_+76CWWznsslNJ; zfq}N5G&>v5`&>}lh|s1DKFOvR-7hKI@fO&><%*>3Y)6UrRD+ApNu}g!vVlEt$HdMW zYU=Mya-rSYKSuF`(e`&e-Eequ?KTqSmtU;qu`jjaI??R3Z z1pFcb*ft#ukZ}-2N;gr#6D0(!#4_oMX5MkGL`hn&wkzzX53Lw@xeEyzHLEc9WIC>s zQUq{XIcva2qH?sBQe!UJ$)zAbj?&=jn$?KIlTo%m)^KT0_!5`d!hg1vi%EQv@Y}W! z17&4Lo=7a5MMn+wvyfbvG^phnA7({z%9d7JQii77qjDWq7Qu5M#=$TD`~X*QV7D%8 zCr=RIv4mI7bTaPy$-Qt_o!GOcm4AD{Z6WI2mMFxr!&nIv{|gJOQwQn8(?>_5?4SGompX6>mxf6^Evthy*1#`-(rd$c zUD*P_n^BWX3WdziJTM;7GCM5HJX$w^!X2auSZx77TO%a;q1T5uT5|ZMstK1f-x(Hs zT4L>2U1cPekjQe3l|0dh=J=C@RL?xaZT*>hL-yLppN{=teTDNpU-`?yQwUmw10T>0 zyS)4Gf!1eLgu@7a3-mqqvqv_Y;dVALQD(INsL?0xH*z(op!{5S9+x{_)TLPq&c_e) z$O!kRf`GiNbLRhF-XWW3QAr9EJOGI(x-#H+@cHc8n#F~!C#v6@u`#Tr5Q2SEqZ!?R z!X`Hi87UHZQgu7!4gHOk_NBEHJUM!g}HwBy%LPD}Lc|faV#*9f)+>&cbvu|g|~IJ>=Hbq=$`&*+mQ zJI%X-=yE}@JCB4qJlb=a=x=9^Bp9sNTG4v7Ku)Oi9+ zTzm4-bi7JxW{K71MqLUD4s!r)JGU=Uw0-3*4*v4`Q$}z*Xr?y2LIraXkncUHwoP`y zmHIA(MoL3Q{5C$w!;7mA51f3oVm5RkApC>*mR?{|x`9>N!t?6HN)Da$Xuk#9A#0KA-O>S^Du)gF!w_CJbk2WdU zZ>+XL04QuFuDGA94JCKGGgwji&dihLJH2}EJovx1Wy><3xTI}zxGHBzX~`FER1wRP zV-us>x6&Xovlo)@O%`XZ&UcNYS@FrfqXDT3N*K1IJR{(NR4Ln$3<0JAf;?!Z{1AwZ zcp4l}uwLW3Z>;<6JQlh9GR0H9(nw+g`G@DQY>z{3^DR*3z!Zt*v=x;1h+_$IyFZKF z`F-T8K~;2<$;$aR^Z?-ABo&+4pdDWfjT;eEn%lZXV{r#Ni9wUz8E47GAq$p z%ta{eZX7o9f&|ynNZu2TzvCtd@5gbv?Y|OcWk-iTC$dVpIW~Ch>}$V(1%RXk@v+zY zp9z-H8=2i)%Ghxr~fAl||eIgjQz;<>c%KQY>#`{|q3(-@M& z9dhhKnAyZ2*FV!gomyn^8m0e^0h!v*P$U=uwMoQ*REJmqqKCe%pQ2EPHA)Q*XE)%S zjfH)KN~Y4)gasHY;)uvSC$U;H*s>31;G>F(1HlHEAc8od<-m4 z)%%edw1Wsr{*i?GU>9~f2Gm^cwea;ad-V2x`?9+n;Q9|`52?Q6eZzQaJC^}Atxu0> zcY!>&xX(cTX`BsE#KfMV66o|9^bQ7U=+oH$Q`p5pBd~yCroPa%fiJ2D1BR=)XWjO&1!4^0e8^+!gVf4)knoTT5$a5lvHpo)O-SfbN=1DYio?mw&&n?OC# za30MUPHltZG6<}CFDH(Gp+n17T`AK@l#YM z%8|!!Vm~~EA(Ku)Gm(a(tGzJG_-hWx;0j8oS63>1)}FMRxW`|;m*zzr!(d41+28XnNB z+NYZ?-$#&d0KPNf)deX-TC4s>sjnH(xhc)9l|HjBE`PEw7drK|S>*o#e+vtx@-3Lg znmDHc1116ZPgxjyiDdfzr)LdIKY1LepU`gD zXF9@XoBwE;;C{!IXmM+e?Ve$`5fdO*-1X{@wt2QQEuyL9KGcb<|E>G0xJ^rBlu zL-Nbk--jBNAQl|8hRRi62WrjMH!JEz_K<2hvxE9rP1w%+$Z!e6Yh}w$C9vId0F{7? z&6&sQ&lG<>nHN`af4Dr!Y(~M6(+$&U7urTSufCV=;`=3EBRY! zXJrqG%k0mA6LiuXZRc~SFHqL>m$SuLqhyYS zyRO=>kuQtZUUezU%PGyX<4#0r$HS<<`vKP#(f3sVvMVJn`qUvHXT;@)GEa!!-7mWD zgW9hj-jn$LiNYE+Ws&Uf=k3U87JJdYG+|jv;$Es8#idGiKDnQ*usW_HcKIX8@eP&s zYvRM<3fX-JDRF!!sUKJZvcjh*QAmS6^k{6<^kbpjx&R^RK|zUek412HCj}sgc6N3i zYzOAt8FXlSMt?y`q08DE4|#WmQEZRy?_wZQ_i61%{oKwLJR?HWlG5Mp_wlMG&5e2p7;ck z60P{78rqsG&en{CR@wQGL~FOtGoH8piiKyeZqTur902?JamUybW!NZYID$rU2*{bU zUJ9=xNm~jBbQEI{#GK2uK(t#d+0&;}VWyYnCj#CSr`7DN&|UKF)hZVWe0dnutgwSB zL4GzZdP2QFT-{4u7{zzHC}5IAsli>`eBCD47zsB|<2o;*cMd_=#MZ~ZQQOGH6x1#; zC#1c<*)~vJ{eQ2;A&Lr_XKN?>Q2_ZG4c_?FY`If@^IBO?O~KR_Ac(++EvD9F9C z)zR@i=rOP$9M4dBX=NUcuJ^^v*@R6J%Ju;jzhbOJwMRcUQuV8ShJqoNaKLmS5y#zW zT;6g_ou|B-?FZ{O^Yj0jvn&Nc(0l@-c!{Ru%1q8UyozF+EdG);F^g$4;CGJT>&@ho zzmhq`7}8m1%NO%%*yMW91~naH3x4cD2ZxexoEY<0z_6%%4W(6HLA~pAWw{UxLBF*Q zr1RVe7Y=&yEU!b`YmH|!V}C1I2O3sD?Dth)+RE?LmcL#N`$RTopbT>%UE8Jk!7slJ z!9Wln`4JAdYH(dP2FKRVo^4fD$vU;61q;!48x0<;!X+T-i&- z6I@!V<6&~8puMfX{(0JX*YNU*2MYqetM=7`9Hiyn%NY#)co<{eS?pG}00m3f2NAab z00RJBzyJUdsR5oA+=pp@008a_kV2|7ABRLZA)8}eNNFd{+LRJ1aEFOrF!_!*Z8S{P zZY*?|2LJ#n4`AU>2M~!|)&Jd1Z_PvfS|JJ1@ z%88BQqGqzqEjVvg|Mu`wFi{tXY@Z@VrMFHiM6il!?vcVJwq||f+lyWV0amDtsY@xM zu#YJ#dp62y;vihsSg!^K# z(fqww303NAYQW3M;bqZ%97h0q^Tv3t1Z9^=d4& z|81S}m{vP%bNQ}yTwzsH#nK5A@G$PD5PZIbP;S=JP+3bSu=Gq%X^vjM+}RN9Utm$L z4*^6qQwSeb>cL)y-f6anB|B8474Xz6>jvR0JX_;!@36cwE0r7w*`}$>;1g0s%JV;G zSnsfk-HqC)MtF=6Y;7=Hk->ymY);JWYPnY=6OwYPPFKQ=kIn3+?J*bIdJEneFwW3d zJ!87!=re|8aa4$GXQ~C}6cCo%iIg)C9hqieLhRlmCU4}Tr*lF)&7P(2uVxG&8^4gU zgK|*mGhtw_RJC#Bv;BGm`4Din*{A!1wyAS2&-#DtvBc9dMZG0Mi3=WUX(8@eYZ|7S zq-iB}S%=|k5)z8A&&Ywcz!qewr2#`9C=AAXQFruqqEi){@M;$XlU}+K*gJeIb)>rA z(@T%Ky)(}g+axx`(rN}5gpWD{1Y1oOK!h^>0x(4$hiam?Ir1ph#$Y*vY3QGud3LMZ zRWpkiKQ{a~6#u*j*3q!ZDRUoOG|saKNdnh(D#f`eqW{O{+ggD-bKa6rD*)K0K=76F_(4Nr0Bm`u73BUbdFE3Xc^BtHdalw#+$W7 zr9>rp4%6GC1&<6w)lFm>4-SZUQStGJB>j`%_B90xF0g*Fp?36A4 zP}xi-iBe&Z6Y)`O2T-A=7K-W;v8#9R^Tq~QdLmCC@xDxSIXr16bJSKI(B}87jXUsO zx&enh7Undt*13LQpd2B0v`) z2%Pgm?mv}WJu&S8FX3RX#lAK|0+=u?Z_K9@ngOgJrK>bjA+R=c!PY)l7Nvneyo~~S z(*y3a=*O>YXekb7lc=ps+J~|ufDrI160e6ZD1E=WK@p*$0w%))lIO6^HCjz5TQ#5x z?A6tEZgjbw6XJr%hjF87n%N#8;vUi{S`bt)+H-fY}9_NbN%9(w{jxj zkqawcd`7N$r0QfUsGKE4!VO^fzo(+WeD|DI$U6%~)mIqJ!M#|LgKx6F;04g4k`b=o zsQLAWY*}*OH1e*DUd_>=4M>%TZp(S1%tjhc=mJMUQZbNo(arf!|L?xU0C(%nu#vu_mmgieyId2O=m2w3ssGFM3W%}D@DhIOLBkS8+IK0 zM1UtWRINY)n;iM@tsa7#~%1{zxHTZQRBY;)y5#w#>G~y80{#JYRs$Nz7-9`*q8&3?WV|hVgW!)V^n<*K>7VLB3Ev0N z)zH^2JNU3$n&^1f;L7KQ&589cy>}R&Iv-cBcMP}8;n|IW_{7MSGiC1sQu*b z#6(^wOxbZkiM#EHxdP^&@`hw;Wk66JBwHFw2DC-6GS$HvC0EOl15D%%Ly_(1BD3=* z5TeOzYtrD)W`%N(L?ghdS#))_8Q2R+s5(|<;W`tlCQJBZ01*fs6f}O=n%}np0JVyX zGx*a-jOTI&7TsuYB!0S-H4c3k7kw_cCWmyF(rL*VZBPdf#DE9SuLECdAm*Nu6l~*A zZYR7-c-it^sPQeG6HX+~SF4M%uL>N*P^laKmT{(w~VLpvi0Yawowu|*+kvKiL0 zahaZ~af!07!JJy@T%D6B&tZtky(Z=^3kLacRobmP@=cu5j=Amq2R}WQ02L}28n;<* zP8e@fF)EGmwQ-T>)C(BL#SJ7^6 zZm~QBVrCyKeF*7C8f24#Id`{$nh29-=R%f)O2aDlaHg8(2(To_aHyW4=1B7t)|e*t zrLT)MS%EuM`)Fd|4x%?|V|dLuzgD;@z=CRi}=({6c3u8@woq7!qAw6ry#Zb;i+&qU<0vD|#G_ zi%~{7?tG>|C{l?P-a_O+DK|8*v|?ws;$FUG#yttQXQ=%J+Y!XtQkRAw_PwX=W4Pn= zI1ch%$nRKm*N_P+j=jqVm3ml>90qeO0PP(2>mQ6Xd(_feaB2<^t}~cUg1Xd^{G8!C zNP99@$yVBaxdb1?y{A3xH74WOl=fJrQrV%wO#4)%_o*%qvhr2?6CZ@p zR^{gB5ZrF}T4Z+*5q=5=aod?%V%?=`F>tRQ7EUPT1~B1KbX<)Jnx=bSqF)wqu6-4| z?CN7}mJahvud--wWUj##=Ts{^=GU9@J)VZhQ0qgs^WJCl!Smox6Aw_0foBwnXlY7e zU)({!BxEhAS7hKJopi^|XB}zqPz{PDSO_l6<%9$xB$hdvWV@K_Kv}@-^pm)3=)RQ74nh z-ft+@&jKY*s;dOb6m-;$2n)C%T23)76JQd1%ARr}X{u-M99E{Mp`vit~Dze0=wCi1)V})uWDytp3Wk z^~;Rny@`hGJ^ z6`I};(WE5C2o8(3Z3U@_y>SrAqOASq`IC6bAdb3t4A#xnqm4WCAUL)XlXqLi*pYm` zuS>hqy{_Z%8+V1_O}GDB7lw}6&!l2gDh<>9Ye5ITL+3?}Vx{g8B?;FH*N(IjbPLT6 zI{??+vOK3HyQy?K7(;INsJDuaqfs1x8fUgw-mzK?l;N~uQN_wIXWPZlhp_CwcVWXR zcp?c70;-%2nKm>1>vI@*!{J7?h#${@CJr_xx`x$X$c}8u3qnx=@G#cUe3)(5ip>Nj zk9pIvA6D<)3d>r3pdWWk8&rIO6>dS0l{T##@aFqE> z;_lE+7lh=f4T-gg_t+OtZ^78G8?4c$3&Z!TxN{xBQ0K63d1*c-K@Ah4>(z*L71BeP zU$%;KAYo#jkjGVA3?Kedkq>o!ZLeRPiR{yiu65jH2g8|lb)Gle-n;BdaD77d-rYXA z@iB{R)#ipk%fOm6Ye?_7s{6^^NcV#y&es-QGZ-ZAnnZQ=8v1Oxqn5Hg+N6|trn=Wb z4z0vAi1+A}P64pZ#Fh}}Mj7!`f;5l2pvnAsV~S$8!?elADhMN={_S>%8-S+okOjV> z^c{AHy`-@F(|(1f;mzrD#z?YnCU~A{yd--kqD68sT?WR>V!$DBB@d~1@zPKV8s|zq ze0nfg)5Z43P+WY=Pc4#NT91u@tXFf4cs26^+W$vz4|1;v)4$f)tFAr2zSw{xS|ib* zphXDC!npVFRyG}ZZ_%(${#(Q23@FL-U2oEy<+6NFF;iO(E@DkbHYxL2KUEUav6J)> zBw_p!b71RcU{c@NEv5&e{0%mC-+D<~yMV(b3yteIy!cOdblJ49MKcY~k!x<*+l16m z(Ev*r&9L14-U19rj$OW4Jf5oyU5+VCd;E9Fyi~d3Xz;xb4C|+%I34m0aX2mhUVegA zw}~5(-1e4>A)X@)A3<87$ggqYq%*sWFY%%^7hmpWTxwhS zeiw9dzBb-;b{j(0{ooLyK08TXG`?z?uY^u%@(@#&9c?kvoCCXJ$s3lN9e`BS@0OoS zuDf>4$`JH#i07;$$|pCCzMmgERej3!D{^Qj{pk+*Yckj#GrkQRnJPYuH%l%k(z|!n zi}7TXOEOpU*`~RLP$^kYKtOtb$MxIhz@W~x3Mv|ERA2f&gEUj9SEg3vzVArrwF#(N zxEMT2N@_lJg23{C7cQC(IHa~mCZnitR^^BHuV}%fW@@Pe)#9;s0up`*T{C^VW*tXY z!fXedTtAsaZ0urC-y5sP<;CDLT+5<~*F6!p09IE~baAQg9~fJ3b*zc4UzjwFjbVJc zas~N@K6LyptvvUF=6z(~MsWStft<;iQ_%q)a-NxHL!)j2E&$L^m>%PMm2NR1=I@@X zXg~dRMk-j2M>TkLGllhQMF1WdpOv2*YRlrS!aeoe%V|)iWWXhB4_kpPLTW0p8oz%9 z5hl8(I@+{BQHt(G0_@XQ#1LE=3LMHnjDVCiGC%|HmxuuPnmnRR@*} z5L4Vz0N(a*X3hVruV4URd#j-%_6`O1N|an4Lu6b7(4!w7=#YiT^$6gA)0ew5)_)vE zS@ui%2CKgMFak)Lw9V&~D|SM^aY8aX@_o|*!0@25*-sQHS8|2&Af5|>$~2f1y^@D- zuV3jf&&lSlot}HRD2jUZ(;f*oKoA zAs)ORoq}2e;Ld)rT)|5hz}8tv`>>=O$?}Jxk75x0-det8%B;y_8HgMEY{R zvhT!C88{(5{{;OmE){>G33k$4)#gl^ z_yq&vlxvW!l7iVgUR}m6wV()(3LdCk6YFtLWJHb^uz|9#jN+h(i-5?GI6&XeI$5E8 zpRc%=RO2spDO!v0c7}ofsWDRg0M>?7vZ|IBVUVo+Dh#Pc%x1nxCK6=O^5WBpp>dN{KT@SG?V;ONXjDj8>6ta>-cot7lLhJU4bcSzmRDy*CHXP zay6g!I$^gQ}Vl$y*&v@9t4vH@>+SCBZ>4;Vz{Dm9bzx4Y+Z zXJ<)^O}(4jf5@D1w(m7K0bDlu)`-??iNq+NkHw;{6%L$Lv6B7(9bqoJ z1Cx48^4kP*pQ~)STs~kME656jr@p4sFx&t4A}u+pQb*Ke2*ZjNPnY;W%^6XCK%Usx z6zKr`>(4?sn2>2@jr?BCyy-s2oRLIK}z-!5jXLliGZgo?yg8-RRz%qrl@ zbyqccK7r*n4zOU-=6hOW1pcCp3M(}au+xRn-`z zDXpeTaYf7jRv{N`fCB1eG6L$qvi^XMR2uWM%La$DKOjVH(P<(NI_(&SrQ`{Tz3qmd zb%PNKp)0+g?bqeJ&9=GCd|MNFEgw+Gz_iv*9MfhT^*St z;ECtlM;A_2aB+SPrFo5YJXzd|gdo+Z0(Ua*1=Udq)rSkH`v#-p#jNq(JsY=DIw1t5 z<3cM8>$*>aKa(?aHtO=G&(5B9cc~3lrWq1(V89ORNDAB0ar4fSh#bok`27L$Q7#f z_i|l~>6zcr_6U5sz(nH{!w)-rwYQ*Xa0g}4VjK1MCsd1JU4H$@*DR}UXCqeOUgGw= zLWX(Gn}!XaVP{TNo0AJ9eJFS8o)w^d3PKCkBHgk%#L$2J;S2>5i3sLPNpsgV%aRj z%Vc7&=Jo>~Ug5tx8B3wwEnV%Sqg$KNGu3$8?3Fs~xpG%j?CxsyDZA_0!B}n9$Bj@z z&Mm_O@z66^>HQg<)PH73RltxVV)nu8brrk9*g9J}hg7zCm~W-~2wv1hGL6mOT2!yR zqXY!llVb?ES*p)($%q`k%{^Y$@D36MoB*}lAzOUKvZ8stHWr}m{O)MK0f#MXztuN( zYReFRBq>giuLN3c*DNXm*$i0Nb_g%VZD@$Dk+V534q&#{cMIW9)2VrosV84A07KHRuzn+%X+8=-*6pnomt5T!C2adj}!pO~3fk|jtmpGnP5g@2>@`Zv75GjIEv;DGKAAX`dAr)2X>W+CE0L!s$~h4B;5P%NtS z$i;P5X{8oS@?>E=v`J7Hdqe%NKsWi?6Tb=k!(`#JQbIGr((VAuVGzw%{2vwFHHmpc=n;8MXydLV;7|PEQmG? zC8&b)+m64dh>EidO{WT+|3=xdNNW%qtw4J5RLRb)G~lQ=ULfr)n_UhHcJo`>=UjVo zRIa-5noC_=$LK^n47uhDqu-aIYNRBXcWjNBUDd56)4tA%C*Ns^V^KdvA(%hdyMo|x zR7p=J`zvHP;*Vi8FcOORk>mHg*;Z8G+U=YcbwTDMpgmOpm`{wC3<5Y^l&y!R&!E1R zXdjjVr|46`RX)B0-F_A0lFki?Bv0sQILy(^%v`$7>EeJnQawK|Jq{8lcG?*bpi^u@ z>uc8yVl2^>}${&l{j%8s_A8S1UnAF?=~;r3haP0}0d0 zeVv!H+MWNi=U*E!C$Rv=o&C7s%6)a%e$vTvfSrz$ikg~bHUOL;+0pc{ML}sxO;C7g zA`ThvZH#R^Wj*EaS|G)w<8E6>sz{1K>biGkL*Lx{YmckDNWI_%%7YNpeV$DQTk2qU z2=Z4Vu4RR@ivYckgGP>&Vbh7f+B(3)^E3XGM8(dc6$&X{0H%Z`Jq`%nJ@TU`w;z7W ze36jDw@yVuobFr%DF2`zKUg#9`#j+Yf}OG-5(88XWN0!d)kCN=P#%?MONI4dlr6yT zkMcmvh;HOA^vK_~fo#>2<4tZ+e^}_}_4Em{`VsJGbnT%fQ9DAve`Cht(G%@n9kA5r z82R#*o?OzU=t$9!s=QvVeS-W+p8{#$gzRb;EK6oa*SiT!`;!_J;7}bNPRAVR8dkbO z50PqZK!p~mjU3{}(_3SgT6g+CGjLM|A~r4e>#E&s%)w{zm`-IQ=XhLRk4^&>34CG1 zGMb;~WE$HuABZ~k8%HLOqo8n`z+IBiu?Vg z!83*}rg+AAh`7N4{R2}$H1lao>QWOp<#G9y0TWYGYWuHxKqGLbeRWTyds!%ph@ zD)~m!XtcX%#6$5+;3I>tDkq%|YE~#HtV(GX;lGyCIEVZQp`OhCNpFOlNW%GT z0`Q6oW*7l;cD(usoG+iKoof^gqE$;?ztdFUXVv#cQLUx@*aL4^m!k903-Qb%S|*Kz ztrXSE0i3EEka@>b=>0>$0K5U~36oX&JDz%#MJgW0fonw7MPi_GUrZrW`iSO$ix zt^bH8E#Bvu*#`(3hgo&aSgJ=PAsy-^ERv$3GsNNMt?~3qLEVqC7zR}ht4L%!+}B8s z-ZijQ6REy+nFeG1{mu2>L_TiBA?Gya4_+)DZ4{|B@cY{L#N9XVo^V|8MC-<|++wx_ z7hm}_pGIDja?d+r2rU>KMa8+sf!qyUcTA!O@>a8~*y-I96!I=8xwhC6%GwKY7rZ3( zc+`uN919Mq4MRlPYEQ!1Q$EktR*Vg;2ZK1el<|S@luhJ)It;%tO>pMrX*Uo|H$zP^ zmeT#Zh-O+Sw>Wc4glURNo;9ohiNJ9fdOUoF`3vBg4Qx3ign}zRmCge>;ml z$U?PETA^VOJd8qW#|%kizWE6UJ{{}PY9cvCqljD@(L0MyKgsErKU(zgJ3J2Fj*;!Q z!L^j-CqGX8+neC`rkj<1p4$ZlaY%Z@lWiB++FQSBMd-hiP7CsJ0)>pVc z83syy>SXNcQB^{2SXm?r|*g;PLqv!MaLKFr5+?vd?FuzDsEt*K6ESL6Z~27-DQZLd?jh~uGF-l`+(61K%&5TK>-vw z5ETcR43kfPe($2Fl>`K{eVk{FMOc~u0q1=-JyVzo&v+K9H*%*iC^|U;?tuZBJEJ8u zVLfmF00RI30{{R6000Hu006#FL7D(C1QF8ysg@UIzRtpCthFux$t%Jz<1|SSlOw=M z;YhvZN|C5B`_3gkXi8QQ#lQ(2kiBpj>8*|}Y=#a)?2>Kx%LS1FsOZsk-SctJ*>|)L zBs!BF915BXG`-=Cn**~F0jp9hN_F<8<0;L`PvkrS>#l{hm7)Zc=B0X7YymA@Ya_kZ zXA>rz5akSS+m|hXlkr(b`Rp@6!uu|sm1na<2}9&=@0b7^>ie*Lp~)$RtAxc2Q!-H> zbT}|i5i#vm5{EzE^kHTo6cSaqv5dAzzA4e3L!bc)3EU!so}Ai`>{uFK+tiLjcfM>h zDk?Nq4(s2aA~)#Slt?_b1z!wzsXsveSE zOO2)rgEdv5jgs@ehudjo_%T z@$H?&xPH9}j)TC8!c&J0Z|@@bgpP;&ODJV4l`ZtquH4epEq%4&Ih&?gzU?7kA?6-PEzk&e# zVy;QzQ(s2es`l_JZGwXX7C28vC0eGYYYr3K*|{V+k~=_m3Hpj8&~~APaVy;4x6leP z2b$S_X>83P0WkCS{_4!1^+_nSEH!1d3TX>Zj9vO;NU#(R(J|U#)U0Bt2~RVIwa@8&S+ER;TLS!SQHWRa`U35aE3ccTwPfq470>90Db_- zO)q>4`d&SonL`Gq0Vk$_y+{h9xW@wDl{mK&g%CBF?H&^`Sw$Xy#Ys$WpM^YNAn9Lp zHm-W?_afl5ov`Q_ulCg#QuE)Hr)5BucV*P9X>EkmScFxb@A0XUK#F0{Bp}LPcov&;Z`r#otr6p68uU!~wBYR+3=#{nzhl-LsiNJ!}m!j06xN2b7IA>Gd>wa^gd@E}wi?8(nbr znkWuNW;rqw6=D#^&++|7&L3c-SZ-bw?5QyAAlf0=`LP3jw%%ckl=?ltsar*jIM#`$ zZr~$jAdI-}km4nR{n*s6RbT+~8*Z?PDEXf_v;8XcQSX)n&hTkR8%Tc-jgIar&hR&f zX!(r`gykNqASLiQ?Tgh39v6pK-V<{tQ2QttJS;~NjgAm~J2+3+2tg27x}s9>8OxQ^*A3h%nt0xXn=;5=teZynaNk%)lqS)2WlN70M$Io~vi-keYl>f5F%x zt*G9xUp2CkYBO$NftxDpo4gP##3!7Ca)B6hu>tXiRCVw-Gi|sXQn3QMj4l6XF8^Nd zD$`|LB%;;*EYW0&*Ah0IvihV^9k9j_iJlkW3Dt-AQgXKl>50J*jC;<91D5R5Mn8`~ z*Rmc+FH(}lQ!XZ7~XoQc|#!B7x9+CA22TWKg7S*qRC3&!<8s10M+{P;X2V z3XHr(22omMcVLeke*>QX-S(Z`x}<$af;$U-UsUWaVN&U|Na(kFD1XD#+ zriLa65kOse*YQGYxi>GFQ>8E%jm$-WWW3+nV`bd+q|oP3Sk4rHd62`81d5uFXFuQk z;+O;^MN6}>78ePX4)I$!xf&M^Q$*q9Fc5!k%L;>kWCq=Yo)B{%*rgbT(Sck{#;{_H znkI)NroavO?_lh5;*(TVDVYZEVcOqOdb5*?4N$pGXYw7oIXtHz7Y;$Y)bL{1ytddc7Bpols3QT6nTz zB2Dr|1>^OCEXJoH*b}f$y(1rZaa-M;dz*CX(R$EaH<*djWV|nXj9*-n12AFN3S(a5 z5!1kI>XyUIsQmU@4GxKp>B;ht#3}rhuAL9EEbT6r1}Cd4%084bIf?t~N}YU1B?+`G zEADgY`S;l2slnjE-#kaFRZ=&jBGr{7Eyef(Tg%wFfkLlMo*#S>hLt;>F+Xd8r|rbS*>(y`$9MoS<+Vx1YFV0 z#<3Fqy%zvmiCY~@Wbcd_Xvv9=H#=0+v4? zxuh)HQ(ZAd)*IbJ0kE$0WjUz+HK-^3lPSFf(E@S@r40z4fdPMV5svamabs|?Om4Nf zMLXh10mfvFVTmF1&UuCQQ>w_#pL0zlkUj(HG`>fAm`nhJ{QlHcinzw{foT8I8P_Is zY)^4Ckhw1X81zlsS6U2@M){t~1tP<@WHCu39}b2**JQ0ZQ;GnuNxrwDnVi2g26ILGeUPYLf^u(%Z3jY~2gdLa zm9ylSI^DdTk6hm3?FOElfV@QaaTHp3$Y`W;miDxP;B{IqF?>m{xJa1bku&`;mJT+K zNacZLCV+}867`P6mb|#3o_YGGLfoIJf0T>WqwvM62$GbkrT1sH*&3Ayawk4>6?2Dg zT_Oz#os@wq%UEm)ylZ_2c{akMG!2FS-Xb_1RZMa-hpqZI+-N&0N} z2#wE*enp?zkFXeeS|?}Jn?&nj!d_Q16{>fRmv>`Kj?6fJ9w_p3TK&}4+HW*=nMd~L zC=L+odel<-nZXJZwiqh0L1@u63;#?WG5m{Fzm|E?PbN}xenXOiim7EalH;>ju;(ql z?}wC*0q_nnAvv4*z`jub!OvmLO&99SaUHht8|0V(Tk+B*5?S3-O1tey*LF>fNwutJ z5Cf&R$qh83a`DQlOlAGiKZ|@EB>*wn8}*M{bOKyGgieTGofkxZT{R6d?(=DNt6@r8 zIgf5^OkhM<(gKx4t%KuLv6W@f)W`1((LxLig+Fi)%UrOaiJLS3mxm1inCXA60n?e3 zs(Aq~2NGm~ZV&?Vp80^H+g43{+BMAbJ@xCzN3H-z)yIb{C$w2r7`$VG(PJO@6K8@! z^hl&wF`#l^|2S8`O2=8TCY4D|smFx^^N7*(X=7I9B{a$&8X!~w3JGb+L+J~smg*su zD3o`(w@OS~IeZvZh<0vrU{P*IxsK`>>JZ?fcKx~cgD)_6H(hCC4T8-Y0IsQ=?MPj`N=W#4u1jnQ$ z272O=%2_V#i?E(h^4+qCzEM_bCI_ zHIkS-pg6CLW@IZ_$#wKtJ#Wy;!jBa=^Iy^7(Bur}q8>@Zf(GX&c-F zm^I_S!}{L(cq3Ldtt|2aDf(FZea@jC20`jl zzKdNTt60{#7^7nH`0O>8dkI&{ZknKHJ)f+Le*F1BLdyO3$BUmokO~hK`ZCGHqlnP-wH(FLkKoPytr$@AS!$*N5P=>y72hq;TJR#pnx2=kK0Nh z`llP+rw#r;^>7laGhbR^JPECpRkM;}{YLB-rI^d(PLomYT1a&25aP3aNYQP|ehoj= zp$O1aRJ}G2oYmD3R)r z-qAQHMi^a|owQp3d`s49R&br1zDA%RVjKr6=01L z&C;>xV`OQEBCAS#IM%wyi0U{{Zk9EhQIv901u1BSC5@6XAhfD%VMpBHX0VE zWdq;OTAWCWrONYd8as=*hfWQ!)BTGN*d@Ac)HT=0!YCe6&6vleavNP zMpEVz0*=Rs!E;n(Pg*!7@}a(lU)X$aM35J2V~b%6mRre4V7yG}M>@0pT}1D*maKf& zDtdVkh-$sn(02Q}LD)#dIhP*kY~mVMenn%EW+IZH89$|Ed-25&lDwUhpUDk7#T*eX z-W}vAq%*PCOjiA!U4qw2Ral4l=gA4@p=lzbnB;T#apw%slDUroUWV!q-O)*VxYVvL z3Kf?A=rGIyr+|2ei&wWU7#UunMZ!`$L@->t5&rv zD?4b_mmg1>r^n%zqCmN?JC%}n_rdhV6QL1fM{rxaY&2;dG-x0d9<$`nbfDw4UcP>~Z8foUkRglCim4el|g1=Tkb!e$9 zq)SfPRc?;??OPg=kV;e3`Fa`IqisMa0YD2vBlDD55go(UU3w0x7eQv6tVw>I4CMU) z5hk1;V-rNhXkHf-O3VK$T#$DSl%Ucc7Cy$g3+U|_G#yNZwTylOX4RRsLn3wG=`>wp z3H>`hLnOqI9;TGedbnFt$w>%;ZZMw0&~w%aR{5;MW|R)kV}V%&8kGr1Z#aYR=5xOb z68=fu1Y&#(dcGXAKV$ON(GYbUxZK6R>@CcGseu8rTQ}1Y&>h*%%7D%vE0C@$#O1ER z!J^i-d*47dA%~)wjoyW~e@CWL>?wc87mgfe)9dWU03yCD?68))?Vt1%Q_`~8pzidH z8|TptD@}wgMjgAXDWMmT0qcp##YZ)hazhf)bzks;&O+(GZkz9n6gVn)1$_Y>LeK5+ zciL%Su%TQHTYq=gU1R?%yLq(Sxl$!aa2ITdMAwW^l0`&(zc1hCQD;^(vWr^(SXyO{ zF{}I_0Fhdc0-UY73oGnCi!sma3b8D)jtrgiPa>=;H-HM~2!ZSS%Rlx`mwUi9UoGl2 zcPN3nDy@8Z!vhH=1+fX_xs${0=!!pUt=#(g2lPe|W$kKoZrjr7S#n%gBAP~pTwU!Ur7%Z6`%amW}t1j0@Xli*Nz12-P^nC z_zna^<7$o$hHG5?v2j`f`>(XHBV4#a**pLp6x!aHU9&R zWt570(nX%wh`g}V0m|@uc&!dZY>r}LYCOTC5$KT*)vK%Gh#&?f_{a?SX6}{t3&qo> z<@-Kx2K zhmspBRu9I1tKpAP*_W7~?U0wd(Xc-F@vx09k2xb*H{?Q!lkRK{UDDCXXr20GY>oaC ziAm&IOt$VyF`Ql*Q)xmgIF3q7N~0eIWz=NSJvUHxl=U>S*#vJJ^9nGs(4&)Vj=FZi&ISfTvaZOIfyze1F4c5kp)?c_224)I(fy zq28j&_1WQ}zs2#^n*gnnw% zcGym~!+9EpcxTgdrozsbH%nCeS5|aKu$t;pQf?rW6GG@|S(wsJ5xcJtxeCFPGsa{wA zl1PSG0@W48;qVD=vVF3IB&m3Z6PR#~ea6?&8?w=43LOTS=x;M_KWv=WvOu3TMz`%t z8%h(s;oA6>iUTGijVj!wh7)I8WkWo_8U+AP+~4}6If`eW$mBWq_rX1Tt#_zE_F?Ol zg!KP`k90qnf2QxcEr$uEd${(-W-)2(;@3T^g@3) zaP;mI#A{Arq&(2|zuOTFP*;B4tVfilqEOF(_p5+O>FE`>>AHK%P)gP7y+DNBpj21S zh(t_mpN37el@i-{)r0+I@zeyx>Pe2TT6Ss`j6sx^W_$3nTl4Wx(VI#lGkf{<&{81j zR@aTEickLSWghdu4OP~f7wT;*v{p9;$o^yWkV=f>-R1Za7wr%F<^;r-JlZh1~pGx zY8T1R=e`V{|ZV5ae{I3J$B;Quj;w^R(2ctqx)j*x;j}Lw?OQOTA(i1 zFCzd~K}7#wH_ivX=4{@(_eljPjI$?;PW$Rr zmc+aR!wji4HCJ=~p484E2xFog6b(%a3NB?&HQoF(TLA7T<(H)^`VGQI6zUOwud=xa z3{&4yu)l;9(}B7BF<~4{mVE9ya}(Sf0Ja~om@v151!|PQk(9;e{qOygKjdX-Mj*lS z6?S8Qe*f-#9B$9D8IC(pGXj^v#V|ti<9jjk&JS%6Hu3ll+v?o4g?6eR(Z6#die*k& zq{}hllD4g^tIN)^W9+T@$L=rr)1=loZO^D9QG=ZvgDLewp%%T6$d-ytwL;8EbD3cy zW37A8ubT@4Cu`ku9S_S;9z5`&htj}Yu6hk`z?u`E(%ENc&iPC{nIOD#z=PutCr}= zM+f1(r()a-{vOa3U@7ZsE-DkgOVM=eLR?8hHS0iEJZu@k>w=LiYnWCo`=cei>z zp=~YgGozNAf?I}G6Xr&IbucaLu?sUI+2Bqx#RTjt?xn`mB)o#|nd-BSlG&v+tlpYi z{=ac8=!u^A|GWCT+>N2E1zrAw!Fus@tDDt?Y~}4Yoz8w!`P*KJ!h)ue9o1!jXbQD; zE*K`{Yp{EO%Fqy|V)iJHhSUY0*al6Y-!qWt#9<;LJfbaUrR++VBJz#@BE23X2^5fF z4!~40M1~hLu^m$7)kk%zTENciV*qR?LXH<49q4O}wfGbNZ>f3xTTKq$cBy^?$NrY08vv9q4{fyl@bwdpS( zW8w*>JyA~MA&WekeAVGc&$7m0G*IphBetx|Y!0K_zGHMvaNqDZ7i`XUu{?h$=^S5R z4Upy|qM(h7wqR5~L)_-caBqO-kIKqzKHdG0_ydpUOpp@)9^@`nPpy9+*K{Lg7OJLw zx-TK+5ZI@zaiF1WNT*oy2gl<-Or$j6LK5lfi!`JCVNt5mlSG=bMh5k;0X|jciibP! zh|`-Kc*B*{}$9c2JCRu%`q_jQ!C1?G99O&g-T zIf)Q#N-7G)6n(;;XNZ%mN`tb3NL~etP~G~gX@X@P1((An2gC1e%H8_a@J39K8hrWVH`urW+HiWO;|}{AC!|9z1l05k zv0VkUj}Y@Ht$M4HcGPjchD+yr1hy~SQPj>n9??=l8UspM=Y_8ZV_T6q9|^SBGvC5|XwXSKa6!W#RLaQSd=|3~5U5@GNLi1zy2T4`ViJlUAN zzfRLhAI$Sm8hot!>ziwyR6(^FkkyzexxmqN#y8e)Tvnbw%ED_M0}3-nj?A74I|F?I zp|fqG5oC!SicC*=s$2IbO5NI1l6>bR3*+_2nm0#t4}VX%-k}4}kZ&1Gpi1UEdu7MT z{PEF<-5GclE#txFgr&eeuLI zZ2~t$jn6*3uL7>p9A;aQ%l|>#_jC=@r5ovvVc4Lx;~7DGW@dS=RaSN)Ctxm~C% zB+a)}$g&x$UU&NmgO8e!u1Fe`+dA*jxq+?b;URBLnbuMgf+^Rdo=U(It~T3$8?x9) znJet~VU$nkZ{`lJI0se|f~5Ig`H-}oKP+`@S^52l+t-Sq*Ab%cn+MofvWdf(bderv z^L7;2ct!E40-=mH(31Z<6R`Xus{BL413!jpw^@o!S$@{c6#s zXB-?dL}NoO-&}DTA_xd?2&Yb5!^i7zZKhbt2?)p}CJ( zt=+NO1j;h=x5jTmd)?fBX+|aASH}-nBpuuRW^}Vp_6@@CGt{%0S1k~D7TH_mSfsR_ z_!+z%rDd`~t9VphO5S|WDfGfRc@=o!tsoA_HG1665xACK!MObq>s?>oOZR8 z3uWK38EW#fP@bDU-XUa2zGd+^Ljcg1qhLx8cCq!{#3@9$_e zH?SM@7qQ>mVqU>eWqd^+yuclo=2)U79e8d9ff)U}Y~GdX;pNp;^)>A@y&Z@Da;5E(c&mB;vz##}T&)36a-BmrG_9 z2{n4(HpsJlKrte2@|_Du8syhTSQgInd=id*_rFGOlv+8kf;z`pi#H$j7VsV2Z+4p$ zk|bn9I<~0_cSbJQ32!%~yS*w=+a?%DI=l_M7LXSlNxKhwhEGgtQ@Ahun%L9qV<(7R zaEKfKlx3A$6S-b<5JDc1_t)dfFF9}I-L{kmo6eUbO2JQ#qhlZhx96OWT6jDKJ$2YH zt(SvXHVbeqPQ~yRClOEc41qY|I_5|-aH)QY!zHrSPoMx3iFm|1H|@=Fz|}_SrXNT_ z%f>)8U)a>BYG@9ulyyZ)JSksy52`UWj9j2n(197zS<7L8nb}PGM_SNSpABAJr6`*J zL1i&{s}kJLkyjiWArv7>8yRC*TIk!}^a@#(o7sZy{qJ zM?bWA$?Ww24Ez2op$h6BWvLF!2%Z_b8#GvNs3v9d)CXfWRD^A23_k7`V|2lW*Q&OO z>B6oBGW_yQpb<7Ar#p0UCaCNa%B~IkW^XX4*lRb@lubxTr+Qk4W8^l9}D2gkHh#C<}oZ}!dXkX z^;%Q9XpBjT*ai@>qDs-N+p;2Et4Aif4>7MNWi>;e+)_7&&fp+yGG(%32E*Qbqu=gI3iTv1N{alyQx?x5b7pRPxprYY3YMtXviW8W80f|N5Kl`O#gH z927m!Z+7YdZR0d*MPG1^>*BqgqXD|C+?vdD`id`EvG!Q2n3s=h2=-*QQL%C1idFXOi9?Xga%kw^ka_# zvDqS7VrIx)e}9|9VEdw^AU-mtu86+&b(Dqfe=NZ<-gn3MOg&pX(LZ3CK#Lb0wx6;C zuk&-7pA;YA1?Q_kRy3TqpmTdO{1?D*-(`D!j|4$SPTP(M0*8dK=Vs_n=K1okk(F%U z0^~aQ5D)DTlh)S#Lp$~|r}tfYow?H7$Y-H=(j7ZU8*L(gU8f7 zRVf(Ni(7OmrIVmntTbl1OszSOhe2JLtu>T*7@DW4Q=N0jD+DI51#1QqZlsojMIW(k zq#$9r#eTa(I1>z(iiJ~b^VCTGJ;5lv1e*N1EX6@+$I`vm}B zIr;2ks6lR+#yQfY2YJ9g0)<|0jmo|L*=DGeHAX)I%Hjyu1Iu8>jYiLE*P)_qe)3;O z5Q)ui+fDiEaoxvbJwEIscG2fpsgNysSCt__t-Jd;S2Z4J`+IJN zSyS_M1MPeGI4>|Fzjp4W;NhlLQ4j*SIp9!^QBHVa==Ah>9xc-jDuCW}Q8VG-)aP$T z&MRTLgmt^SM{YO05=W#C%9I_l&^!V+qdW4_FELbzViFf|ecA7;$Bf*`cCllbw z+oJa=?E~EEnt8Dq)>+(a^_IOtJh@n)cnl$5>tjwTp#YlQoOIjD4V~28v@Y+WsY#La z8;57Wt-bcZ;5o74^#9*sP#TDriyz_H3iz)3EM(N};5yBIMK?e~RNc;RKe$vt+TPpC z*o=4wdb07)?4SOm6OAePWXB|7;E+r?&h1*qeOa)Fw>5e9_q9$NyM4lCWhA$Uv82av zD~e<20SAvCa2Yy3K828Wc(mqoJVq1yBY{sna=9wOoiak2 z_PtLP;z8$Ls)EGN9^!}m5IJ>zxTkUm^C`R1rTnk^hU)^(rOZaER{D*w*&i6>RrvQT za*(hh=1r9V^mRr_v3t4>;qzb7zYkWd#~q+-nMcb2IDVkW;<(Ud@k$v!n_kvM#T>ry z8XY0{J5?eF6#QO7b}r@5|Fd-;-`hsWhLH$Q#eYj_4ABYHec2Z773&-Z5vow8^0eMz zP)Wp#BBVF&?F#eDghYuH2J=tfK}ooEosXcNzSmmFC-PQ7J8j97ad-mdm8N0rt{W~I z+*l12Zbl_Ar1brs#>0kcumztgUfQHF5eafoes!H-*gOS`%=I~5^(-S93Ba&GbLPrb zl_#lP^}q*XM$aj9GO43S>!8afRB_HAolL@29f(-~+;es4P-m zIiU}8^<6fNPlHtY5fBL*qyd>L?Dg2rn<5od-Q8ei?Hc$YR-;aG;NZwKN@ zy)W>{zjk+=y0QGY^q4>JpR)q>*-T%IC7PF zy!*-%w4txcaP~{pMse4&uQ2kb)ne3V$Z3gB9N)AhLfgfjn@Ew#z4JQWgA60r8O9k= z;$IAzodLJ3oCj&Dp;7G>%8-55ZpAiXuPqfAaEhL=Z~q+KgSU(7Y*@-nmhpS|bT|)9 zT?{|8=E{;1*3MO0NYL*#%6t2U{kdEPlCE4wWJ4(s&2R^X8vJf)Y?=JNW;xf1$4pz- zH&eoyJ?dewG)W^>4;Id0*SK87)#PdU(GrJ$bYgh{U165C538jGPe^4N%a-5i5A#8$ zTaDKTTe8b9~)*ZTsgTwYkxg z-s0vv-!W8P2}l9BK%fM|FvmYi&7-XtUq~;!6)-NA-)1*vdh?-eh?5kMoe4bk{C56} zoCE_)rPAM?7dhD*@GkINQ+&H~0~ zVuQf8T4=FymVlQf(qzB_9WehPpx~EO@5w&^-4WTg^d3RJJo#oSESqz}&BO>fO#XHd zMlz12e>HuzP6>dZ)w0W+>)m_tS8aOocl1YWlEB-|mkv4!@o9dT6JVSk_{x$h;WB^L z^ShIU534&)5QAz|5v6r6^V+s z>fd)jncrWO>rH%h_*uJ48s5>z76#~xwB8h%kRRGAs*8G-hR;Pe{d(KN_5m-P!}a}S z$NP5Qr#hoyI_3JgUlr>w$pZO1nEufv+)O-BA?%(mcRZsuOtrwyQ&}*1x5^8Z zcmD`JIHgaaf3A7LPHR)Tt;MV|M)RXmG03x&pINLj5I_j1kUqZJ(X+a&r|Ez))H)Iw zBiHg*k~u9AJ9kIy%R1>se6T7*Q1pwi0^SsfzUQ+*oX33Ed>=h#tGNAaL12|AU~HNg z?ItC1D+scy>^|iqgOVpXqVP#4z!P%Gu1N z&r@!L<{EEGT7e+v`l(uRtV2EiKq~zzkCkDO?;x%Zv&2*3&5*eC=VT?d+ZcV&=!!%OwYf~w^JP&d}~hqWIL~Udv(t2@0qh}nLXKO z&!n5MH$lVP@c%x$YG(d}SlY9epE>UBYi*RMGM~P#{xmBQu=UzTrTrvb04U^_mJWJW#LhyD+RuKNujk5TXBut!N7*W7q|y!%LR$ac8HDRItanqrlV?q#FgnN zN+Uz=W_9*sfuW85A7=Q<6g-0+ZwAuKUBfxu86hg^;(t&IRQx{k;Z#OUYcMyi8(u>RzXnipN--03fr6byTNnZdE= z*ZC(C7Btw>1)hw397x!J_=Cuhl=NAO^W75;M**YOL!1O;rpynO7RG5$tXs)rzVH(DeNrwvXx;lDZzR}k7Iflhn zs|A)umO~9P9ZF!rabKOCY5?W9*?#!u)R5=5$5+^LM9H*(P~qseQoSw@(QU(m*Ya+k z+~b5RY{3t@N0*#2B^D~GCvSGTjp2b}YVQD`;8f@{i09y#+2`-49e9{wKqKMBkp0(j zV{0y7zay=Y;0xM_!(3CTZ7QfP<<|kXEEWR-<4ew5OToo&d)0xKqV8#hOY=UH9Cg0Y>jq5Vwv_ z?G8O$0*9!FU4=`Y!GfeOAAlD$H;!#|&N?9T@`J}MTk^p$C~n5Q*X^Si`?;BD9m00u zixiz$_bMBImfP~(dv5f+SF@uAbR4cHEu~b3c}Qi=e(jNbN^)!NyiEe#v1W9SyA0h* zx#P<=sfyZ2`6|y)+rd)AtR#i(l4-rmZ8$CCfrmD$LHAhHE;SWPS%U*`=wdF>45hE3 zU^$3D6bIdi21(pFqPvWa7Q9v*Z#fUT{^D;FnQhFy>}6YDr6x=Jpo3;#4`UMpBiyX- z`CcCl5k%;0{@ZI>$<=vk>g1ZCfCr!@JzKK>y_B7Zf_ld-AM_))NXS|H`XhiyklNYk z#I;k426m^2VabwN+;8K$aKjnV6vRriit-Ssfhz7bhSH4QJ*+52Ij=50!R*xpmj8cV z+b}8)q@k?42O9B7D7(+>639{b6gcB!sm+sE;<)jbifXQTx-+4l0tq9ud}`&~Gm$yK ztiR*IFgfoV#Bcf{{5T6*iZIE}#C3zmURQ;0Jv@TN09}yqNF9ar3Uz>-_%#1#QoMqR z;X;AZ75lnD9&4FNF@9T^TL3g%MVEJHpfVyYIzuD5Q1%mV`Ajp;ZM-G=bf+?A_ohkE z_GwfLPoYDB6z@&Kk1Qyjjrf-8T5XdHTnTjl6$A6Md1$T4 z_|<{6HRKq$HT-qVhLkv;{$*qPG5&m?`d$OY?>Vh9U@x2fVb`I0E&b}#90_il%i=W; zp%xWAy0zD_ndlsr9a3xg96!yTbhJkpG1G@i>yYDK1Jo83$$P3mZDR4TcW5>;ezG)4`|!kAta-+D9FwkUf`JX zE^sg;u&hWedps;4)r%(^K@-+uEu_O{FtDgvdKnrmYVDrXZ9qacYp@kc$%u7udGU^y2YA8OX zqv*T2Ife^CYg7>(k%=(Q!o`CQyuVkx+NAc!))$8w_efi!@!SYpyKb|u#$R7^Pt*CFyS<-}wI$e> zJo<+OderAsF1g$Adpe7;J;E4*hQ6f*wCdbG*t$sN_g3>()x{3qhnhfpnPsB01O`|1 zlc6<))6dXg0vj~A3D50dqy=$pHMqiMAYFawF294FByJVNbAhU4<~O2}4l!e~vp`D; z>$bjVlNyWgTZ0}D#h&6t3mrRI{Pvq$K6OZ6rAyCc>*!q8(7cBZVWi+X_73jA!wLGSQ24Iq{$v~H<>TIIhyAfdb zaK^x)G><&|<4&*hog}X2VVlMNgCsD=$%k%wy;s?yuBCmgNT=utKrmaG70VQVgW$B6dl@=16_D4P$yus zIGYf2V`UyF{_{8RLDV&m5#^(GI(xEbXQ|3_@sia%h+~`OX$i1N0mOa2I%k7hf6Wt! zXQ0*W&Iv5Z*q*8U20*n-hjTG&D+Sxu@rfhKss8^i4lrBeH=e(CTevGe z0uf8oGLfXSO(L|KME(@qu4=)F5Kk%<#wmnxLEgq^BXsTiBQf?ES7InMLHIN7ce|e> zblVC7C;h%P$(hNoQlJOfJ9z-Q@z0~H28AU^mrB~a9_&X)pgk9)Zwq!|yu4}fn%4P% zYy$NNS3~@HKuQ8;2kR)%Y_m7#tna+@_FG+#Do=B~Z}Jm4iduQ8+^A8W=m^PU)>nd< z7UB5dR3FR$AXmuf6X$&}J$=@S@8T>dQ#r)XpP+mmvAqtN{5n@%IZ=lzE7MJ@5;jvD zVc9_~ z9UC(*L7TjmxL^@v2oJl!#a*~Z{kwwk!5P<#Jx`$qoxmiL2Z?ns#3g*7DwjY&VeN}R zaWFPKZ~iT zShwS!Ni6_ipqe0TC(F_R2hm;a59lDoV!Zfo1v#L-z80c=`Qlj?$8Gn`xn-$JFl%MS zXvL+k8LCym0>bGEoThD4CT-JtW3nx+A1xbb<2-^tb`Yd47$4HE|;cH z_<}#}==c5H5~si6YOCZ@`Y~-dvhEY#QE5zA0BZ%8Wz*6^1gD?WG1}!-0{S|QJO}fM z;|bmhAmd9z{w3i26&ao1r?1w@zOebkSsNY20%tPR19mtnG#!4pGEgWzvQyOB_+3X* zI*Qt&<9erc<)KzA=rW*^)PR~ogd0heU*XFe1v#1CJ_g#VE2_0QKYghEVn80c&AM$X z0AdiagL;gf|HuNzT%VSU%uhw|inkR*D^sRdJQaE6O~)qe*}^EYj+xv;m7xTK85`R# z?F=VAQ-e5ycwUyo!IB<*ob^04Pr81&-#VWq z7Rz9ICUOU2RH1`yC92(oKmWk>U(;p``DNR5WIe7R0T~!ro;(y~t2_y4>(=<&RN&Bu zHr%?Ycs^e^mGo{9VaI>OZXkagQUQKSS-q!)q+p9qZbqAPda%)GolMvkmSoB9?K_DYy9d;DFQh}MrJ()^I~)EhF`8R$ zZd$k+jPyD;iBOB2YqG5r|zCF}qvSJsm4Clvwz zB(<*6be^ozUkjZM_b+E!-xyO7)_sIvZPW}_M=f5W^EHq2W6m(;rOan$@RrQh3{r0d zX2*iQ;CF)Xli5E+gDigXsaE@@N%qnyMgqTgOqVr!;POqfYe9e~m+Hu5o^s$uvDN-j z;})B*Zk|4S;SaT$yu4Lq6Ei_2OE5%jI9OW@k=T7%e>)JeD#8zyVN0?8Ghtc5*8__h zH-@17b81@!hC5H!0ltqcC{@7u9a(SBJ=)v?-lASay56%`l@Bl+@6iqilQdK7aJq+h zBBxIqab+@?Ad?l*fg>K;e(7ZPJ?Ha{P=TKJ{IiP1#Vc+n-r3wdeo$ffhRM=~WF(gq zayOY*!^1HAOcw$0=fQ*Tr%$o5_grZny`;EN2K`|tR@k~-mg;&Y+t45zX6&6cuKh(r zHgd%bQVsCmU-+M8U%NztW`W*|mh|p1w*O*8OR@$kuyWT?|H<@u~k6RB~I_sGxIsc-z7OXhxaLpTzWKe)B8(GXsetLnw|oXQ*N1igRn7JntfL48XqS zZ7f0ozo`g>rHZa;{@SFu$Xnc}(x&%}4F3o8)=4q-SyF0;DzXOI(XNp30DY`zh-n3- z`43$A#Dc8(R$mu+{QOobSNZQH)Cs+B6Xvbgw_NEX-9FhC)!L_7FVdx>WF^~?r^i=Q z7tTZU0b*8M%qjL-x5`I=t?D~JA*t9@4fH9n1btWx#go@E3I#dJ?2`X}t2^Wb22EGY zWR{dxpPvzm?N@|5U6(nmc>N}JLYm*gD2oTT*1=jsv%@g!fKj13>^gjKqZ-2jFmSYr z!;(3%$a~_q2QOa)C$%+eG({V*TT*vS0aU z`IC&v37p1!_td@nBH)UsQ)ud~>z7|9DYi?_dy(E9B95@~^DZWx+_1l`@-}q7)tPWp zhdq-$J@D4m#Tg1JHI}X700JpGRJTGkvn$OCnTfu=)qa;J?(phiq!VJ-aIk&gxPSAxc4*8p2 z)CmFSZH>SqN!PBg`!85b45ls#nzCEFK*nH$;2>2olsp-u*q8wXByT76q0I(8G4!a( z4TsP9cBuM47DB{NXKpX?S1+&z5h$w2t?$&6;Bs8B@bAr7&*X|&Z<0bAR#%}-YtsdZ zg%WN_BQFIe5y<~aEL`I?sbVyJ`**KmG)mT3z~J#p!EI}GW(Mi2%#C=DDtf4NZzW4< zJ-9nJbi`<4f?BN!RDoaD93p9Ss7<`*l^r*P4g}APO)nZh zw;i?A6XxTzg*WSO6I4bGRWFNcsi(T|%gf{PM(R@99Gk494J{NMHIu{cS`hSM;qEeZ z+8Fn?|2}~cSWOs}RB~o@1n#O8J801+I8e2%RTW2{7m?|?l;6?J%cfn{NA}ems{hxX zU5GjqlV)7G07j*WiV&rQUx&o*op#!6x|r5fw@-D!*t9m65^c9Uktsf8o+yJN47R9N zy9%ob{_^&s#UDcS(O-wCPUM)>!>TB?(v9-7Z~TV zDWTpS^Jgbtl5*%N)UYRe3sqqlkae%Q*l#GB$Ncj*$m1PDTFBSXPtE=P=v_}zFtCcj zZMvLFt~5X#&M0H;`lKYi62W8L4AD4e^nP6dwn06;5Llupn6QvD^K!g935fT;N`KqR z$^|sx;H)IW+Tx_>$$4FtTFC920t*~e$@zeWrvpyVuH56!E-itB{wIT+dQVfJB140; zaK$1Iu2B=!lMgvo!~4JOvH*A)Tn+tMiLb{o!+=00zbO4%f&$g4L&qbCThXvgG{%a zSOvO|)tJ*_vPb@ZgoHm@84yU=ecRL59}|aq2ek;MlEJx(moB=3Lt-oMV-TpfSgFnkb$s+F(i{!ta z@9LIq)T(;iZ7bT~+ZB8hE6w)OL$#y~%&@8+cXnG>IH(qCnYQ*T?+$ad)}xr zHS)hZ>bn*y;>3dFy05vbK!00rm3ynKhpRWkXmEV}GCj%mV00ePJ!m%AsuVRz0}iig zm5o{{rH!5ln&M1A44%#LLY$^X3>IJgP<^oY3^=N`OP7Z?LVc;?{}$idgDt}QB^TZK-@YEU3CcXa2=WS?){TxhJS-ZkREqg>4 z)S-T@r1Efwasyw2BQlJK7uWOwQn~s57GK?rJgEtJswj?uYk5`8s4|e zo}nyQ!5HJ-Q$>trZbXuY#`+g_is^x3CyR+*-Ewbe*e1$6-PF(Q6n7OJT|NqO{HxA!7C zlvcBsm+=H3b5aq~u*dXwoF7~gR(R>&?AQXme|9kK=<}KlKx#Q%^68OC$#eU~M5!k1 zcocpkvGp_7jLtg(tRTdt`+gN)yrpW>^>I~h6O5b@+N?z6Qvj(OhuIc|Zye4EQqFPi z1i$6kzdljFc1Sm`&3tL}<@M&~t3tg95-MeMHRC9u6-rl+Ruro&6ddBBm)j#MFgI)R z{)W@f&(V3}wqMi)PY^$JjqWB=qPy{m*!=Z zTJZDFaO668?&0Y7CDRTW_m!H6D{K1)GPbzl{D*ApqQj+#%|P2ub=ZRpCnRU%vCnX!=z@D2 znR`88Cuj8`V{Zt^_x$$njenfNNb`<1TJG7w(v3A;PZK z|4@ds`ep2lpkX4`&hQ>I;I+#&$8-9d0Ax3Qh~9n;ZyJmJ0C=3430Lc(ZAeBsj7?}_ z83+ywro$k~SqsT31YmZos`PdJRwDZ^OVTT#r3J!IiS+lLHY+<;0?ezPJYWnla2joei%ClSyeuV}s6 zY+>l#uz5l?y_(dGXL-%K1p|MrHeUZnS$w@DIf>iFL%#YkN=txCU>)E=9VMs)5rTBX zuL!lsdN*#iIAk+qTg-x@Y{dA8i>1|QEkjj#+WngXUs!@oGXa19(;xrjNB^%vKcLYi zF+YL9t86jE9{Kn!3plBS)#n-Lp17fnoT4Ilw6oTL17~li)>GF8A=5qh;yCdgs~N+@ zo304X&C%MCB4|amUx&lVcn0yckYm6#$KtVOGZQU#IJKXMZE$asO<#+hlvs{g+tJ6< zNtFSqs<@~J9E&x5FLqk%j44d5>+5kJZF3`?*77%m{Wc_o^gg2syy)^drH7u zg+`9AH~v&~4G!056?SvvT32NsnE>srwt;uEp@l-wn7< ze@%iX>tpy(Yi$H*fmyh;syK{j+9cMgEmj@J9Vwe$>kBn+MYYiSxBikhH7<)E%)(XZ|s5zhcY0gO|fooICG+8okgyVn90kq9smmCubW|Nn%oBgZh5d#*aTjxS+{ zT16eaRg&C>I|!1d@L^$caVCp1$lK!slTkO+h}U0A_jw=*veOWeN8fQFlNg(mhU??g zvj_1**6;m{r9{J}RD4UewIt>64i?D;OE#{Z>^>6%yCuNbylTux9O&+Ufz2ySy)PAw z&%2(VZ9d(JIZh=m{(M<7TvHV9U-y5`SqmMcJ?OVJ>pCmTVkN&{rKLmqL;nIwD;gsw zIK_ELR2${W=Uev7-q!pUT|=56*r#$DmTbo*r9!{Y>EEt9wAf_Y(ensRt_KhS*zu}k z@P9_-m_`gD^9UGk=k~J|Q)YHnRjqY$f{swHJ{|!RV^h;FdoE&!6vjf6PiZW)wZc(3 z8vgVlqUwodDNsa)V6r4|6JV$vrO zKYt_Cy~V{JslUU#p;1r6piXvvK2*FP4(X(6TMs)YF@99s_lK|>FATN~m_-iO-UP%V zw?+*tVhxT)&qA|r44E^u-jEoBgRGNsu!5bUI%9z7sA+tmm~*$Q^_q~`X_(=pKt2?a zuWSn9U!$1QrVJ$``}OcIza@-pB{phS^I-$Chc7TXl6Tj?LfnSfWfbqUSp%$Ea=FIV zl>qs1Pck6NPQe%t9bE}!`B+vk1Z1T@+LW`O7C!sMz(@QmS~dPOKsBP4n;{sg-e_Y* zf#{`lKVtj8rS8jC-F>Ou!(i2kQ`_Dz3Xv!8j`?)6kuGwjU;i{_U6~4tM4Eb2S6$RR~=Rqdmc;77^)EVIx5{i!b;T zlHT(^I75iTM@C8vARLZB0W3pKb1@}OI{kUKX7Sl}CWYN*DE27@93#s*TuJu`cBLJ# zt0Czt{OZ&o)XukK&!`*$77JS+hx**?f>w)@G|L&|qGp_Jbj~g4w@$vD_lc$Av+Tep zN!c5#`#%bO1y940CwcpHMovr*#R>~(gi+&9hmfOgD2xD{ri8(tTBIjgx(k;BZC`XO zse0*=Aw%BV*{?#@ScD_Qj?Gj^af^T7uFt-D)i<+ji#pUe9-jf%prprWatpO7&q!E| z$aYhM$)2_fIjRLCv;4jon&CQX)rv$s$a5@T_63s<(jykc5^f^vDel4_F%8HnB%|BO z+;AS+OCedZp(>{|C*67B9cYCRCx=+9HuE6y9Pfs)*c+4t>VC;IYObqTI zoM(kvaX{J;XkjK;=i!~G9Rb{T>d5c5#ITI~wgj)Sm*k}l&d6_G{xx!CU8n6R-(clO zrr#kw4v6pb)Bl!l?zFjm0aFU%zj^xs7lr@Y^HBMgo?ntJ2!S2SGU4S|6+p*(3C=Rv zNYfQEavm$Xqc5X)kVOZSNfs+I&p{;<1<`6U412q8=zZGvO7+J>Yu|goMt4%xtj(ZD z88$f+7}_Mei*X^k(3SIUSV+E)mP%QgJg zUN~)OHeiIXfhF$>m>X0#EF6kyg=8qzd(!2OfbngBusDgLq+-nA-~jAbyMi3S0Rt8> z-64s&K?0lDBlqQTpG*85BDo>^c|4RK)^|vTVNby^b8C}YQw-G0DA1ESO+aEuy$4&h zCW@n2YLmdnP&uvpeyU>tqxL9Fp9Oh=ouuD*aySI7NdDeNcRW2npWQQOtsXo+qbqlw>9n!z_jo|==inqwI5)BRsfk{u zNnfh}(w-YkhU+ltXbamfLfcF}gKb|XAX#xr-J3CUvQ* z;O9Y21(AbmjBu>pDUS_F7E54k5wTgh75hz(x(H|x|KPB|-}1sKgXJqQ7>v7b!!RSh zk?a|*shL!Wfnc3S)5OMzBTunl!EPnG^gZy7ajCuoAo{DE6d>SUGp4x*=Nyfpz#+@& zkt=BDMd21(U!L}Bw}lo36!yOrb+@)dwT4S$E&QNP$zKa7B!5lTBi&Em3Vb_n3;X6L z5ieM|u&xcwQJvJEhUfrt+Hq6x-W%!KnlGF`l0E+b@yb(n1YX4h;V=udTTm9(IhmBB zSZYk>&b~MkpZ|v)8s);mJ$!khBsBxPj(2*CD>BMa>1foo4h*nXiC`V-fpC0oULjN>T%&yKUX9olf|%zafaz(Yk_>@p-om= zb)J}mDyoE}hG<4#8AP^yrn=|x=0?cJW2Ghrn(K;82Orysbq?f5EvRAl5<(TKYk4wd zdW`3DsJNE4Q?5oOCJhBi$Ki5>(xEdOOoW<&fD%N zq40-4_Y#WLCc0BW>6Ub8?yxxVrp=oS&?4*r(sKgqpFQnrEUf!zFekoV z*!GnF7;M{@dwSD`7WYn+R1ZOsly4FGuCzUqJV~gxrB?$qBYc_Smc~o_x$C$0MQ1B3e<}~l4*uKuG zSNF-I3s)?dWf(Q-Q*dv-*}s&!<6Jc4fy4gI|JYzAb%~^{%2&hj#bAgwG{?w&Uky1W z;Ync$EcS|sEapLwKeg82a+|z6H2SXH0q&Eb4_I>mm{Ma@Z&WkWBMp>Ub;K}n%1o11 zqKD!|?b`<_$ONbZhc99%)izZtwP{1aFJ^TDSAzzre5_UaBxy&?qR6zj5!vpRdQtSz zdoRTM=XNN0;Tm}Xa#(SSR7T_Nh+HEv`sAT}} zawrnn0vlmjV_Xgb=W!%-|4yNxdx5KkL}{Nl{x1DKumY%Us3~NVpV^4uW$gdK z*YF~t#Uih!QxU!ihp}44eYNB|C5D`q?JB@=Isst+>VdRI};7aQLlqZ7+{(=_C&vh?di}Yl0+Ih1D-LnIi&fq+bZFu$NJ?T5S5seD1 zyFunwHgps3Z8nv2SJsm3DD+byIQsEe^L4)Xr0GpezrYNB7`&Kd_c7xo)n$QP@00V) z&lCn@8Gi{%2qKDh$hBk86z*Z>yoXy;j>y|~43H|rzs7aY>|t$X6{a`aKiLOQEBJJ^ zXcQTrv%t%^^(eCIpdRyGH%0;%|B81{NEu^!KuM-R8$iu(BuoOacjgp#>cR;E9t#-Jh%;m~fuC2AZBFqA;yZlAV(-*y+@rF}UN8Wu!Oo z&w@W!Qs)Y)zB|0CVidXTC6)|LYN$)|;M_RPRO8ICp$jg`?os>f8lNZ~Z{V|P*H#O${r~ zw{ewE#bXDoPCeoSp@*^y;_ruM4+FZv8czZg9-M^%miMpCe zSY~r1u@5j|7fQ5DDIh|ka#q0|YrJ!n%bM^efD<_!tk9VpFxU{pkck^W4^l=Qmf3`^ zC>R34ALB2Z@4V>s!=_N3uuWdO>pw7ph)DpiH!{hoX%by=#|B)M*5fS#eo-w>)fXzA z>(#2ik8I%*$A^$oMPxg2dMz8M-;1v*CS}~$#UHg4CBpUxO{m8Qq|WYg`^#KF7gG^j zz$yxK&&M`Erc+F5*Pr$c{-E@V8y!W_+^RSNUy)#I*Kh0EC1V1)F&Yb z&#dqpkL5E?l#Utk2z3+V2@K_14uHRFxPfVi=sHxoT4{KeK8g<@>%j>w`Rj!dYsz{+ zi#g?-vkMQ}nCv&qmTrmWA!=sJIZ5iooS?z`Y?EC#NAs88G`WC!BK;@Oaip~JQ`aw$ zM2HL&e)_D31;-VSKYdaM;?_Hz)OG@$qrq=X<)Ma;0TYRYjgFFEwSE%s>zdiiUN==V zjK)<@7fx#?su7B$oSCDTjs3ASyQRh6L@w3|gzH7X1NJR(*;h3_`p@5Q2`vUAnJIH+ zVb*-FF=2O}8&E)V~kXY?+C;%X5J`aUi)tj zTW!UAMivH#Z%Tc_B9`-h^g-jDG8td!7~pOfL+!TBfQ$?*is)Uh+>2%y&be6z_BTL;?8k2~VXY(O`r3sRTN2A>Lv8 z9cr&|4E24hRdm0|s}c>k=FDn2LAJku>@hN-o5X9131YNTwrrM?w2f>>5Y@=o>GW5g z>%5fneLg<6AdL5jy3Sl3BbZ~f#rc86x(oA&4w}Ve33{;)z(;hZIEGCenhbThm1F)$ z{t>oq@@*3+?VHILZ^y!7_Wtb7TSp{2sNeK$sWKo1P>_}oqt-oYte_vm#^+oHV)JVw{UONj zTQiZ0*RM)i!u2?Q=TTyLFvF2f!Es&TIB=D^FbCRF_;YB7@&yl3Mu5M2G(8%BE)=u8 zs~hpXNt0iNG@Uz@V`G zCk{it!SD%xFmT*QxhOp>%sFw<>7chknY|Oxp*QKD;{ITU)|2_H%+)$VwBy8p3>PK& zMhGZIVh05UwD0;<9@dN8w9N^_EjFe`lxfS%Kg40i0I73kb@GAnyv?hOZoU6wvA&52 zm1nG~^yNVu#_r6B!od1A=UCi^I!YPt-J)S#-O%lJIo;j@-WYu@3rYq_LO)sr6ho2a zdhwcCy3Fr@%(ZM3(o2-Kh-VwM;D=p$+yUWPwdv^HGFNuGAfRxkliAIfY!QVizfg0b z(whpNmkjjmg_p&=yiu^m6n+^Y=N#7_2`ODfY(L%u|6*vJ$u}$n6$IFx|e9bR&}x zi^`-l&N0$PK|AIE=j-R$&A!CdM(Ng*#@#uFHtSD4Cq5>zq24*0w!DTC_>X~F{Sk)) zjLS!IsFdbf--Q9)-D+M3_Q0S#^Ccf^j@Vb5vRf>Y)%yuLW4d|BZAI+Rc`3R4sI8BY zr61vi^s8}?5>$chZ{E>M~Ml&oClJ-*%}dc4igGhqwMh1eP8;L}QGUgw4ntl39@$dW~X0;7uzg}Nzs zvpz3w$KkpZ$39GYosw{k{0)JakzZP`~-t1jDkj@_ODJq_p^2A;&j3`0E zQgOH5yAhkbZ{}9yr)}2rsDXY2W{XzGqY7k7U%c0b`^8Z4NBIDb{(#zv#??h+=8#Lz zCcBHndX!6Xz7{S3J*eyBFIh7vS|$ccPp2n5f%cAEX&rGdKU6YU!*9OB3QGdDn{N4s zfsqHX0hw#<{6Z5PRnUL?!B>02Bv6)+WwP$5b9|R9v$VgH?E zmXH)n?zv_+pT1{rf7X9B%(yN>*aY9N(IUor{LCs$WydW3|MV%DP&n;L>VwHk^GvxN zl#s~Kms`h%MeO(>Y+Dti?;LgVv`BC7K4yt9da_-*r>E4uDS2^>e0M&E-bv7B^1o+F zczVe!TFJa51J{|7IHdnQshkdHLm8<|2*ntQ$+}-g!`FZSLZ?-#5?_pr7bU3z-mZqe zHzL{`!}P;KuzIX1s&LemY^pnyP(z3z*<&NA)?s7$O(L5$)W>hP;z7)tj41#mfS~)S z8^U(amr=y0={*<-E|+1LLFpIRA2GBK(=P7b+{{gZGE;xIg2e0myp*PzMU~FX>WD;#+*8dx+BFqfjBOS9k1ZK{)ne0`| z<~Kywwwj8_U5>aZ)5(U`v{%~(e~1_wF|lS1rXxJrE|=|;G-RaC#<)~v-Omqklv%Nv zV0dVXRFFG`{_(Q6FfQx*JT-j(k%P8;P|7#q5D&b107@XUi(>P|x)6;}(^ML1*bwJq*|l^o`rvcti8Xko{z3b?DtZ zFvn*dieiXoNy3P{^v^aV44{Ce4KBWBlDR8bB08q)jsx0xmP)2vo3nyL6pN4({6HTI zP;%-7zp;#INjh?|=D%(1O&@vPa62z5$PI$r+x(b?S)~^`!#4_K4QQR zpxbXT^DIh_+!EfxCyTkPHrmOXrci~Ypyn`@7~IMzPw()@foR=hkVGFj2Jx9Cx9>`! zhs%Q^^8oAlUL}oq_?KA;RVk`YyT?r?#q5qDTF^b!o+Uy3fHc=4dq))*F-_W#|KetK z@FB-Y^AjtFaRlLO^RRc)b`wupkGUud>~+*BA81aEP1SRHZ%2we0 z<9naqJc!5)2RaeM|JPbS;1ZoWjaJraFh3=!gu9*&Bo}+tN4gkeopeubS0g_l8EEL9 zmOaWWx5MA*S|o+8G!c)?E>Ji2D2}Li#aRmWg2caupsA+Z51sW@h^U4vv@O&m3|)d= zg!*Zzh)pW#`f_WN!eQ+_)r1~88w32kT^IEyjo=V~TQ$v| zEQtdb#y!&YvsOIHAF-Uv7DxwAOei02WxY39A)WArun0|)=mTdhGNs55pWnjQr3DfI zrGg_hJ&^5ZOT9pDj~wV_8nL_DV_Nw4hif(mkXzQ&F-p(6qn3wie{IL-sb`MksO^cB zMJbt^{YXUU5E7bO`GCLJo&n300+38}UH(}9LWI6XzcE;sP8Zz}!+9~NgxqD%LzdY1 z%1HHc_JOyLCN2%x#aD2o`R=MAIl9#$e*`77`Pv_$sK6JaN8_87#(oxW7{75Yah4Qb z;y@3&ddZRT{RYQaR9Jw5A zv3+AW|Lv1bDzNmFODNExI*qpDrOS3a04i@7#+wmaNNNJDLoO~VpXC*(f#Nsz%w#zh zcXJn1D|~VNuYqyOv16LBegg4RQk2g-7X)1kXr1`yUQ?lDCl=tS#x#7QEpT|a2aPZQ zKjk^xKP{B_H|)O>{|T!g&4RNmvK4lk|Gy!=R+3pYtj}`_qMM8jQIBj+j^^BewG)56 zw_6swlYT`5-|yVqFU3kJz?jXUJ$3m$lSWM7e=8QBNV|bT&-;d~?oQZb@B>0H9_&ve z5%qQkwcH&pApHb0id#Pp$uaL~>)h$G9_^hfUvXTDc2b@-wo!5K97ldfbE!g+EgdSxA?0d^?H%nKgYsi1CR2dgga>4p zw7=5vkLw5^3VNCSOSEb*|IFw&r^idYfz!KTqW=K2kJgx4n~skkh9$N~IYFE2AH0R< zbdc{tvK^OL7?{bT1{G-=XyPYR7hL2$+UEVS^q~Ky1pO8C`Sj`@n`We1cdP%9(WOhG zkqI!|gYklJ#taf*A^Y7-Kq6i9kYBy!N;Y)j`FO}XXKy$aRO_7|n*|zJxw#o4X5l-= zCZ?c$1=11EoS=1Y-PSnV7EBwUGlroc@F69~MOon?us?fd|7yWxcM@E+?y|LOv^^sVm_7<<~;I>GGN zxx&?yRlRG>f$h)=6;C7vPbnmCjWz!R0F4M=$REHPLzS&UM`i}93WzeOrDuT8bNrvK#qa?y4Zi;R{)A2<& z7f9jd6ldkb@RusXyz6=U7sR!@oydev( z?|jB7rzKi}McyeGNfa#N$~~SK;EXQvdx(OV5;|^nxOr5I7#O^il2IoQ#QxEL;6sff zt;84W>VFQSin(z%SkBOwbuN=^Mj5n%^-wD>s*?{5$>VzDf%K#eVU`H5B;7sTW<$0| z-O&5HcXB>GL@HX)(Xg9sITd?pMG8j@rt*TND-}lk`#RRCChXd}IV{Payr!*}SOfM_ zCD$vsk7c}E{NiFdb(62x35P451mnIp6LRxt@a zV2l2z#8`Zw`1JHk!MczE!DE7uQ7(OXv4VB+LHD=^6fJtA^V zE8#O`BGU@NR5JD(Lto`Z^B|3oXuDH6eghYx-{C$ujS&3~1uTC%M);!?bh-x+;{e+c zSmTUNbSTjet?vA`pE0!bBqhcwEcZcexNrzU2}TcU$XJ8;@{aHEKG=;$JMI)Vtci;7 zs-o@$ByG|EX~6_y?K;)HSh(pC=+*mUTC&p^${QrDPGHi~VSS?v$1L;y37EF#8dD7x zhYN-@`xjo_<4mDTF&=EZL7nX_Ef@y!$Sxb?+?))!gE^MsDx?$bHc6`ngMK z&TJP|E5Z~)a^cnIp_)FD70O5T6khf5Uc2k&Ri{kCpi->T#br}?kG7Hrl0*jU`11Lg7U+d$hhmHAjNPhty1P{(E^X~c}m+*^YaAm`C9vKgY8a`HsO5nh^KY>p*jH5&v%#17Q|e_ZI~$xK1ex0gs76tEpf?fZgqRc zFpUIDqpTo>hghgKfim_+KfGPB4W1af?k<1Kv=Pv-4XCR;ae&tDc2*UDZa`&DXtsfE z<1W4j{z5>W+i?We0`Ca3H)SN7;5lux8KWmTB@2L3@twA-jvc`j1)e9NYzKHNR+!Fy ze)n#FB5WCm8ON}+6hyYtb)Qo#L|FPOX(YuTxs(%;NQPx09c~4t%xjvi|9#BAh53YU z1~+HCU_uL{ZbadMgY(wt1dHw(9bjkF4t5DtAZ^WZenyLxC`kVu?g<(b|I0+ASB8lD z6K(LTPL7WH>m6T(`@5T`8<(AB2f&l{o~*B9AdT^#~Su^?-h_*#>t4wZ3F?lX}zO;=N1uhYUjn2j~S0rv$Hi38~ zC;3q_{Z4Nr)dyn1R&`0*m7&9(XU~6hasHn1I=_cPFa`(Y87{aqQ1@v%5 z4rsyg2f8^e`XCjWe=R^#f*EgQyjIWXl$W(IDCseXkGL`KJck_JOG`>wqugnq%{PUR z*-}XHg6z2IR*Kq$5V?3jzBziDMvm^Gh5N0?v1qrhgtZhBiy(0t5$8^%)Qtn20e2te zR57UZ7$Zv+O*Zz+lyr9&YX;nX&^C7s-rJurFF_|W{sB?50EUJ$8ko9E|q2*|vQYJEk$mNq|Z;fC3&JvLInkg4s&JKKQ zm{AEC`^j!S+3fW29;Q1?T2^@2H%MLb<3_6$5VEceE2yE7G-tLW3s8029^`pRovMLh zX8~{Pz|2Z7Q`d`8U>zb&-Qw+uEGo^9_7TWI;zy7;j2PxTAtoYd>ej`-evsQ`Iz@V8 z3HWhtybqPPdZkoCBE^!Bm*^`44O`au!f0iO_tGydbTD1?S#T5sm*6O#KTEl! zY{xol1MH0c@UfN!!ln1)SzA0%(bp_4)`hrv5Vj?_Rhp$g(j5P31Upy#p6aRrT)B(6 z*nZbba$ti=tpfM(7P0J_R$8>n#AC!lbaH%kcC80iKc>{Ql6&5$RU>x=Is!i#PBaKIV-7a6Zg)m}$ zco|8^A{c0|HsGI%yf!ee0KxlObg)u9;kPW@%7*vRl=|9F3iok7Rcvhqxi%+iRR1iN zpkXtKAXnf?bv?c({6VW9%F=K$jazFi^IsR&G9u9EegybqLM zU?Rad7i6TQY>F=fHW0lL61MlxcZ1~X&P!JYy=zUT2?2YCkJ3H$zzsBbj01kP+GLD4 z14I$_Y{;&72Ft>VPCfn26z0?uB%T{(DtYCt?%=_El@=$>)mo8EQH8HpRIkJ-s>NTk z;U0okGC~yXV3wNzUn>!i2;9D^7Pq7hqrv*3l|p~cnC=h)+M(dA*Wln`1mMc%_AUQE z$eS2OiI>kCiH-zo2~wjA)-_*QvnCrwuxa4-%4| zM8Dz=ZYJ&fE#6rQK`DWi9thhVtZOHjWd9S7$u-61PjcDpxsWxns&{TVoG0S7*?9C? zIwaC>%RcE=Ri`Eo^1dFTaD$-MeaI149BoHy8Soz9qcxb1e)lVaiX3}yPuccjQY9Xf z88B4ZlB+n(un~w{(m328em3~799=fJcWmUgibZNHnT zOQO?koZNlG(J-;vZqa54rN$cU8VW#;K!z*{G^AWqDj}yWEOjM15_VRM2CwnbNS@hhfG=}WWVDpdZb!}D^m-KHnnk=`9{tzBx2zH z7ql^KDupCtq_ps!nna1fh4eSy*|awUg+|*vlcV)^iHM^7aVirz#HG+*9Le4Q&iX<*|EMq zFf-O@-`CvZx8FRg0DCGNPp`wf18C1*T8E$^&Kf@J~A$&wL9GQVC0AzCQQK z7sV8M-2%}Q-A`h<94LqZ3^vwoGDnVR`Vlmv;M|G=ZzpV2z5BLs`#WnVDMNUj^u6&} zy6$s|)^4+iJ^(g^wn%Nj-f?RDwPQ>&)}FLVSw&zK!sdVy#1Ed5Ql3mI>jjtnP;glp zwZOp@E}OofnGucyQo{sF)7Gg`3}JUZ7-syL{uU#A#)P-YPeqT*;WjQ*?vmV(t$-Jv ztyF6w+=7L(Emd#e-`6xa%P9`Sb1u&33e*%o2(fmzb%8xkckFpf=Q*7HK(B~y+ZNZz z@gbN`(V9I_b~EFF6B0~R(71-)#P>htdk!G{h!@V(%GJH^yFjD%l3h3mpAxRmSyzqe z;|Vv33wMgGS`f(-EaU0V*`5p?3WjSw-7Tg=Fws9DksCc}l^zmYo4S#5zy_99r5?PU zOpc==LE}+<+Ba~E6ns;%?!izlc#n-J6do3u&>m)x_T4ctl)7>^m^n?l7YH^mi10%_ zRIRF)-25r+Z6AZVJVKKdFm!5@p+UDH5e5Q^=2hjmDFO%3GWPN*|0qw;f7&BCI)YFH z<~tY{aaduX-oUSZ0`)p)sCZVZ>I-A!+L<572pOdFvIQ<6^c&A7^o1HqO@51EV~b7P zzn!Pl6)VuoT!Vm1_}UCVKwt|LxviCbohEX35Zk(GbSPl+ISbelmO?0+l&J)ThO<3} z-e=+MAtetXRB4~e*)kEpE|2IJ@q+_$4w_lwv&qdd8Xe*jI}crJ?JRw+WtyuGv-hm` zAmG0Q`6)Wkhf0Q%FJ<75jB16N*3rE*15n7JJmC$ZR)#!lct?|u$gYXJ^?eBEAad+I zVJ~x+nTi)!co`b71f6Vs^>f%$B6*)>V7|9nIra^6_o$lMo)j-=9LYLw({xC1pr+n3W0I+ zjNiK-S$JJPa=1itmpTw5NSCARG-p*S5Te>>L0q`=*y}vdXr%3O`U<+-(CK5Y3`{Xm zjGl91&5P$S_J!s8#a->Hs&h|Pjk=1G4&bxxzK+Gr;lXP6z8ZT-OWqp|?5dVFWnwnH zOF_BAz*_P(Gaue!TpODS_h(k(*t6*~LG4Ffr=QZ?>naW00XoKvt zNJUI^(I6&YzaeVM{Zz3Byf#=eK^CL#3KA?yGhB`Tr(sJ7xJ{lUt_$#32!ny*#bcFqrFGdvqRPREFGvX)*rnjBPrpKS4HmDp78A&O>TW@n?ug^( zf2$w_;lfBz>xa$aCSx_<@x=H28?}zei^{`gHQ_#In9D0jZ}3!5y_P(rKqeA&R$>q{ z7#tTXyn(5!q@KR&q}PU+n?F+s->Is>#u1Ft=Q+fgG(b5y8a+s13Y6_PQcGeW+$QRw zZ@`gbDR|9IW?e_{-;&g*uyRYU136~|ilSP*0NIh9+7#~Dj_BmOYf2H|%mscBk;7OG z_7#^bm^DuM9Wf9A1v&N6m#{m}G_$>~^bSdxT!(>~Qwa$IRcoZcyh_Mc_=a|iT9Gz% zEHH!isnulMR?(~ymx(SEb}wnD>J%gD`%Lu8+p=yA*Js-3t=r$1fmtmon<2c3ik$M2 zS)F(brS0;M1q4#t$2xVg<2%$%OET4WNFuf!=;;{ik5>Mq)y#A~;l6bKNYIirTa%)E!b4U;+t9~OlZ*Z)9<%ii z%b_9mM}9RxDQR{eq5Hl=FEm_4d+DqAxF-G}8INKrSver26Ges#A#c0(?*AIxAIiLl z<=nysRZvceq)a63gXWm$F2+=nl2gALGI!$O3&XChIuF9S`+7iTJSiiOz zcml;ccmI?h;4Kn-t?co{@b!hYm3kt13!AtgNHG% zJaYF@Bu?$+;KqMQ(3R-6=JAg==BD>b+FJU+*9JT!7+kZg%}O8t*bG($8f{|>#CoC? z%i-|KP_G#YwK2>+>E@u-|8wmge>e9Z%qz6q&EZ;j;h*T^DBT+6iE{revII6>6``#Zn68J6U>h?K;V(Oj>8mJKwXKbj;+9~2YfJVQV zZ-kD>8)0j}1j)<5C<>D;A>c3971l(oNJA1-MBr9w4ge{i#RUeN^(6iSlm8@)4e8Qq zj1Wt#X|~uQMndS1dsITicIB4^TFYYQV7X1?+(8EWb5Zx4U~z*tm`6p4Ye-3*? z>Uc#OP2KuyT1fM5Eef4ckQ&&-$11qc&(C6Kh7K-L%-{4L(?`-k{E#Db-#?Byd?6m! z_xB7e)7^ka>e_O2MK_z_K8bF3Kr-ewRCPBp_}C`Mb4$-d zJ?j+ELq}DNsoj#X+D@EcCphNlDM!}$5UGn6N&ZSH?P|TwAJi3bOJI71G=zR1eGxtD zy{8=#N`Nz(eB6D(KTcK>lbV$7MW>)Bgh3(drMK_g{&G5TJslw*y$DYWR*bMjTzy2; zF>v1e>3{Z%p?Gorio*DN29}GR8C;M7i~b2&3~O@x^?=ZWHkA6l%lkq{)M#?XswZ`y zxSNk&e(rVG)MUcQzfpl=I%xn^KZ!P*+;+Tj>ZBVRwwQ^Zo8STBv6DDWQ308xy#f< zaqDdNkl#P=bGsY#CB!bG+)c*ytPFmWnExH_7vX~O2Gm)Mr-PBiKvO_)atp|Eb*D`i%<7ma{do>(*BbLppUda z^QP_!K~j)Ji+gpMZDjubm_?=ZE7$yDSZ>5*9KLVKEY_2Jf(Y{za;UiTz~J10Lq*nC ziJ=A#@|bPpK}xu1SCrmP3`<3{NRNlSYo+NSwwC$t&gQ`&z{CoHyghwV`9IuB_3~GVA-45{II&Er)~)<7%AdjhvR zar*1{XV9%v;g(d z=E7=^q0APgROaYI5KV33hv@5NR8*n=d_CwgX77K-Mf!4pwoHrlj=;n%#E=xuFhf=f zOx$4K)e;&p%qj@#fjQQ;nmaG=)ajuP(?0}{pOJ{WZgqLuBZh50J)!xsswCG!&GwdV zu;o!5K>DL2CCARbQU)S_%XXP|cSi{by2|bNCqDJ{NzhM9O`447PD9x9qD!EhR&Z9Z zo{%CqFr6-1c@n;>fC7geEm)Lw=F07*CLZMor5@vij532E)fUB+pQ01x0t1P$Mv_F%3YUp%;(dBJb?$^hW4(;#?jP$ zsur*Jh@qXI4X!RDWSfH0gJOEu`f zAdh38)NzZ&Ax{+FCKU{+7|7SCGj}7!#Yw@L!p2)4%oM=xFo-WP(%F$>cw9FWR|VT1 zMC1D=mT_WQ`{*!eW3y-;e2)U*_hCFy5HobJM`@WV2BVu+=(7w37yY-2GfTk^;t@TR zV4#`Jb1nt-)mpDb{CMa#AIOZ35=m_5^WYteN}?j*HHIH~1ytZ7)YMs@mIlKKlpPJ! zGQ>Hm5DA(#9dUOxMJQXN`9P|7^Qe$BE_C5?mmRCA!K0B~LsYLmPo{$?AV1s|_SwBX z?@?ZuTnQt>;3T#=SXJlZE{5Q*eX``egy6+h13RT;$4>zD7b-C7$^Z&ia>6S3m-sY= zIQL7{qHhsae_h_LZ~qnQ{RMG_f2B**X7w_*S6;f;-!8%pB)p>OYwC>Y^j?OP7PWp) z!M-&+zsn>7Sz41UkgO{XT;^!Agye*c>4x(k8E@n@$>&DHRg72_;!t`3zsIjy1IX2T zFkBgWH(R3wU0Rj*GX(FY963qMt|~4F3q#9j5Q&IMYt3ztA`;1E%1UVbl#kv~yw*+> zM(}eW`m2L;UcYnGK`V!eMMF)KfoziR(H(9JzSn5L z?ObT9{deI-K#-=^KgVQ&l5Q9fwkf`lkq7{%)>@t>#T5ORwI~D$Or`X^>E}mlcDUoB z2r5R=K`WsM4W)G9#V=e31H-lyoNHAHYW!=`GwD&ZLpQy&-Tr^luHgN>1 zru1<|V|(0%RXYWM@X_K2|8Vz35iA+yFxr*YWIm%QngZ!+6^O68^P_ z%mMdTXQxBZ!PHJ&V(p4PHwFL3VtnKZ(<=F;i0Trr)Gh2748qx)7t1bS^5c0dN? z5#rJgwK6#qao%n6gMX99jPMT4VF_dy;Sb@`F2FA1Sxo;y$$uW4crtyan4vH5XV7?C z#^^^yWsjTH1*rD%Ay$Zdro5%>rfZa_e9x}&c$DS2wx2ckcs~dg87F!V(zJ~H%UAM} zcHfr;muxAZaH+#oHqs2G2Ob!;4P`bL8s9?t55GYczRvK0_de6|3;YY#?}+GdxV~Oy{HJqNtVxfot42Va#2S5tGd765Sv|po)FKq6|>I%kV1V zplw^0z@8fRCVz^;21Q}7G-?zdXs$@5jOsXY&;g!r!L$OfPM8S1cM$8pjixjGZ6de^3*AkflW_KM>HC*k6U{v`36sXi!wN;F+!^S>MXTpg@etZfY zpQ=QcPqG~+6CW+r_h`}$A; z9{8Z}ZXik#=${V444}uKUY?FGV$=6yAp5Y*r^J z^{E*UKITk_EMj9j6!!E_NV(@8-wTv8nwjW&l-RAq)95ihT4+=OXNbQfM+!#=!bDLJ zQhqhftUc2u6DjrnQI%Y))?L!oP@8A$$y~muuFGQ}xGtgiauSAX+4Kihtu zhmCP3t`Cp>KQ5BS3j_K7gT&)<-?8<#*a@ zh3XOdTnB@v;{i5dgw$Rq*N6ZZI{NRguTAwg3rUiSIu=K?Yu1+9{-i?Zb6qju$y&?* zTb;^oWW>czB3Gmu+1#W{gwGk?7ug~lX$DFI@qf1^@^V4{)<6nR&~mms+L0=0a^D-J z92a=m+c=ufgVC-&nUx^zaekJr8!u0JLmw&! z_Mo>NS}U_Re`79Cr_7}0&2Pab(R|I z-BERn%Aa9rzBkZ2^iN=yYmZ#Q@Xxs47W(ek?VZx6$xeUNq)V!&CO7D@E^eUA-EAe1D0CExgV`mdl!MMo@xcgXzIs~PH#0H5z1xvkE> z#9!{#XcvPnOplK@v30=(_lCP@p&g^HWwS24Ru^89%s+#P(nIdGKtpbrG}}$Mvw|SH zP)FZYc8JY_NsWsGSC?XP;jF>d@B2U{OU|Z5hf8TFEFGq4EAxh$JZR{#HicLy7Hen( zrVpJo)u+@&(?h)|US&|mNzOgn(%Eo^1s}UHjkXOg2Hq_RJX1;Ai!2W-JA#_-qiPUh z!L;b%sN-Bl0gD}ZfCgi2fX()9n3WzceM!7-i^CdG>i6l_%Vt!Ia0^mTwHuUV9p~G` zaT?7b1)e=!MdMU(MF{b@J!_vP#xmVt!n5D4eigdAy0soUWffK9(cJ2TVbYR~U4Y_k zk>2&1@s$Cc0qAhA{Em>t5N3;Nq5BTg+t9>9BPHPvBoGzdBL(~~abdkmc zMozWrPidc@ej2vjLtW*)YU9!lnIhH8u#`J1)0uyPY)m1)u3lGzUNtNuqHWU4giwc` zY$s=sM7KcrsVJqx6fyLZiZ7_P>c(rau>{q?jZhkz!vfycSEVi&VW%3b8!Z>05sVri zk*X`6O@oL69Vut?@+G&tX=vEn);csTi~U|Ph*tKwtl?D;8gSlG4b(}pedi0#+tPqL zWF&EItg?az$=S2KJ~mPzHsK5ijqNI6ljvE}{m=CjQ`g5%9eRUWn4m*pg^U(U^u3H_ zz7y>QC#nHNtn1g44-v1qF%>ZCBWYkQ(7v zII@vvM9wKF)~A9XukGujMmU|78gedMS9l)2wqwS8mBnp{qD&2@^GV?m)MN9~4GgVmg@ zN}`k0`tE22(&!^8gl6NC$T$+>R(3a@a}I+>KadF6JcPEY0JlGIw)J$EMDMW;sBcs| zQM>=;TL%uEL#`#?2|l!-0|;iV;klaAlBG+ROwF5zTkOo$+2K&l54O(SxA-$hsMSj=U)I0!z@n!AKe#kUpIyV zGlsmrt;Fi77)CPP;%4P)I6i*-OuDK+LpX0!91T2TbrxDRU??`2v~P zOzqBMy6%htBLIq2?S7SWml%tnmg<4F#?uGTu~72we4|4+0#CX&fdgBnYp)LCTjrzC zYI{Z#AR1z~1aO|a60lDdGJZvLlA5My!~P2T{0|WONlLsrun~3l7$+Y1 z+k2C`Fs&FfMh`~xe_o6(a{wWbdI`V53R2LDbKVwVm(s(nHA6Xf3}HPH!D6}kW|T+$ zm(kM`jTuwQj%L{={)eK!*xk>nZkS~4X4zcZVK5@RcRQOZJ|y;g?gRMuk$s28cEV)I z;cna9jsljcv3#vWB4@|^QIT4D_+i z@1Wo7e5QECCBMbe2fGhPEdXtSk8?Jer% z^>H}f_=JM)R-Ysc>ov7it~|r9I=>bFf6y0GhHyD>U|hm!*|Bb4qI*wr^sY>^>i2Sk z7}jeD^2BXP)tGu8K&^cJ1HZ^@x>!62ELR(@ot9O}tnmZzCaEh|F@58g?u1!%XvplU z=|8139YVrds-_szq?4rWIisu*Mt;jR_$oCLBY!)s@p4sdCCV<`d)0OuW35Y&?9#~U zvO!TM+R6Xl%I=6~dprct{Lpv8K&aR5hP`}`)L!T*istiQ@=#!XY#8JV{)#iD(lTY) zH5dx44ZcUs88cYqzn#$?n9mwNK9qOLMLNJf}c{K+3yz*B9xjo zYe@(+WUg(YsJArbkp*j#Fy`dD5L#~$l&pRrIa+R+aSpFDO4Jmp5k|8`np)Y){6r?l zi+NO24|fvh5Be_lDIG^wr9-eU-Rn|)hy;UEbgYLQ4iJz4tIXOQOyIPR25w}T@l5+r zLCAV5nfBE%akWx2h}bVhlCj+LG7T9S-1cY-Mb2_4$03jhwJduLhDEti>^#!eQ5w#_f|$zH;#Qi#QZ;d6lhwqrc2?QJs>ydNK}^zonl?d`!?-JU*EcLH2SPe(0mc}oRLAD`kYL1>P2_+^*qEOf~m--_-dITnRMsE4tKkQSxWgtakp2x_!ghyu19YEJqns*!)BNC*4cme>HJ| ze0UVbb)m6vS@6mPFi<|sr^mJcik|1UiS@0ju4Ygz6urV&B5TD-iSm&rR8MTv;x3^Q!_ig$C|rKWVK3VS?rV>BI~k-Q)FvQF(sV+gW7ZLvsM_| z$1J9G3JS|S1|Sf|w~Rjdg4!QF^%kJvv34lB3C^&)^hRU?r~uyVam(CWUBE9Psy^`- zsp_RwZ;{r!rzK|Ac4lTTZ4EIG214KPMz%mIgcxds;-ZtkPLF6z1m0=>%UiKzW+&em z|H1h}Xs<0yd>%yYH#^Nt(%9CR1bKX*6_#2p31UV$&y>le4h?1iKWYyQMN8xHM?G?k zA$_jV8kt-ck7CY-`q*sLt>j`pOS=1x^6XSaVLD3;rhI#E*-Qli4GuT4KhQ!IZ%UBI zZXrlp<{`kZw>C1z2{|G^s+`x*>m%Eg#Xl?h0kudMy^>Ve;l!MPfktJy<;g>P;+e)b$2M^!K* zV47|tAZr6}v*W@vH^5h!)>qDI%vnfEDditdg&K~_DprcVx;~2mK!gW5wi-0r2PIY* zEro4{@tHs6f2CTJXi^XT>GF?`wK^nbHKMtAgG*woGhV`%m%qN}QR{q zj?uOzGp&-^8&^y9N?uaXFifhp7kg#h7Fl1KKC#7^AXsN8X;4hv=A98RjCs$<+X6;p zHLmZz!4_M2!tB`enJgHtoy8!z8-V?l$lmTy5LsgbFFAVCsMBkR)G)nG-Oc7o2OE{9 zJz1h53f(sFms_Jy4<#>4NLV0k9_Xhk_ax{-?1sOmepO5=M|G4At1L~=a#B_dcI!jC zdqmFZa(rL`69cw)BaGDw;1iavNB;N`Y{{ zU8mD4Uw~egH#itk_YmEH0gOYP$BLt+@*n#bc+1d0MPw(*wr(-)SD;wQ-^|QRz$8`^ zih4-LHh9gX0vqXY_U;);YN-3tl0t%SyN2fy#%r+t>joX1%Yg3->>FsDI)jWvwgkwy ze62coZ3W8xeOXrjSg7wr$ZE-1(Ok0%LREqjm%tazaP#3OrYX>BlwC-Ee_N)Ux!yeJ zM0F4ub)}h>X|_tM(a!=;HKB1CZqBUVw=tO%OT36J7{dfiYYF_@vq@Whi-m<1bJVBC zjgpHwfSd2i4{_zA07|ali%rxgxSV+%to>Qc5yeb#Vc?Z3t$bI%@IgDP4|cqslIDCY zL;5ZvJqg=zot09bMdB&y$!#VqrG?65t^^0_%?26aJIG%><$~h_Hqk`vw+KZ2je?4km z_H2}1;IMucBY^KH;dhXYbXGWRB>mJf+GKz06pN?fh&tf~KURZAuFmN-8&Y%8Yy_cW zpHT2$^XdUyskgy}W{6Gau#hKb;z=ORznK0sV)OqiTK<%*h3q6A65 z*&8MA{@DZgCWE@#losfa$g_)>z%CkqyY8A4vN$@m1^&+VUZuL5tF~caArCqv&Bsv- z;$DNaP$N$dFyS@cu4oJNZASvE+8TCml#eU48i{p)JW1_{F)F3c3zlKwjA&2*SwN=0 zxB=Lod|+|0zPU1VwvWz@AO&EZm@4n7WA5#{tA~VW$qS5&m4zQYaGE)i7 zj!^7eQaw{W|D^qhsRt+gZ~bRHDq>P%^#FNX!Ah=%>>PLl(BT#oMkE2>vsM+m;Y-mv zS+0qbkRY$w+jW_rL?gT20?F8Z*=Ho31?V-fnr^MI%qB!D)_lBlj>$pt6{}E4wCX2C zjCORdYbs9wXxG6#dnvO7Wp7*sTvvWJQB~aM{)y?Az&rT^kXTq=~hIgD>HrLt_SYn+> zZVFF}XzLxW9~=`D_y4Qri zcU_VDe--6$Ds;B44X$5)eBSLjJ3@;*8gJ|E40efNE_YK)t)m7JO6j54?g&7~ike%R z2hoIaDK7Rxr|g``Uo~HY~IC*EnR$H-!pFrd#nFbU7wSA zw;|!@l`WH@Qi+?UKfjPp-8M=|BN*WWcoO+{!_lt}wwSsePHE;jIOENx8Pnsu8gI>M zY?KDWX_7%91D(R42E!g=5prKCB>cxEED^h9<*@72?o%`k&Fsc|tF%coLl6ac5FQMz zWqQ{E$bT>Mtp^GgR&XXyZd5j61IO_2m`Ezbp9dUJK9Ct|u{c2#2~MbCSkasg638cN ztll*CiKB-n+t4TUhz#Zm{^blqcx$$dL_j~vOCX+b5wA6OL0! zCsT@ns$AmIR;_l5?hsu~rhf-vyA-PE}3WZYTH zk*!pY8JQGCx;p3DCBo_4JKb!t5&7>ipY8aLfKHjVSLF!52foz0YwW>DW<)caybu1ivm% zsS|ly7x=td$6JG`v@{hpeypT!hdGqu0OIjaFYtZEtC7|Rxo~wNj?}|FwFm{ZPCJ%Q z^z&(8v(gRronj|OxgtIjuPgxo8Ezk?^sf|e?S7k)$|3Xl9t<&g2)oskfM~+*$T!Lp z995{+a?C74A}kMGSOkA_IVr}#0#82QAfx%3 zxASj-Fv?rmASuO{Y}^FM(JmNO;v^Ws*?lI-W=_9dAd6ErFhd9B(Cz{BS80dIV?O_u zTy@*+GH)qX=+Jh`;nARB7j}*ap_yGmDl6f!ebYT%F~ewyLwVVOC>@(wGcN+zlri`M z2~aCJ*Z-T#1@#4^WAJ!F%!GfFc-?pD`iBsc>EF0d%kF*cm! zBT+T!;%x47DTX%cxo$7gykKvq@N4=4JMkICgl1!86wfyh4XLqOVt929+4ISLM%zI$ z23;G7P7Y{T5~Noq8|ZyeSH*XXKnU~dq$1Aik$-^Je;4RT9pXNtGG z-^dyU@?WHDUA*wDu>5yf*d70fC(J@}m97)O5RRSsqK-0Hjmmsj6foc|P&HC~L8NU| zN#tIIzr+sFbh6cyuS3TF6lWhbfve)9an?OtT~31^$yq8C!CvQm5udxu(?VHF!iVQv zGl~oeXk{DEKe1X?nAivKojwUF$wEkM!@|WACqD>_txi1#_7RPQQv=A!!>(0{J0v6N zB?D2nJeL{l&$o8u)))qjkyOmiC}~g`$%FP} zZ!T#3s)sSh(XaQ3>ma3eQif~>Mx@MG{S&Ti6_O19-AD#((79=|(pQuJ*J+f5%rJaJ z&R^f2&3N^T!mq1nS&)>s1;6%TV`sKt{^EW*;4h03ov;P*MHGeK>5|B9`4Ze9!sIgBNXKEwfHi~7~4~j zCeE#@>9%khmG9-VFC{|{Z&W+G(j6PPu9c68?Hmg3+7okN_`Epr7*0Iu?qK71893y> z2)QNuzzl*m@lRV7BKqAc^wd1d%nUJ-4%UF)4_e&;{lGD1~WE2*gWyURoGC^nD$Z)dyMx@<0aFGrfao*F{98^t5VHqwjyt( zz*gDKsHwxBx6yY7RFAEd10gkdK-P2*!=!g{`Fs7%gs zo;i|hoW zZ1cmRT?NlqD^)oZ7^~?i2l@ zw|)qSBlL_^JZWbUEX^Wp0T)M7ja6Z8Emnj0#T≻VlazhDKzUO+GNigLoVjz<8>_ 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 T0l)`n+6syAhzjuB=i~b?_A*nf literal 0 HcmV?d00001 diff --git a/web/src/app.ts b/web/src/app.ts new file mode 100644 index 0000000..9312970 --- /dev/null +++ b/web/src/app.ts @@ -0,0 +1,33 @@ +import type { RequestConfig } from 'umi'; + +export const request: RequestConfig = { + errorConfig: { + adaptor: (resData) => { + return { + ...resData, + success: resData.success, + showType: 0, + }; + }, + }, + requestInterceptors: [ + (url, options) => { + return { + url: url, + options: { + ...options, + timeout: 180000, + errorHandler: (e) => { + console.log('---------------------------------------------'); + console.log('error.name:', e.name); + console.log('error.response:', e.response); + console.log('error.request:', e.request); + console.log('error.type:', e.type); + console.log('============================================='); + throw e; + }, + }, + }; + }, + ], +}; diff --git a/web/src/global.less b/web/src/global.less new file mode 100644 index 0000000..7ea066c --- /dev/null +++ b/web/src/global.less @@ -0,0 +1,394 @@ +@prefix: ob-table; +@subTitleColor: #5c6b8a; +@actionColor: #1677ff; +@tablesRowSelectedBgColor: #eaf1ff; +@tablesRowEvenBgColor: #f8fafe; +@tablesHeadColor: #5c6b8a; +@nestingTablesBg: #f8fafe; +@colorBgContainer: #ffffff; +@colorFillQuaternary: #f8fafe; +@colorPrimaryBgHover: #eaf1ff; +@colorTextBase: #132039; +@colorBgBase: #ffffff; +@colorPrimaryBg: #eaf1ff; +@colorTextSecondary: #364563; +@blue: #1677ff; + +html, +body, +#root { + min-width: 1040px; + height: 100%; + margin: 0 !important; + font-size: 14px; + font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, + 'Helvetica Neue', Arial, 'Noto Sans', sans-serif, 'Apple Color Emoji', + 'Segoe UI Emoji', 'Segoe UI Symbol', 'Noto Color Emoji'; +} + +#root { + .ant-pro-card { + border-radius: 8px !important; + } + + .ant-row { + margin-right: 0px !important; + margin-left: 0px !important; + } + /* primary btn */ + .ant-btn-primary:not([disabled]):not(.ant-btn-background-ghost) { + background: linear-gradient(-45deg, #002bff 0%, #0080ff 100%); + border: none; + box-shadow: none; + &:hover { + background: linear-gradient(-45deg, #002bff 60%, #0080ff 100%); + } + &:active { + background: linear-gradient(-45deg, #002bff 90%, #0080ff 100%); + } + } + + /* Table */ + .ant-table { + .ant-table-thead { + tr { + td, + th { + padding: 12px 16px; + // 弱化列标题字体 + color: #5c6b8a; + font-weight: normal; + font-size: 14px; + line-height: 22px; + background-color: @colorBgContainer; + } + } + } + .ant-table-tbody { + tr { + td { + padding: 12px 16px; + // 去掉表格边框 + border: none; + // 去掉 hover 时的背景圆角 + border-radius: 0; + } + } + // 斑马纹样式 + tr:nth-child(2n + 1):not(.ant-table-placeholder) { + td { + background-color: @colorBgContainer; + } + } + tr:nth-child(2n):not(.ant-table-placeholder) { + td { + background-color: @colorFillQuaternary; + } + } + // 伪类选择器样式优先级: hover < nth-child,因此需要将 hover 样式写到后面 + tr:not(.ant-table-placeholder):hover { + td { + background-color: @colorPrimaryBgHover; + } + } + } + } + + /* ob-table */ + .@{prefix}.ant-table-wrapper { + color: @colorTextBase; + font-size: 14px; + background: @colorBgBase; + border-radius: 8px; + .ant-table-thead > tr > th { + padding: 12px 16px !important; + color: @tablesHeadColor; + line-height: 22px; + background: @colorBgBase; + } + .ant-table-thead > tr > td.ant-table-row-expand-icon-cell { + background: @colorBgBase; + } + .ant-table-tbody > tr > td { + padding: 12px 16px; + border: none !important; + border-radius: 0; + } + .ant-table-tbody > tr.ant-table-row:hover > td:first-child { + border-radius: 0 !important; + } + .ant-table-tbody > tr.ant-table-row:hover > td:last-child { + border-radius: 0 !important; + } + .ant-table-tbody .ant-table-row:nth-child(2n - 1) { + background: @colorFillQuaternary; + } + .ant-table-tbody .ant-table-expanded-row td { + background: @colorBgBase; + } + .ant-table-tbody > tr.ant-table-row-selected > td { + box-sizing: border-box; + background: @colorPrimaryBg !important; + border-bottom: 1px solid @colorBgBase !important; + } + .ant-table:not(.ant-table-bordered) + .ant-table-tbody + > tr.ant-table-row.ant-table-row-selected + > td:first-child, + .ant-table:not(.ant-table-bordered) + .ant-table-tbody + > tr.ant-table-row.ant-table-row-selected + > td:last-child { + border-radius: 0; + } + .@{prefix}-footer-bar { + display: flex; + align-items: center; + justify-content: space-between; + width: 100%; + } + .ant-pagination-total-text { + display: flex; + flex-grow: 1; + justify-content: flex-end; + } + .@{prefix}-total-options-bar { + margin-right: 24px; + } + .@{prefix}-total-text { + color: @colorTextSecondary; + font-weight: 500; + font-size: 14px; + } + .@{prefix}-total-number { + margin: 0 8px; + color: @blue; + } + .@{prefix}-option-bar-cancel, + .@{prefix}-option-bar-open { + color: @blue; + cursor: pointer; + } + .ant-pagination.ant-table-pagination { + margin: 16px 24px; + padding: 0; + overflow: hidden; + background: @colorBgBase; + border-radius: 8px; + } + + // 嵌套表格样式 + .ant-table-tbody > tr > td > .ant-table-wrapper:only-child .ant-table, + .ant-table.ant-table-middle + .ant-table-tbody + .ant-table-wrapper:only-child + .ant-table, + .ant-table.ant-table-small + .ant-table-tbody + .ant-table-wrapper:only-child + .ant-table { + margin-block: 0; + margin-inline: 0; + } + .ant-table-cell { + .@{prefix} { + background: @colorFillQuaternary; + .ant-table-thead > tr > th { + background: @colorFillQuaternary; + } + .ant-table-tbody .ant-table-row td { + background: @colorFillQuaternary; + } + } + } + } + + .@{prefix}-tool-selected-content.ant-popover { + .ant-popover-content { + .ant-popover-inner { + padding: 0; + .ant-popover-inner-content { + padding: 0; + } + } + } + } + + .@{prefix}.@{prefix}-expandable { + .ant-table-tbody tr.ant-table-row > td { + background: @colorBgBase; + } + .ant-table-tbody tr:nth-child(2n) td { + background: @colorBgBase; + } + .ant-table-tbody tr:hover td { + background: @colorPrimaryBg; + } + } + + .ant-pro-card { + .ant-pro-card-col.ant-pro-card-split-horizontal { + border-block-end: none !important; + } + } + + .ant-select { + .ant-select-item-option-selected { + background-color: #eaf1ff !important; + } + .ant-select-item-option-active:not(.ant-select-item-option-selected) { + background-color: #e2e8f3 !important; + } + } + + .ant-form { + .ant-form-item-label > label { + color: @subTitleColor !important; + } + + .ant-form-item-required::before { + display: none !important; + } + } + + .ant-timeline { + .ant-timeline-item-head-custom { + background-color: rgba(0, 0, 0, 0); + } + } +} + +/* primary btn danger */ +.ant-btn-dangerous:not([disabled]):not(.ant-btn-background-ghost) { + background: linear-gradient(-45deg, #ff4d67 0%, #ff6a80 100%); + border: none; + box-shadow: none; + &:hover { + background: linear-gradient(-45deg, #ff4d67 60%, #ff6a80 100%); + } + &:active { + background: linear-gradient(-45deg, #ff4d67 90%, #ff6a80 100%); + } +} + +/* default btn */ +.ant-btn-default:not([disabled]) { + color: #132039; + background-color: #ffffff; + border-color: #cdd5e4; + box-shadow: 0 2px 0 #f8fafe; + &:hover { + color: #004ce6; + border-color: #004ce6; + } + &:active { + color: #004ce6; + border-color: #004ce6; + } +} + +.ant-modal-confirm { + .ant-modal-confirm-body { + .ant-modal-confirm-title { + color: #132039 !important; + } + .ant-modal-confirm-content { + color: #5c6b8a !important; + } + } +} + +.error-color { + background-image: linear-gradient(-52deg, #ff4d67 2%, #ff6a80 97%); +} + +.warning-color { + background-image: linear-gradient(131deg, #fbe031 0%, #f6bd16 100%); +} + +.default-tag { + color: #8592ad !important; + font-weight: normal !important; + font-size: 12px !important; + background-color: #f8fafe; + border: 1px solid rgba(0, 0, 0, 0.15); + margin-inline-end: 0 !important; +} + +.green-tag { + color: #0ac185 !important; + font-weight: normal !important; + background-color: #eef8f5; + border: 1px solid #b3e6d5; + margin-inline-end: 0 !important; +} + +.blue-tag { + color: #006aff !important; + font-weight: normal !important; + background-color: #eaf1ff; + border: 1px solid #b3ccff; + margin-inline-end: 0 !important; +} + +.ml-8 { + margin-left: 8px; +} + +.ml-10 { + margin-left: 10px; +} + +.ml-20 { + margin-left: 20px; +} + +.mr-6 { + margin-right: 6px; +} + +.mr-10 { + margin-right: 10px; +} + +.remind-color { + color: #ffac33 !important; +} + +.ellipsis { + display: -webkit-box; + overflow: hidden; + word-break: break-all; + -webkit-line-clamp: 1; + -webkit-box-orient: vertical; +} + +.form-item-no-bottom { + margin-bottom: 0 !important; +} + +.list-tooltip { + .ant-tooltip-inner { + max-height: 200px; + overflow: auto; + word-break: break-all; + } +} + +.card-header-padding-top-0 { + & > .ant-pro-card-header { + padding-top: 0px; + } +} + +.card-padding-bottom-24 { + & > .ant-pro-card-body { + padding-bottom: 24px; + } +} + +.card-padding-top-0 { + & > .ant-pro-card-body { + padding-top: 0px; + } +} diff --git a/web/src/models/global.ts b/web/src/models/global.ts new file mode 100755 index 0000000..41a7ab2 --- /dev/null +++ b/web/src/models/global.ts @@ -0,0 +1,68 @@ +import { useState } from 'react'; +import useRequest from '@/utils/useRequest'; +import { finishInstallAndKillProcess } from '@/services/ob-deploy-web/Processes'; +import { queryDeploymentConfig } from '@/services/ob-deploy-web/Deployments'; + +export default () => { + const initAppName = 'myoceanbase'; + const [currentStep, setCurrentStep] = useState(0); + const [configData, setConfigData] = useState({}); + const [currentType, setCurrentType] = useState('all'); + const [checkOK, setCheckOK] = useState(false); + const [installStatus, setInstallStatus] = useState('RUNNING'); + const [lowVersion, setLowVersion] = useState(false); + const [isFirstTime, setIsFirstTime] = useState(true); + const [isDraft, setIsDraft] = useState(false); + const [clusterMore, setClusterMore] = useState(false); + const [nameIndex, setNameIndex] = useState(4); + + const [clusterMoreConfig, setClusterMoreConfig] = useState< + API.NewParameterMeta[] + >([]); + const [componentsMore, setComponentsMore] = useState(false); + const [componentsMoreConfig, setComponentsMoreConfig] = useState< + API.NewParameterMeta[] + >([]); + const [componentsVersionInfo, setComponentsVersionInfo] = + useState({}); + + const { run: handleQuitProgress } = useRequest(finishInstallAndKillProcess); + const { run: getInfoByName } = useRequest(queryDeploymentConfig, { + skipStatusError: true, + throwOnError: true, + }); + + return { + initAppName, + currentStep, + setCurrentStep, + configData, + setConfigData, + currentType, + setCurrentType, + checkOK, + setCheckOK, + installStatus, + setInstallStatus, + lowVersion, + setLowVersion, + isFirstTime, + setIsFirstTime, + isDraft, + setIsDraft, + clusterMore, + setClusterMore, + componentsMore, + setComponentsMore, + clusterMoreConfig, + setClusterMoreConfig, + componentsMoreConfig, + setComponentsMoreConfig, + componentsVersionInfo, + setComponentsVersionInfo, + handleQuitProgress, + getInfoByName, + nameIndex, + setNameIndex, + }; +}; diff --git a/web/src/pages/components/CheckInfo.tsx b/web/src/pages/components/CheckInfo.tsx new file mode 100644 index 0000000..09b496f --- /dev/null +++ b/web/src/pages/components/CheckInfo.tsx @@ -0,0 +1,489 @@ +import { useState } from 'react'; +import { useModel } from 'umi'; +import { Space, Button, Table, Row, Col, Alert, Tooltip } from 'antd'; +import { ProCard } from '@ant-design/pro-components'; +import type { ColumnsType } from 'antd/es/table'; +import { EyeOutlined, EyeInvisibleOutlined } from '@ant-design/icons'; +import useRequest from '@/utils/useRequest'; +import { createDeploymentConfig } from '@/services/ob-deploy-web/Deployments'; +import { handleQuit } from '@/utils'; +import { + componentsNameConfig, + allComponentsKeys, + onlyComponentsKeys, + modeConfig, + obproxyComponent, +} from '../constants'; +import styles from './index.less'; +interface ComponentsNodeConfig { + name: string; + servers: string[]; + key: string; + isTooltip: boolean; +} + +export default function CheckInfo() { + const { + configData, + currentType, + setCheckOK, + lowVersion, + setCurrentStep, + handleQuitProgress, + } = useModel('global'); + const { components = {}, auth, home_path } = configData || {}; + const { + oceanbase = {}, + obproxy = {}, + ocpexpress = {}, + obagent = {}, + } = components; + const [showPwd, setShowPwd] = useState(false); + + const { run: handleCreateConfig, loading } = useRequest( + createDeploymentConfig, + { + onSuccess: ({ success }: API.OBResponse) => { + if (success) { + setCheckOK(true); + } + }, + }, + ); + + const prevStep = () => { + setCurrentStep(3); + }; + + const handlePreCheck = () => { + handleCreateConfig({ name: oceanbase?.appname }, { ...configData }); + }; + + const getComponentsList = () => { + const componentsList: API.TableComponentInfo[] = []; + allComponentsKeys.forEach((key) => { + if (components?.[key]) { + const componentConfig = componentsNameConfig?.[key] || {}; + componentsList.push({ + ...componentConfig, + version: components?.[key].version, + key, + }); + } + }); + return componentsList; + }; + + const getComponentsNodeConfigList = () => { + const componentsNodeConfigList: ComponentsNodeConfig[] = []; + let currentOnlyComponentsKeys = onlyComponentsKeys.filter( + (key) => key !== 'obagent', + ); + if (lowVersion) { + currentOnlyComponentsKeys = currentOnlyComponentsKeys.filter( + (key) => key !== 'ocpexpress', + ); + } + currentOnlyComponentsKeys.forEach((key) => { + if (componentsNameConfig?.[key]) { + componentsNodeConfigList.push({ + key, + name: componentsNameConfig?.[key]?.name, + servers: components?.[key]?.servers?.join(','), + isTooltip: key === obproxyComponent, + }); + } + }); + + return componentsNodeConfigList; + }; + + const dbConfigColumns: ColumnsType = [ + { + title: 'Zone 名称', + dataIndex: 'name', + width: 200, + render: (text) => text || '-', + }, + { + title: 'OB Server 节点', + dataIndex: 'servers', + render: (text) => { + const serversIps = text.map((item: API.OceanbaseServers) => item.ip); + const str = serversIps.join(','); + return ( + +
{str}
+
+ ); + }, + }, + { + title: 'Root Server 节点', + dataIndex: 'rootservice', + width: 200, + render: (text) => text || '-', + }, + ]; + + const getMoreColumns = (label: string) => { + const columns: ColumnsType = [ + { + title: `${label}参数名称`, + dataIndex: 'key', + render: (text) => text, + }, + { + title: '参数值', + dataIndex: 'value', + render: (text, record) => (record.adaptive ? '自适应' : text || '-'), + }, + { + title: '介绍', + dataIndex: 'description', + render: (text) => ( + +
{text}
+
+ ), + }, + ]; + return columns; + }; + + const componentsList = getComponentsList(); + const componentsNodeConfigList = getComponentsNodeConfigList(); + const initDir = `${home_path}/oceanbase/store`; + const clusterConfigInfo = [ + { + key: 'cluster', + group: '集群配置', + content: [ + { label: '配置模式', value: modeConfig[oceanbase?.mode] }, + { + label: 'root@sys 密码', + value: ( + +
{oceanbase?.root_password}
+
+ ), + }, + { + label: '数据目录', + value: ( + +
{oceanbase?.data_dir || initDir}
+
+ ), + }, + { + label: '日志目录', + value: ( + +
{oceanbase?.redo_dir || initDir}
+
+ ), + }, + { label: 'SQL 端口', value: oceanbase?.mysql_port }, + { label: 'RPC 端口', value: oceanbase?.rpc_port }, + ], + more: oceanbase?.parameters?.length + ? [ + { + label: componentsNameConfig['oceanbase'].name, + parameters: oceanbase?.parameters, + }, + ] + : [], + }, + ]; + + if (currentType === 'all') { + const content = [ + { label: 'OBProxy 服务端口', value: obproxy?.listen_port }, + { + label: 'OBProxy Exporter 端口', + value: obproxy?.prometheus_listen_port, + }, + { label: 'OBAgent 管理服务端口', value: obagent?.monagent_http_port }, + { label: 'OBAgent 监控服务端口', value: obagent?.mgragent_http_port }, + ]; + + if (!lowVersion) { + content.push({ label: 'OCPExpress 端口', value: ocpexpress?.port }); + } + + let more: any = []; + if (obproxy?.parameters?.length) { + more = [ + { + label: componentsNameConfig['obproxy'].name, + parameters: obproxy?.parameters, + }, + { + label: componentsNameConfig['obagent'].name, + parameters: obagent?.parameters, + }, + ]; + if (!lowVersion) { + more.push({ + label: componentsNameConfig['ocpexpress'].name, + parameters: ocpexpress?.parameters, + }); + } + } + clusterConfigInfo.push({ + key: 'components', + group: '组件配置', + content, + more, + }); + } + + return ( + + + + + + + + + {oceanbase?.appname} + + + {currentType === 'all' ? '完全部署' : '精简部署'} + + + + + + + {componentsList.map( + (item: API.TableComponentInfo, index: number) => ( + 1 ? { marginTop: 16 } : {}} + key={item.key} + > + + + {item?.showComponentName} + + + {componentsNameConfig[item.key]?.type} + + + {item?.version} + + + + ), + )} + + + + + + + + + + + + {currentType === 'all' ? ( + + + + {componentsNodeConfigList.map( + (item: ComponentsNodeConfig) => ( + + {item.isTooltip ? ( + +
{item?.servers}
+
+ ) : ( + item?.servers + )} +
+ ), + )} +
+ + + ) : null} + + + + {auth?.user} + + {auth?.password ? ( +
+ {showPwd ? ( +
+ +
+ {auth?.password} +
+
+ setShowPwd(false)} + /> +
+ ) : ( +
+ ****** + setShowPwd(true)} + /> +
+ )} +
+ ) : ( + '-' + )} +
+
+ + + + + + + + {home_path} + + + + + + + + + + {clusterConfigInfo?.map((item, index) => ( + + + + {item.content.map((subItem) => ( + + {subItem.value} + + ))} + + + + {item?.more?.length + ? item?.more.map((moreItem) => ( + +
+ + )) + : null} + + + ))} + + +
+
+ + + + + +
+
+ + ); +} diff --git a/web/src/pages/components/ClusterConfig.tsx b/web/src/pages/components/ClusterConfig.tsx new file mode 100644 index 0000000..f0765a0 --- /dev/null +++ b/web/src/pages/components/ClusterConfig.tsx @@ -0,0 +1,758 @@ +import { useState, useEffect } from 'react'; +import { useModel } from 'umi'; +import { + Space, + Button, + Tooltip, + Row, + Switch, + Table, + Spin, + Form, + message, +} from 'antd'; +import { QuestionCircleOutlined } from '@ant-design/icons'; +import { + ProCard, + ProForm, + ProFormText, + ProFormRadio, + ProFormDigit, +} from '@ant-design/pro-components'; +import type { ColumnsType } from 'antd/es/table'; +import { handleQuit } from '@/utils'; +import useRequest from '@/utils/useRequest'; +import { queryComponentParameters } from '@/services/ob-deploy-web/Components'; +import RandExp from 'randexp'; +import Parameter from './Parameter'; +import DirInput from './DirInput'; +import { + commonStyle, + pathRule, + onlyComponentsKeys, + componentsConfig, + componentVersionTypeToComponent, +} from '../constants'; +import styles from './index.less'; + +interface Parameter extends API.Parameter { + description?: string; +} + +interface FormValues extends API.Components { + oceanbase?: { + mode?: string; + parameters?: any; + }; +} + +const showConfigKeys = { + oceanbase: [ + 'home_path', + 'mode', + 'root_password', + 'data_dir', + 'redo_dir', + 'mysql_port', + 'rpc_port', + ], + obproxy: ['home_path', 'listen_port', 'prometheus_listen_port'], + obagent: ['home_path', 'monagent_http_port', 'mgragent_http_port'], + ocpexpress: ['home_path', 'port'], +}; + +export default function ClusterConfig() { + const { + setCurrentStep, + configData, + setConfigData, + currentType, + lowVersion, + clusterMore, + setClusterMore, + componentsMore, + setComponentsMore, + clusterMoreConfig, + setClusterMoreConfig, + componentsMoreConfig, + setComponentsMoreConfig, + handleQuitProgress, + } = useModel('global'); + const { components = {}, home_path } = configData || {}; + const { + oceanbase = {}, + ocpexpress = {}, + obproxy = {}, + obagent = {}, + } = components; + const [form] = ProForm.useForm(); + const [currentMode, setCurrentMode] = useState( + oceanbase?.mode || 'PRODUCTION', + ); + const [passwordVisible, setPasswordVisible] = useState(true); + const [clusterMoreLoading, setClusterMoreLoading] = useState(false); + const [componentsMoreLoading, setComponentsMoreLoading] = useState(false); + const { run: getMoreParamsters } = useRequest(queryComponentParameters); + + const formatParameters = (dataSource: any) => { + if (dataSource) { + const parameterKeys = Object.keys(dataSource); + return parameterKeys.map((key) => { + const { params, ...rest } = dataSource[key]; + return { + key, + ...rest, + ...params, + }; + }); + } else { + return []; + } + }; + + const setData = (dataSource: FormValues) => { + let newComponents: API.Components = { ...components }; + if (currentType === 'all') { + newComponents.obproxy = { + ...(components.obproxy || {}), + ...dataSource.obproxy, + parameters: formatParameters(dataSource.obproxy?.parameters), + }; + if (!lowVersion) { + newComponents.ocpexpress = { + ...(components.ocpexpress || {}), + ...dataSource.ocpexpress, + parameters: formatParameters(dataSource.ocpexpress?.parameters), + }; + } + newComponents.obagent = { + ...(components.obagent || {}), + ...dataSource.obagent, + parameters: formatParameters(dataSource.obagent?.parameters), + }; + } + newComponents.oceanbase = { + ...(components.oceanbase || {}), + ...dataSource.oceanbase, + parameters: formatParameters(dataSource.oceanbase?.parameters), + }; + setConfigData({ ...configData, components: newComponents }); + }; + + const prevStep = () => { + const formValues = form.getFieldsValue(true); + setData(formValues); + setCurrentStep(2); + }; + + const nextStep = () => { + form + .validateFields() + .then((values) => { + setData(values); + setCurrentStep(4); + }) + .catch(({ errorFields }) => { + const errorName = errorFields?.[0].name; + form.scrollToField(errorName); + message.destroy(); + if (errorName.includes('parameters')) { + message.warning('更多配置有必填参数未填入'); + } + }); + }; + + const onValuesChange = (values: FormValues) => { + if (values?.oceanbase?.mode) { + setCurrentMode(values?.oceanbase?.mode); + } + }; + + const portValidator = (_: any, value: number) => { + if (value) { + if (value >= 1024 && value <= 65535) { + return Promise.resolve(); + } + return Promise.reject(new Error('端口号只支持 1024~65535 范围')); + } + }; + + const formatMoreConfig = (dataSource: API.ParameterMeta[]) => { + return dataSource.map((item) => { + const component = componentVersionTypeToComponent[item.component] + ? componentVersionTypeToComponent[item.component] + : item.component; + const componentConfig = componentsConfig[component]; + // filter out existing parameters + let configParameter = item?.config_parameters.filter((parameter) => { + return !showConfigKeys?.[componentConfig.componentKey]?.includes( + parameter.name, + ); + }); + const newConfigParameter: API.NewConfigParameter[] = configParameter.map( + (parameterItem) => { + return { + ...parameterItem, + parameterValue: { + value: parameterItem.default, + adaptive: parameterItem.auto, + auto: parameterItem.auto, + require: parameterItem.require, + }, + }; + }, + ); + const result: API.NewParameterMeta = { + ...item, + componentKey: componentConfig.componentKey, + label: componentConfig.name, + configParameter: newConfigParameter, + }; + return result; + }); + }; + + const getInitialParameters = ( + currentComponent: string, + dataSource: API.MoreParameter[], + data: API.NewParameterMeta[], + ) => { + const currentComponentNameConfig = data?.filter( + (item) => item.component === currentComponent, + )?.[0]; + if (currentComponentNameConfig) { + const parameters: any = {}; + currentComponentNameConfig.configParameter.forEach((item) => { + let parameter = { + ...item, + key: item.name, + params: { + value: item.default, + adaptive: item.auto, + auto: item.auto, + require: item.require, + }, + }; + dataSource?.some((dataItem) => { + if (item.name === dataItem.key) { + parameter = { + key: dataItem.key, + params: { + ...parameter.params, + ...dataItem, + }, + }; + return true; + } + return false; + }); + parameters[item.name] = parameter; + }); + return parameters; + } else { + return undefined; + } + }; + + const getClusterMoreParamsters = async () => { + setClusterMoreLoading(true); + try { + const { success, data } = await getMoreParamsters({ + filters: [ + { + component: oceanbase?.component, + version: oceanbase?.version, + is_essential_only: true, + }, + ], + }); + if (success) { + const newClusterMoreConfig = formatMoreConfig(data?.items); + setClusterMoreConfig(newClusterMoreConfig); + form.setFieldsValue({ + oceanbase: { + parameters: getInitialParameters( + oceanbase?.component, + oceanbase?.parameters, + newClusterMoreConfig, + ), + }, + }); + } + } catch { + setClusterMore(false); + } + setClusterMoreLoading(false); + }; + + const getComponentsMoreParamsters = async () => { + const filters: API.ParameterFilter[] = []; + let currentOnlyComponentsKeys: string[] = onlyComponentsKeys; + if (lowVersion) { + currentOnlyComponentsKeys = onlyComponentsKeys.filter( + (key) => key !== 'ocpexpress', + ); + } + currentOnlyComponentsKeys.forEach((item) => { + if (components[item]) { + filters.push({ + component: components[item]?.component, + version: components[item]?.version, + is_essential_only: true, + }); + } + }); + setComponentsMoreLoading(true); + try { + const { success, data } = await getMoreParamsters({ filters }); + if (success) { + const newComponentsMoreConfig = formatMoreConfig(data?.items); + setComponentsMoreConfig(newComponentsMoreConfig); + const setValues = { + obproxy: { + parameters: getInitialParameters( + obproxy?.component, + obproxy?.parameters, + newComponentsMoreConfig, + ), + }, + obagent: { + parameters: getInitialParameters( + obagent?.component, + obagent?.parameters, + newComponentsMoreConfig, + ), + }, + }; + if (!lowVersion) { + setValues.ocpexpress = { + parameters: getInitialParameters( + ocpexpress?.component, + ocpexpress?.parameters, + newComponentsMoreConfig, + ), + }; + } + form.setFieldsValue(setValues); + } + } catch { + setComponentsMore(false); + } + + setComponentsMoreLoading(false); + }; + + const handleCluserMoreChange = (checked: boolean) => { + setClusterMore(checked); + if (!clusterMoreConfig?.length) { + getClusterMoreParamsters(); + } + }; + + const handleComponentsMoreChange = (checked: boolean) => { + setComponentsMore(checked); + if (!componentsMoreConfig?.length) { + getComponentsMoreParamsters(); + } + }; + + const parameterValidator = (_: any, value?: API.ParameterValue) => { + if (value?.adaptive) { + return Promise.resolve(); + } else if (value?.require && !value?.value) { + return Promise.reject(new Error('自定义参数时必填')); + } + return Promise.resolve(); + }; + + const getMoreColumns = (label: string, componentKey: string) => { + const columns: ColumnsType = [ + { + title: `${label}参数名称`, + dataIndex: 'name', + width: 250, + render: (text) => text || '-', + }, + { + title: '参数值', + dataIndex: 'parameterValue', + render: (text, record) => { + return ( + + + + ); + }, + }, + { + title: '介绍', + dataIndex: 'description', + width: 500, + render: (text, record) => + text ? ( + + +
{text}
+
+
+ ) : ( + '-' + ), + }, + ]; + return columns; + }; + + const getTableConfig = ( + showVisible: boolean, + dataSource: API.NewParameterMeta[], + loading: boolean, + ) => { + return showVisible ? ( + + + {dataSource.map((moreItem) => ( + +
+ + ))} + + + ) : null; + }; + + const getRandomPassword = () => { + const randomPasswordReg = + /^(?=(.*[a-z]){2,})(?=(.*[A-Z]){2,})(?=(.*\d){2,})(?=(.*[~!@#%^&*_\-+=|(){}\[\]:;,.?/]){2,})[A-Za-z\d~!@#%^&*_\-+=|(){}\[\]:;,.?/]{8,32}$/; + const newValue = new RandExp(randomPasswordReg).gen(); + if (randomPasswordReg.test(newValue)) { + return newValue; + } + return getRandomPassword(); + }; + + useEffect(() => { + if (clusterMore && !clusterMoreConfig?.length) { + getClusterMoreParamsters(); + } + if (componentsMore && !componentsMoreConfig?.length) { + getComponentsMoreParamsters(); + } + }, []); + + const initPassword = getRandomPassword(); + + const initialValues = { + oceanbase: { + mode: oceanbase?.mode || 'PRODUCTION', + root_password: oceanbase?.root_password || initPassword, + data_dir: oceanbase?.data_dir || undefined, + redo_dir: oceanbase?.redo_dir || undefined, + mysql_port: oceanbase?.mysql_port || 2881, + rpc_port: oceanbase?.rpc_port || 2882, + parameters: getInitialParameters( + oceanbase?.component, + oceanbase?.parameters, + clusterMoreConfig, + ), + }, + obproxy: { + listen_port: obproxy?.listen_port || 2883, + prometheus_listen_port: obproxy?.prometheus_listen_port || 2884, + parameters: getInitialParameters( + obproxy?.component, + obproxy?.parameters, + componentsMoreConfig, + ), + }, + obagent: { + monagent_http_port: obagent?.monagent_http_port || 8088, + mgragent_http_port: obagent?.mgragent_http_port || 8089, + parameters: getInitialParameters( + obagent?.component, + obagent?.parameters, + componentsMoreConfig, + ), + }, + }; + + if (!lowVersion) { + initialValues.ocpexpress = { + port: ocpexpress?.port || 8180, + parameters: getInitialParameters( + ocpexpress?.component, + ocpexpress?.parameters, + componentsMoreConfig, + ), + }; + } + + const singleItemStyle = { width: 448 }; + const initDir = `${home_path}/oceanbase/store`; + + return ( + + + + + +
+
+ {currentMode === 'PRODUCTION' + ? '此模式将最大化利用环境资源,保证集群的性能与稳定性,推荐使用此模式。' + : '配置满足集群正常运行的资源参数'} +
+
+ + + + + + + + + + + + + + + + + +
+ 更多配置 + +
+ {clusterMore + ? getTableConfig( + clusterMore, + clusterMoreConfig, + clusterMoreLoading, + ) + : null} +
+
+ {currentType === 'all' ? ( + + + + + + + OBProxy Exporter 端口 + + + + + } + fieldProps={{ style: commonStyle }} + placeholder="请输入" + rules={[ + { required: true, message: '请输入' }, + { validator: portValidator }, + ]} + /> + + + + + + + + + {!lowVersion ? ( + + + + ) : null} +
+ 更多配置 + +
+ {componentsMore + ? getTableConfig( + componentsMore, + componentsMoreConfig, + componentsMoreLoading, + ) + : null} +
+
+ ) : null} +
+
+ + + + + + + +
+
+
+
+ ); +} diff --git a/web/src/pages/components/DeleteDeployModal.tsx b/web/src/pages/components/DeleteDeployModal.tsx new file mode 100644 index 0000000..3b9e3ec --- /dev/null +++ b/web/src/pages/components/DeleteDeployModal.tsx @@ -0,0 +1,198 @@ +import { useEffect, useState } from 'react'; +import { useModel } from 'umi'; +import { Modal, Progress, message } from 'antd'; +import { getDestroyTaskInfo } from '@/services/ob-deploy-web/Deployments'; +import useRequest from '@/utils/useRequest'; +import { checkLowVersion, handleResponseError } from '@/utils'; +import NP from 'number-precision'; +import { oceanbaseComponent } from '../constants'; +import styles from './index.less'; + +interface Props { + visible: boolean; + name: string; + onCancel: () => void; + setOBVersionValue: (value: string) => void; +} + +let timerProgress: NodeJS.Timer; +let timerFetch: NodeJS.Timer; + +const statusConfig = { + RUNNING: 'normal', + SUCCESSFUL: 'success', + FAILED: 'exception', +}; + +export default function DeleteDeployModal({ + visible, + name, + onCancel, + setOBVersionValue, +}: Props) { + const { + setConfigData, + setIsDraft, + setClusterMore, + setComponentsMore, + componentsVersionInfo, + setComponentsVersionInfo, + setCurrentType, + getInfoByName, + setLowVersion, + } = useModel('global'); + + const [status, setStatus] = useState('RUNNING'); + const [progress, setProgress] = useState(0); + const [showProgress, setShowProgress] = useState(0); + const [isFinished, setIsFinished] = useState(false); + + const { run: fetchDestroyTaskInfo } = useRequest(getDestroyTaskInfo, { + onSuccess: async ({ success, data }: API.OBResponseTaskInfo_) => { + if (success) { + if (data?.status !== 'RUNNING') { + clearInterval(timerFetch); + } + clearInterval(timerProgress); + if (data?.status === 'RUNNING') { + const newProgress = NP.times( + NP.divide(data?.finished, data?.total), + 100, + ); + setProgress(newProgress); + let step = NP.minus(newProgress, progress); + let stepNum = 1; + timerProgress = setInterval(() => { + setShowProgress( + NP.plus(progress, NP.times(NP.divide(step, 100), stepNum)), + ); + stepNum += 1; + }, 10); + } else if (data?.status === 'SUCCESSFUL') { + let step = NP.minus(100, progress); + let stepNum = 1; + timerProgress = setInterval(() => { + setShowProgress( + NP.plus(progress, NP.times(NP.divide(step, 100), stepNum)), + ); + stepNum += 1; + }, 10); + try { + const { success: nameSuccess, data: nameData } = + await getInfoByName({ name }); + if (nameSuccess) { + const { config } = nameData; + const { components = {} } = config; + setConfigData(config || {}); + setLowVersion(checkLowVersion(components?.oceanbase?.version)); + setClusterMore(!!components?.oceanbase?.parameters?.length); + setComponentsMore(!!components?.obproxy?.parameters?.length); + setIsDraft(true); + setCurrentType( + components?.oceanbase && !components?.obproxy ? 'ob' : 'all', + ); + const newSelectedVersionInfo = componentsVersionInfo?.[ + oceanbaseComponent + ]?.dataSource?.filter( + (item: any) => item.md5 === components?.oceanbase?.package_hash, + )[0]; + if (newSelectedVersionInfo) { + setOBVersionValue( + `${components?.oceanbase?.version}-${components?.oceanbase?.release}-${components?.oceanbase?.package_hash}`, + ); + setComponentsVersionInfo({ + ...componentsVersionInfo, + [oceanbaseComponent]: { + ...componentsVersionInfo[oceanbaseComponent], + ...newSelectedVersionInfo, + }, + }); + } + setTimeout(() => { + onCancel(); + }, 2000); + } else { + setIsDraft(false); + message.error('获取配置信息失败'); + onCancel(); + } + } catch (e: any) { + const { response, data } = e; + handleResponseError( + data?.msg || data?.detail || response?.statusText, + ); + } + } else { + message.error(data?.msg); + onCancel(); + } + setStatus(data?.status); + } + }, + }); + + useEffect(() => { + fetchDestroyTaskInfo({ name }); + timerFetch = setInterval(() => { + fetchDestroyTaskInfo({ name }); + }, 1000); + return () => { + clearInterval(timerProgress); + clearInterval(timerFetch); + }; + }, []); + + useEffect(() => { + if (status !== 'RUNNING') { + setTimeout(() => { + clearInterval(timerProgress); + setIsFinished(true); + }, 1000); + } + }, [status]); + + return ( + +
+ {isFinished ? ( + <> +
+ {status === 'SUCCESSFUL' + ? '清理失败历史部署环境成功' + : '清理失败历史部署环境失败'} +
+ + + ) : ( + <> +
+ 正在清理失败的历史部署环境 +
请耐心等待
+
+ + + )} +
+
+ ); +} diff --git a/web/src/pages/components/DeployType.tsx b/web/src/pages/components/DeployType.tsx new file mode 100644 index 0000000..c467589 --- /dev/null +++ b/web/src/pages/components/DeployType.tsx @@ -0,0 +1,64 @@ +import { useEffect, useState } from 'react'; +import { Space, Card, Tag } from 'antd'; +import styles from './index.less'; + +interface Props { + value?: string; + onChange?: (value: string) => void; +} + +const optionConfig = [ + { + label: ( + <> + 完全部署推荐 + + ), + value: 'all', + desc: '配置数据库集群及相关生态工具,提供全套数据库运维管理服务', + }, + { + label: '精简部署', + value: 'ob', + desc: '只配置数据库集群,以最精简的数据库内核能力提供服务', + }, +]; + +export default function DeployType({ value, onChange }: Props) { + const [selectValue, setSelectValue] = useState(value || 'all'); + + useEffect(() => { + if (value && value !== selectValue) { + setSelectValue(value); + } + }, [value]); + + useEffect(() => { + if (onChange) { + onChange(selectValue); + } + }, [selectValue]); + return ( + + {optionConfig.map((item) => ( +
+ setSelectValue(item.value)} + > + {item.label} + + + {item.desc} + +
+ ))} +
+ ); +} diff --git a/web/src/pages/components/DirInput.tsx b/web/src/pages/components/DirInput.tsx new file mode 100644 index 0000000..7669bde --- /dev/null +++ b/web/src/pages/components/DirInput.tsx @@ -0,0 +1,116 @@ +import { useEffect, useState, useRef } from 'react'; +import { Input, Tooltip } from 'antd'; + +interface Props { + value?: string; + onChange?: (value?: string) => void; + placeholder: string; + name: string; +} + +export default ({ value, onChange, placeholder, name }: Props) => { + const [visible, setVisible] = useState(false); + const [currentValue, setCurrentValue] = useState(value); + const open = useRef(); + open.current = { + input: false, + tooltip: false, + }; + + const onMouseEnterInput = () => { + open.current = { + ...(open?.current || {}), + input: true, + }; + setVisible(true); + }; + + const onMouseEnterTooltip = () => { + open.current = { + ...(open?.current || {}), + tooltip: true, + }; + setVisible(true); + }; + + const onMouseLeaveInput = () => { + setTimeout(() => { + if (!open?.current?.tooltip) { + setVisible(false); + } + }, 300); + }; + + const onMouseLeaveTooltip = () => { + setVisible(false); + }; + + const addEventTooltipOverlay = () => { + const tooltipOverlay = document.querySelector( + `.dir-input-tooltip-overlay-${name}`, + ); + if (tooltipOverlay) { + tooltipOverlay?.addEventListener('mouseenter', onMouseEnterTooltip); + tooltipOverlay?.addEventListener('mouseleave', onMouseLeaveTooltip); + } else { + setTimeout(() => { + addEventTooltipOverlay(); + }, 500); + } + }; + + const addEventInputConatiner = () => { + const inputConatiner = document.querySelector(`.dir-input-${name}`); + if (inputConatiner) { + inputConatiner?.addEventListener('mouseenter', onMouseEnterInput); + inputConatiner?.addEventListener('mouseleave', onMouseLeaveInput); + } else { + setTimeout(() => { + addEventInputConatiner(); + }, 500); + } + }; + + useEffect(() => { + const tooltipOverlay = document.querySelector( + `.dir-input-tooltip-overlay-${name}`, + ); + const inputConatiner = document.querySelector(`.dir-input-${name}`); + addEventTooltipOverlay(); + addEventInputConatiner(); + return () => { + tooltipOverlay?.removeEventListener('mouseenter', onMouseEnterTooltip); + tooltipOverlay?.removeEventListener('mouseleave', onMouseLeaveTooltip); + inputConatiner?.removeEventListener('mouseenter', onMouseEnterInput); + inputConatiner?.removeEventListener('mouseleave', onMouseLeaveInput); + }; + }, []); + + useEffect(() => { + if (onChange) { + onChange(currentValue); + } + }, [currentValue]); + + return ( + 48 && visible} + title={placeholder} + overlayClassName={`dir-input-tooltip-overlay-${name}`} + > + { + setCurrentValue(e?.target?.value); + setVisible(false); + }} + autoComplete="off" + style={{ width: 448 }} + onFocus={() => setVisible(false)} + /> + + ); +}; diff --git a/web/src/pages/components/ExitPage.tsx b/web/src/pages/components/ExitPage.tsx new file mode 100644 index 0000000..b70dcb4 --- /dev/null +++ b/web/src/pages/components/ExitPage.tsx @@ -0,0 +1,38 @@ +import { message, Card } from 'antd'; +import { CopyOutlined } from '@ant-design/icons'; +import copy from 'copy-to-clipboard'; +import styles from './index.less'; + +export default function ExitPage() { + const command = 'obd web'; + + const handleCopy = () => { + copy(command); + message.success('复制成功'); + }; + + return ( + +

+ 部署程序已经退出! +

+
+ 如需再次启动,请前往中控服务器执行 + + {command} + +
+
+ ); +} diff --git a/web/src/pages/components/InstallConfig.tsx b/web/src/pages/components/InstallConfig.tsx new file mode 100644 index 0000000..fbb2560 --- /dev/null +++ b/web/src/pages/components/InstallConfig.tsx @@ -0,0 +1,811 @@ +import { useEffect, useState, useRef } from 'react'; +import { useModel } from 'umi'; +import { + Space, + Button, + Form, + Tag, + Table, + Alert, + Tooltip, + Select, + Modal, + Spin, + message, +} from 'antd'; +import { ProCard, ProForm, ProFormText } from '@ant-design/pro-components'; +import { + CloseOutlined, + SafetyCertificateFilled, + InfoOutlined, + InfoCircleOutlined, + CopyOutlined, +} from '@ant-design/icons'; +import type { ColumnsType } from 'antd/es/table'; +import useRequest from '@/utils/useRequest'; +import { queryAllComponentVersions } from '@/services/ob-deploy-web/Components'; +import { + queryDeploymentInfoByTaskStatusType, + deleteDeployment, +} from '@/services/ob-deploy-web/Deployments'; +import { listRemoteMirrors } from '@/services/ob-deploy-web/Mirror'; +import { handleQuit, handleResponseError, checkLowVersion } from '@/utils'; +import NP from 'number-precision'; +import copy from 'copy-to-clipboard'; +import DeployType from './DeployType'; +import DeleteDeployModal from './DeleteDeployModal'; +import { + commonStyle, + allComponentsName, + oceanbaseComponent, + obproxyComponent, + ocpexpressComponent, + obagentComponent, +} from '../constants'; +import styles from './index.less'; + +interface FormValues { + type?: string; +} + +const appnameReg = /^[a-zA-Z]([a-zA-Z0-9]{0,19})$/; + +const componentsGroupInfo = [ + { + group: '数据库', + key: 'database', + content: [ + { + key: oceanbaseComponent, + name: 'OceanBase Database', + onlyAll: false, + desc: '是金融级分布式数据库,具备数据强一致、高扩展、高可用、高性价比、稳定可靠等特征。', + doc: 'https://www.oceanbase.com/docs/oceanbase-database-cn', + }, + ], + }, + { + group: '代理', + key: 'agency', + onlyAll: true, + content: [ + { + key: obproxyComponent, + name: 'OBProxy', + onlyAll: true, + desc: '是 OceanBase 数据库专用的代理服务器,可以将用户 SQL 请求转发至最佳目标 OBServer 。', + doc: 'https://www.oceanbase.com/docs/odp-enterprise-cn', + }, + ], + }, + { + group: '工具', + key: 'tool', + onlyAll: true, + content: [ + { + key: ocpexpressComponent, + name: 'OCPExpress', + onlyAll: true, + desc: '是专为 OceanBase 设计的管控平台,可实现对集群、租户的监控管理、诊断等核心能力。', + doc: 'https://www.oceanbase.com/docs/common-oceanbase-database-cn-0000000001626262', + }, + { + key: obagentComponent, + name: 'OBAgent', + onlyAll: true, + desc: '是一个监控采集框架。OBAgent 支持推、拉两种数据采集模式,可以满足不同的应用场景。', + doc: 'https://www.oceanbase.com/docs/common-oceanbase-database-cn-10000000001576872', + }, + ], + }, +]; + +const mirrors = ['oceanbase.community.stable', 'oceanbase.development-kit']; + +export default function InstallConfig() { + const { + initAppName, + setCurrentStep, + configData, + setConfigData, + currentType, + setCurrentType, + lowVersion, + isFirstTime, + setIsFirstTime, + isDraft, + setIsDraft, + componentsVersionInfo, + setComponentsVersionInfo, + handleQuitProgress, + getInfoByName, + setLowVersion, + } = useModel('global'); + const { components, home_path } = configData || {}; + const { oceanbase } = components || {}; + const [existNoVersion, setExistNoVersion] = useState(false); + const [obVersionValue, setOBVersionValue] = useState( + undefined, + ); + const [hasDraft, setHasDraft] = useState(false); + const [deleteLoadingVisible, setDeleteLoadingVisible] = useState(false); + const [deleteName, setDeleteName] = useState(''); + const [installMemory, setInstallMemory] = useState(0); + const [form] = ProForm.useForm(); + const [unavailableList, setUnavailableList] = useState([]); + const [componentLoading, setComponentLoading] = useState(false); + const draftNameRef = useRef(); + + const { run: fetchDeploymentInfo, loading } = useRequest( + queryDeploymentInfoByTaskStatusType, + ); + const { run: handleDeleteDeployment } = useRequest(deleteDeployment); + const { run: fetchListRemoteMirrors } = useRequest(listRemoteMirrors, { + skipStatusError: true, + onSuccess: () => { + setComponentLoading(false); + }, + onError: ({ response, data }: any) => { + if (response?.status === 503) { + setTimeout(() => { + fetchListRemoteMirrors(); + }, 1000); + } else { + if (response) { + const errorInfo = data?.msg || data?.detail || response?.statusText; + handleResponseError(errorInfo); + } + setComponentLoading(false); + } + }, + }); + + const judgVersions = (type: string, source: API.ComponentsVersionInfo) => { + if (type === 'all') { + if (Object.keys(source).length !== allComponentsName.length) { + setExistNoVersion(true); + } else { + setExistNoVersion(false); + } + } else { + if ( + !(source?.[oceanbaseComponent] && source?.[oceanbaseComponent]?.version) + ) { + setExistNoVersion(true); + } else { + setExistNoVersion(false); + } + } + }; + + const { run: fetchAllComponentVersions, loading: versionLoading } = + useRequest(queryAllComponentVersions, { + skipStatusError: true, + onSuccess: async ({ + success, + data, + }: API.OBResponseDataListComponent_) => { + if (success) { + const newComponentsVersionInfo = {}; + data?.items?.forEach((item) => { + if (allComponentsName.includes(item.name)) { + if (item?.info?.length) { + const initVersionInfo = item?.info[0] || {}; + if (item.name === oceanbaseComponent) { + const newSelectedVersionInfo = item.info.filter( + (item) => item.md5 === oceanbase?.package_hash, + )?.[0]; + const currentSelectedVersionInfo = + newSelectedVersionInfo || initVersionInfo; + setOBVersionValue( + `${currentSelectedVersionInfo?.version}-${currentSelectedVersionInfo?.release}-${currentSelectedVersionInfo?.md5}`, + ); + newComponentsVersionInfo[item.name] = { + ...currentSelectedVersionInfo, + dataSource: item.info || [], + }; + } else { + newComponentsVersionInfo[item.name] = { + ...initVersionInfo, + dataSource: item.info || [], + }; + } + } + } + }); + + const noVersion = + Object.keys(newComponentsVersionInfo).length !== + allComponentsName.length; + judgVersions(currentType, newComponentsVersionInfo); + setComponentsVersionInfo(newComponentsVersionInfo); + + if (noVersion) { + const { success: mirrorSuccess, data: mirrorData } = + await fetchListRemoteMirrors(); + if (mirrorSuccess) { + const nameList: string[] = []; + if (mirrorData?.total < 2) { + const mirrorName = mirrorData?.items?.map( + (item: API.Mirror) => item.section_name, + ); + const noDataName = [...mirrorName, ...mirrors].filter( + (name) => + mirrors.includes(name) && !mirrorName.includes(name), + ); + noDataName.forEach((name) => { + nameList.push(name); + }); + } + if (mirrorData?.total) { + mirrorData?.items?.forEach((item: API.Mirror) => { + if (!item.available) { + nameList.push(item.section_name); + } + }); + } + setUnavailableList(nameList); + } + } else { + setComponentLoading(false); + } + } + }, + onError: ({ response, data }: any) => { + if (response?.status === 503) { + setTimeout(() => { + fetchAllComponentVersions(); + }, 1000); + } else { + if (response) { + const errorInfo = data?.msg || data?.detail || response?.statusText; + handleResponseError(errorInfo); + } + setComponentLoading(false); + } + }, + }); + + const onValuesChange = (values: FormValues) => { + if (values?.type) { + setCurrentType(values?.type); + judgVersions(values?.type, componentsVersionInfo); + } + }; + + const nameValidator = async (_: any, value: string) => { + if (value) { + if (hasDraft || isDraft) { + return Promise.resolve(); + } + if (!appnameReg.test(value)) { + return Promise.reject( + new Error('首字母英文且仅支持英文、数字,长度不超过20'), + ); + } + try { + const { success, data } = await getInfoByName({ name: value }); + if (success) { + if (['CONFIGURED', 'DESTROYED'].includes(data?.status)) { + return Promise.resolve(); + } + return Promise.reject( + new Error(`已存在为 ${value} 的部署名称,请指定新名称`), + ); + } + return Promise.resolve(); + } catch (e: any) { + const { response, data } = e; + if (response?.status === 404) { + return Promise.resolve(); + } else { + handleResponseError( + data?.msg || data?.detail || response?.statusText, + ); + } + } + } + }; + + const nextStep = () => { + form.validateFields().then((values) => { + const lastAppName = oceanbase?.appname || initAppName; + let newHomePath = home_path; + if (values?.appname !== lastAppName && home_path) { + const firstHalfHomePath = home_path.split(`/${lastAppName}`)[0]; + newHomePath = `${firstHalfHomePath}/${values?.appname}`; + } + let newComponents: API.Components = { + oceanbase: { + ...(components?.oceanbase || {}), + component: + componentsVersionInfo?.[oceanbaseComponent]?.version_type === 'ce' + ? 'oceanbase-ce' + : 'oceanbase', + appname: values?.appname, + version: componentsVersionInfo?.[oceanbaseComponent]?.version, + release: componentsVersionInfo?.[oceanbaseComponent]?.release, + package_hash: componentsVersionInfo?.[oceanbaseComponent]?.md5, + }, + }; + if (currentType === 'all') { + newComponents.obproxy = { + ...(components?.obproxy || {}), + component: + componentsVersionInfo?.[obproxyComponent]?.version_type === 'ce' + ? 'obproxy-ce' + : 'obproxy', + version: componentsVersionInfo?.[obproxyComponent]?.version, + release: componentsVersionInfo?.[obproxyComponent]?.release, + package_hash: componentsVersionInfo?.[obproxyComponent]?.md5, + }; + if (!lowVersion) { + newComponents.ocpexpress = { + ...(components?.ocpexpress || {}), + component: ocpexpressComponent, + version: componentsVersionInfo?.[ocpexpressComponent]?.version, + release: componentsVersionInfo?.[ocpexpressComponent]?.release, + package_hash: componentsVersionInfo?.[ocpexpressComponent]?.md5, + }; + } + newComponents.obagent = { + ...(components?.obagent || {}), + component: obagentComponent, + version: componentsVersionInfo?.[obagentComponent]?.version, + release: componentsVersionInfo?.[obagentComponent]?.release, + package_hash: componentsVersionInfo?.[obagentComponent]?.md5, + }; + } + setConfigData({ + ...configData, + components: newComponents, + home_path: newHomePath, + }); + setCurrentStep(2); + setIsFirstTime(false); + }); + }; + + const onVersionChange = ( + value: string, + dataSource: API.service_model_components_ComponentInfo[], + ) => { + const md5 = value.split('-')[2]; + setOBVersionValue(value); + const newSelectedVersionInfo = dataSource.filter( + (item) => item.md5 === md5, + )[0]; + setComponentsVersionInfo({ + ...componentsVersionInfo, + [oceanbaseComponent]: { + ...componentsVersionInfo[oceanbaseComponent], + ...newSelectedVersionInfo, + }, + }); + setLowVersion( + !!( + newSelectedVersionInfo.version && + checkLowVersion(newSelectedVersionInfo.version.split('')[0]) + ), + ); + }; + + const directTo = (url: string) => { + // 在新的标签页中打开 + const blankWindow = window.open('about:blank'); + if (blankWindow) { + blankWindow.location.href = url; + } else { + // 兜底逻辑,在当前标签页打开 + window.location.href = url; + } + }; + + const getColumns = (group: string) => { + const columns: ColumnsType = [ + { + title: group, + dataIndex: 'name', + width: 195, + render: (text, record) => { + if (currentType === 'all') { + return ( + <> + {text} + {record.key === ocpexpressComponent && lowVersion ? ( + + + + + + ) : !componentsVersionInfo[record.key]?.version ? ( + + + + + + ) : null} + + ); + } + return text; + }, + }, + { + title: '版本', + dataIndex: 'version', + width: 130, + render: (_, record) => { + const versionInfo = componentsVersionInfo[record.key] || {}; + if (record?.key === oceanbaseComponent) { + return ( + + ); + } else { + return versionInfo?.version ? ( + <> + {versionInfo?.version} + 最新 + + ) : ( + '-' + ); + } + }, + }, + { + title: '描述', + dataIndex: 'desc', + render: (text, record) => { + let disabled = false; + if ( + (record.key === ocpexpressComponent && lowVersion) || + (currentType === 'ob' && record.onlyAll) + ) { + disabled = true; + } + return ( + <> + {text || '-'} + { + if (!disabled) directTo(record.doc); + }} + target="_blank" + > + 了解更多 + + + ); + }, + }, + ]; + return columns; + }; + + const handleCopy = (content: string) => { + copy(content); + message.success('复制成功'); + }; + + useEffect(() => { + setComponentLoading(true); + if (isFirstTime) { + fetchAllComponentVersions(); + fetchDeploymentInfo({ task_status: 'DRAFT' }).then( + ({ success: draftSuccess, data: draftData }: API.OBResponse) => { + if (draftSuccess && draftData?.items?.length) { + const defaultValue = draftData?.items[0]?.name; + draftNameRef.current = defaultValue; + setHasDraft(true); + Modal.confirm({ + title: '检测到系统中存在以下部署失败的历史配置', + okText: '继续部署', + cancelText: '忽略', + closable: true, + width: 424, + content: ( + +
+ 继续部署将先清理失败的历史部署环境,是否继续历史部署流程? +
+ +
+ ), + onOk: () => { + return new Promise(async (resolve) => { + try { + const { success: deleteSuccess } = + await handleDeleteDeployment({ + name: draftNameRef.current, + }); + if (deleteSuccess) { + resolve(); + setDeleteName(draftNameRef.current); + setDeleteLoadingVisible(true); + } + } catch { + setIsDraft(false); + resolve(); + } + }); + }, + onCancel: () => { + setIsDraft(false); + setHasDraft(false); + }, + }); + } else { + setIsDraft(false); + } + }, + ); + } else { + fetchAllComponentVersions(); + } + }, []); + + useEffect(() => { + let newInstallMemory = 0; + if (currentType === 'ob') { + newInstallMemory = + componentsVersionInfo?.[oceanbaseComponent]?.estimated_size; + } else { + const keys = Object.keys(componentsVersionInfo); + keys.forEach((key) => { + newInstallMemory = + newInstallMemory + componentsVersionInfo[key]?.estimated_size; + }); + } + setInstallMemory(newInstallMemory); + }, [componentsVersionInfo, currentType]); + + useEffect(() => { + form.setFieldsValue({ type: currentType }); + }, [currentType]); + + useEffect(() => { + form.setFieldsValue({ + appname: configData?.components?.oceanbase?.appname || initAppName, + }); + }, [configData]); + + return ( + + + + + + + + + + + + + 部署组件 + + 预计安装需要{' '} + {NP.divide(NP.divide(installMemory, 1024), 1024).toFixed(2)}{' '} + MB 空间 + + + } + className="card-header-padding-top-0 card-padding-bottom-24 card-padding-top-0" + > + + {existNoVersion ? ( + unavailableList?.length ? ( + + 如当前环境无法正常访问外网,建议使用 OceanBase + 离线安装包进行安装部署。 + + 前往下载离线安装 + + + } + type="error" + showIcon + style={{ marginTop: '16px' }} + /> + ) : ( + + 如当前环境可正常访问外网,可启动 OceanBase + 在线镜像仓库,或联系您的镜像仓库管理员。 + + 请在主机上执行一下命令启用在线镜像仓库 +
obd mirror enable + oceanbase.community.stable + oceanbase.development-kit + + + handleCopy( + 'obd mirror enable oceanbase.community.stable oceanbase.development-kit', + ) + } + /> + + + } + > + 如何启用在线镜像仓库 +
+ + } + type="error" + showIcon + style={{ marginTop: '16px' }} + /> + ) + ) : null} + + {componentsGroupInfo.map((info) => ( + +
{ + if ( + (record.key === ocpexpressComponent && lowVersion) || + (currentType === 'ob' && record?.onlyAll) + ) { + return styles.disabledRow; + } + }} + /> + + ))} + + + + +
+
+ + + + +
+
+ {deleteLoadingVisible && ( + setDeleteLoadingVisible(false)} + setOBVersionValue={setOBVersionValue} + /> + )} + + + ); +} diff --git a/web/src/pages/components/InstallFinished.tsx b/web/src/pages/components/InstallFinished.tsx new file mode 100644 index 0000000..0a9984e --- /dev/null +++ b/web/src/pages/components/InstallFinished.tsx @@ -0,0 +1,379 @@ +import { useEffect } from 'react'; +import { useModel } from 'umi'; +import { + Space, + Button, + Table, + Alert, + Result, + Tooltip, + message, + Tag, + Modal, + Typography, +} from 'antd'; +import { + CloseCircleFilled, + CheckCircleFilled, + CaretRightFilled, + CaretDownFilled, + CopyOutlined, + ExclamationCircleOutlined, + CheckOutlined, +} from '@ant-design/icons'; +import { ProCard } from '@ant-design/pro-components'; +import useRequest from '@/utils/useRequest'; +import type { ColumnsType } from 'antd/es/table'; +import copy from 'copy-to-clipboard'; +import { + queryDeploymentReport, + queryConnectionInfo, +} from '@/services/ob-deploy-web/Deployments'; +import { + componentsConfig, + componentVersionTypeToComponent, +} from '../constants'; +import { handleQuit } from '@/utils'; +import styles from './index.less'; + +const { Paragraph } = Typography; + +export default function InstallProcess() { + const { configData, installStatus, setCurrentStep, handleQuitProgress } = + useModel('global'); + + const name = configData?.components?.oceanbase?.appname; + const { run: fetchReportInfo, data: reportInfo } = useRequest( + queryDeploymentReport, + ); + const { run: fetchConnectInfo, data: connectInfo } = + useRequest(queryConnectionInfo); + + const handleCopy = (content: string) => { + copy(content); + message.success('复制成功'); + }; + + const handleCopyCommand = (command: string) => { + copy(command); + message.success('复制成功'); + }; + + useEffect(() => { + fetchReportInfo({ name }); + fetchConnectInfo({ name }); + }, []); + + const connectColumns: ColumnsType = [ + { + title: '组件', + dataIndex: 'component', + width: 200, + render: (text) => { + const component = componentVersionTypeToComponent[text] + ? componentVersionTypeToComponent[text] + : text; + return componentsConfig[component]?.showComponentName || '-'; + }, + }, + { + title: '访问地址', + dataIndex: 'access_url', + width: 160, + render: (text) => text || '-', + }, + { + title: '账号', + dataIndex: 'user', + render: (text) => text || '-', + }, + { + title: '密码', + dataIndex: 'password', + width: 160, + render: (text) => + text ? ( + +
{text}
+
+ ) : ( + '-' + ), + }, + { + title: '连接串', + dataIndex: 'connect_url', + width: 300, + render: (text, record) => { + let content; + if (/^http/g.test(text)) { + content = ( + + {text} + + ); + } else { + content = ( +
+ {text} +
+ ); + } + return ( +
+ +
+ {content} +
+
+ + handleCopy(text)} /> + +
+ ); + }, + }, + ]; + + const reportColumns: ColumnsType = [ + { + title: '组件名称', + dataIndex: 'name', + render: (text) => { + const component = componentVersionTypeToComponent[text] + ? componentVersionTypeToComponent[text] + : text; + return componentsConfig[component]?.showComponentName || '-'; + }, + }, + { + title: '组件类型', + dataIndex: 'type', + render: (_, record) => { + const component = componentVersionTypeToComponent[record.name] + ? componentVersionTypeToComponent[record.name] + : record.name; + return componentsConfig[component]?.type || '-'; + }, + }, + { + title: '版本', + dataIndex: 'version', + render: (text) => text || '-', + }, + { + title: '安装结果', + dataIndex: 'status', + width: 150, + render: (text) => + text === 'SUCCESSFUL' ? ( + <> + + 成功 + + ) : ( + <> + + 失败 + + ), + }, + ]; + + const getEpendedColumns = (component: string) => { + const expendedColumns: ColumnsType<{ ip: string }> = [ + { + title: '节点', + dataIndex: 'ip', + render: (text) => text || '-', + }, + { + title: '日志', + dataIndex: 'log', + width: 200, + render: (_, record) => { + const command = `obd tool command ${name} log -c ${component} -s ${record.ip}`; + return ( + + 请前往 OBD 中控机执行以下命令查看日志: +
+ {command}
+ handleCopyCommand(command)}>复制信息 + + } + overlayStyle={{ width: 300 }} + > + 查看日志 +
+ ); + }, + }, + ]; + return expendedColumns; + }; + + const expandedRowRender = (record: API.DeploymentReport) => { + const serversData = record?.servers?.map((server) => ({ ip: server })); + return ( +
+ ); + }; + + const handleFinished = () => { + Modal.confirm({ + title: '是否要退出页面?', + okText: '退出', + cancelText: '取消', + okButtonProps: { type: 'primary', danger: true }, + content: ( +
+ 退出前,请确保已复制访问地址及账密信息 +
+ + + 复制信息 + , + <> + + 复制信息 + , + ], + onCopy: () => handleCopy(JSON.stringify(connectInfo?.items)), + }} + /> +
+ ), + icon: , + onOk: () => { + handleQuit(handleQuitProgress, setCurrentStep, true); + }, + }); + }; + + return ( + + + } + title={ + installStatus === 'SUCCESSFUL' ? ( +
+ OceanBase 部署成功 +
+ ) : ( +
+ OceanBase 部署失败 +
+ ) + } + /> + {connectInfo?.items?.length ? ( + + handleCopy(JSON.stringify(connectInfo?.items))} + data-aspm-click="c307514.d317299" + data-aspm-desc="部署结果-复制信息" + data-aspm-param={``} + data-aspm-expo + > + 复制信息 + + } + /> +
+ + ) : null} + + collapsed ? : + } + bodyStyle={{ paddingLeft: '0px', paddingRight: '0px' }} + > +
+ +
+
+ + + +
+
+ + ); +} diff --git a/web/src/pages/components/InstallProcess.tsx b/web/src/pages/components/InstallProcess.tsx new file mode 100644 index 0000000..a932847 --- /dev/null +++ b/web/src/pages/components/InstallProcess.tsx @@ -0,0 +1,255 @@ +import { useEffect, useState } from 'react'; +import { useModel } from 'umi'; +import { ProCard } from '@ant-design/pro-components'; +import useRequest from '@/utils/useRequest'; +import { + queryInstallStatus, + queryInstallLog, +} from '@/services/ob-deploy-web/Deployments'; +import { handleResponseError } from '@/utils'; +import lottie from 'lottie-web'; +import NP from 'number-precision'; +import styles from './index.less'; + +let timerLogScroll: NodeJS.Timer; +let timerProgress: NodeJS.Timer; + +export default function InstallProcess() { + const { setCurrentStep, configData, installStatus, setInstallStatus } = + useModel('global'); + const name = configData?.components?.oceanbase?.appname; + const [toBottom, setToBottom] = useState(true); + const [progress, setProgress] = useState(0); + const [showProgress, setShowProgress] = useState(0); + const [lottieProgress, setlottieProgress] = useState(); + const [lastError, setLastError] = useState(''); + const [currentPage, setCurrentPage] = useState(true); + + const { run: fetchInstallStatus, data: statusData } = useRequest( + queryInstallStatus, + { + skipStatusError: true, + onSuccess: ({ success, data }: API.OBResponseTaskInfo_) => { + if (success) { + clearInterval(timerProgress); + if (data?.status !== 'RUNNING') { + setInstallStatus(data?.status); + setCurrentPage(false); + setTimeout(() => { + setCurrentStep(6); + }, 2000); + } else { + setTimeout(() => { + fetchInstallStatus({ name }); + }, 1000); + } + const newProgress = NP.divide(data?.finished, data?.total).toFixed(2); + setProgress(newProgress); + let step = NP.minus(newProgress, progress); + let stepNum = 1; + timerProgress = setInterval(() => { + const currentProgressNumber = NP.plus( + progress, + NP.times(NP.divide(step, 100), stepNum), + ); + if (currentProgressNumber >= 1) { + clearInterval(timerProgress); + } else { + stepNum += 1; + setShowProgress(currentProgressNumber); + } + }, 10); + } + }, + onError: ({ response, data }: any) => { + if (currentPage) { + setTimeout(() => { + fetchInstallStatus({ name }); + }, 1000); + } + const errorInfo = data?.msg || data?.detail || response?.statusText; + const errorInfoStr = errorInfo ? JSON.stringify(errorInfo) : ''; + if (errorInfo && lastError !== errorInfoStr) { + setLastError(errorInfoStr); + handleResponseError(errorInfo); + } + }, + }, + ); + + const { run: handleInstallLog, data: logData } = useRequest(queryInstallLog, { + skipStatusError: true, + onSuccess: ({ success }: API.OBResponseInstallLog_) => { + if (success && installStatus === 'RUNNING') { + setTimeout(() => { + handleInstallLog({ name }); + }, 1000); + } + }, + onError: ({ response, data }: any) => { + if (installStatus === 'RUNNING' && currentPage) { + setTimeout(() => { + handleInstallLog({ name }); + }, 1000); + } + const errorInfo = data?.msg || data?.detail || response?.statusText; + const errorInfoStr = errorInfo ? JSON.stringify(errorInfo) : ''; + if (errorInfoStr && lastError !== errorInfoStr) { + setLastError(errorInfoStr); + handleResponseError(errorInfo); + } + }, + }); + + const toLogBottom = () => { + const log = document.getElementById('installLog'); + if (log) { + log.scrollTop = log.scrollHeight; + } + }; + + const handleScroll = (e?: any) => { + e = e || window.event; + const dom = e.target; + if (dom.scrollTop + dom.clientHeight === dom.scrollHeight) { + setToBottom(true); + } else { + setToBottom(false); + } + }; + + const getAnimate = () => { + const computerAnimate = document.querySelector('.computer-animate'); + const progressAnimate = document.querySelector('.progress-animate'); + const spacemanAnimate = document.querySelector('.spaceman-animate'); + const sqlAnimate = document.querySelector('.database-animate'); + + lottie.loadAnimation({ + prefetch: true, + container: computerAnimate, + renderer: 'svg', + loop: true, + autoplay: true, + path: '/assets/computer/data.json', + }); + + setlottieProgress( + lottie.loadAnimation({ + prefetch: true, + container: progressAnimate, + renderer: 'svg', + loop: false, + autoplay: false, + path: '/assets/progress/data.json', + }), + ); + + lottie.loadAnimation({ + prefetch: true, + container: spacemanAnimate, + renderer: 'svg', + loop: true, + autoplay: true, + path: '/assets/spaceman/data.json', + }); + lottie.loadAnimation({ + prefetch: true, + container: sqlAnimate, + renderer: 'svg', + loop: true, + autoplay: true, + path: '/assets/database/data.json', + }); + }; + + useEffect(() => { + if (name) { + fetchInstallStatus({ name }); + handleInstallLog({ name }); + } + }, [name]); + + useEffect(() => { + getAnimate(); + const log = document.querySelector('#installLog'); + log.addEventListener('scroll', handleScroll); + return () => { + log.removeEventListener('DOMMouseScroll', handleScroll); + clearInterval(timerLogScroll); + clearInterval(timerProgress); + }; + }, []); + + useEffect(() => { + if (toBottom) { + toLogBottom(); + timerLogScroll = setInterval(() => toLogBottom()); + } else { + clearInterval(timerLogScroll); + } + }, [toBottom]); + + useEffect(() => { + if (lottieProgress) { + lottieProgress.goToAndStop( + NP.times(showProgress, lottieProgress.totalFrames - 1), + true, + ); + } + }, [lottieProgress, showProgress]); + + return ( + + +
+
+
+
+
+
+
+
+
+
+
+
+
+
+ + 正在部署 {statusData?.current} + +
+ +
+          {logData?.log}
+          {installStatus === 'RUNNING' ? (
+            
+
+
+
+
+
+ ) : null} +
+
+
+ ); +} diff --git a/web/src/pages/components/NodeConfig.tsx b/web/src/pages/components/NodeConfig.tsx new file mode 100644 index 0000000..b5983e6 --- /dev/null +++ b/web/src/pages/components/NodeConfig.tsx @@ -0,0 +1,752 @@ +import { useEffect, useState, useRef } from 'react'; +import { useModel } from 'umi'; +import { Space, Button, Tooltip, Select, Popconfirm, message } from 'antd'; +import { QuestionCircleOutlined, DeleteOutlined } from '@ant-design/icons'; +import { + ProCard, + ProForm, + ProFormText, + ProFormSelect, + EditableProTable, +} from '@ant-design/pro-components'; +import type { + ProColumns, + EditableFormInstance, +} from '@ant-design/pro-components'; +import { getObdInfo } from '@/services/ob-deploy-web/Info'; +import useRequest from '@/utils/useRequest'; +import { handleQuit } from '@/utils'; +import { commonStyle, pathRule } from '../constants'; +import ServerTags from './ServerTags'; +import styles from './index.less'; + +interface FormValues extends API.Components { + auth?: { + user?: string; + password?: string; + }; + home_path?: string; +} + +export default function NodeConfig() { + const { + setCurrentStep, + configData, + setConfigData, + currentType, + lowVersion, + handleQuitProgress, + nameIndex, + setNameIndex, + } = useModel('global'); + const { components = {}, auth, home_path } = configData || {}; + const { oceanbase = {}, ocpexpress = {}, obproxy = {} } = components; + const [form] = ProForm.useForm(); + const [editableForm] = ProForm.useForm(); + const tableFormRef = useRef>(); + + const initDBConfigData = oceanbase?.topology?.length + ? oceanbase?.topology?.map((item: API.Zone, index: number) => ({ + id: (Date.now() + index).toString(), + ...item, + servers: item?.servers?.map((server) => server?.ip), + })) + : [ + { + id: (Date.now() + 1).toString(), + name: 'zone1', + servers: [], + rootservice: undefined, + }, + { + id: (Date.now() + 2).toString(), + name: 'zone2', + servers: [], + rootservice: undefined, + }, + { + id: (Date.now() + 3).toString(), + name: 'zone3', + servers: [], + rootservice: undefined, + }, + ]; + + const homePathSuffix = `/${oceanbase.appname}`; + + const initHomePath = home_path + ? home_path.substring(0, home_path.length - homePathSuffix.length) + : undefined; + + const [dbConfigData, setDBConfigData] = + useState(initDBConfigData); + const [editableKeys, setEditableRowKeys] = useState(() => + dbConfigData.map((item) => item.id), + ); + // all servers + const [allOBServer, setAllOBServer] = useState([]); + // all zone servers + const [allZoneOBServer, setAllZoneOBServer] = useState({}); + const [lastDeleteServer, setLastDeleteServer] = useState(''); + + const serverReg = + /^((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.){3}(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])?$/; + + const { run: getUserInfo } = useRequest(getObdInfo, { + onSuccess: ({ success, data }: API.OBResponseServiceInfo_) => { + if (success) { + form.setFieldsValue({ + auth: { + user: data?.user || undefined, + }, + home_path: data?.user === 'root' ? '/root' : `/home/${data?.user}`, + }); + } + }, + }); + + const handleDelete = (id: string) => { + setDBConfigData(dbConfigData.filter((item) => item.id !== id)); + }; + + const setData = (dataSource: FormValues) => { + let newComponents: API.Components = {}; + if (currentType === 'all') { + newComponents.obproxy = { + ...(components.obproxy || {}), + ...dataSource.obproxy, + }; + if (!lowVersion) { + newComponents.ocpexpress = { + ...(components.ocpexpress || {}), + ...dataSource?.ocpexpress, + }; + } + newComponents.obagent = { + ...(components.obagent || {}), + servers: allOBServer, + }; + } + newComponents.oceanbase = { + ...(components.oceanbase || {}), + topology: dbConfigData?.map((item) => ({ + ...item, + servers: item?.servers?.map((server) => ({ ip: server })), + })), + }; + setConfigData({ + ...configData, + components: newComponents, + auth: dataSource.auth, + home_path: `${ + dataSource.home_path + ? `${dataSource.home_path}${homePathSuffix}` + : undefined + }`, + }); + }; + + const prevStep = () => { + const formValues = form.getFieldsValue(true); + setData(formValues); + setCurrentStep(1); + }; + + const nextStep = () => { + const tableFormRefValidate = () => { + return tableFormRef?.current?.validateFields().then((values) => { + return values; + }); + }; + + const formValidate = () => { + return form.validateFields().then((values) => { + return values; + }); + }; + + Promise.all([tableFormRefValidate(), formValidate()]).then((result) => { + const formValues = result?.[1]; + setData(formValues); + setCurrentStep(3); + }); + }; + + const formatOptions = (data: string[]) => + data?.map((item) => ({ label: item, value: item })); + + const getAllServers = (dataSource: API.DBConfig[]) => { + const allServersList = dataSource.map((item) => item.servers); + let newAllOBServer: string[] = []; + allServersList.forEach((item) => { + if (item && item.length) { + newAllOBServer = [...newAllOBServer, ...item]; + } + }); + return newAllOBServer; + }; + + const onValuesChange = (values: FormValues) => { + if (values?.auth?.user) { + form.setFieldsValue({ + home_path: + values?.auth?.user === 'root' + ? '/root' + : `/home/${values?.auth?.user}`, + }); + } + }; + + useEffect(() => { + const allServers = getAllServers(dbConfigData); + const allZoneServers: any = {}; + dbConfigData.forEach((item) => { + allZoneServers[`${item.id}`] = item.servers; + }); + const obproxyServers = form.getFieldValue(['obproxy', 'servers']); + const ocpexpressServers = form.getFieldValue(['ocpexpress', 'servers']); + const customOBproxyServers = obproxyServers?.filter( + (item: string) => + !(allServers?.includes(item) || item === lastDeleteServer), + ); + const customOcpexpressServers = ocpexpressServers?.filter( + (item: string) => + !(allServers?.includes(item) || item === lastDeleteServer), + ); + let obproxyServersValue; + let ocpexpressServersValue; + if (allServers?.length) { + const checkPass = serverReg.test(allServers[0]); + if (!obproxyServers?.length) { + obproxyServersValue = [allServers[0]]; + } else { + const newOBproxyServers: string[] = []; + obproxyServers?.forEach((item: string) => { + if (allServers?.includes(item)) { + newOBproxyServers.push(item); + } + }); + if (newOBproxyServers?.length) { + obproxyServersValue = [...customOBproxyServers, ...newOBproxyServers]; + } else if (customOBproxyServers?.length) { + obproxyServersValue = customOBproxyServers; + } else { + obproxyServersValue = [allServers[0]]; + if (!checkPass) { + form.setFields([ + { + name: ['obproxy', 'servers'], + errors: ['请选择正确的 OBProxy 节点'], + }, + ]); + } + } + } + + if (!ocpexpressServers?.length) { + ocpexpressServersValue = [allServers[0]]; + } else { + const newOcpexpressServers: string[] = []; + ocpexpressServers?.forEach((item: string) => { + if (allServers?.includes(item)) { + newOcpexpressServers.push(item); + } + }); + if (newOcpexpressServers?.length) { + ocpexpressServersValue = [ + ...customOcpexpressServers, + ...newOcpexpressServers, + ]; + } else if (customOcpexpressServers?.length) { + ocpexpressServersValue = customOcpexpressServers; + } else { + ocpexpressServersValue = [allServers[0]]; + if (!checkPass) { + form.setFields([ + { + name: ['ocpexpress', 'servers'], + errors: ['请选择正确的 OCPExpress 节点'], + }, + ]); + } + } + } + } else { + if (!customOBproxyServers?.length) { + obproxyServersValue = undefined; + } else { + obproxyServersValue = customOBproxyServers; + } + if (!customOcpexpressServers?.length) { + ocpexpressServersValue = undefined; + } else { + ocpexpressServersValue = customOcpexpressServers; + } + } + + form.setFieldsValue({ + obproxy: { + servers: obproxyServersValue, + }, + ocpexpress: { + servers: ocpexpressServersValue, + }, + }); + + setAllOBServer(allServers); + setAllZoneOBServer(allZoneServers); + }, [dbConfigData, lastDeleteServer]); + + useEffect(() => { + if (!auth?.user) { + getUserInfo(); + } + }, []); + + const nameValidator = ({ field }: any, value: string) => { + const currentId = field.split('.')[0]; + let validtor = true; + const reg = /^[a-zA-Z]([a-zA-Z0-9_]{0,30})[a-zA-Z0-9]$/; + if (value) { + if (reg.test(value)) { + dbConfigData.some((item) => { + if (currentId !== item.id && item.name === value) { + validtor = false; + return true; + } + return false; + }); + } else { + return Promise.reject( + new Error( + '以英文字母开头,英文或数字结尾,可包含英文数字和下划线且长度在 2-32 个字符之间', + ), + ); + } + } + if (validtor) { + return Promise.resolve(); + } + return Promise.reject(new Error('Zone 名称已被占用')); + }; + + const ocpServersValidator = (_: any, value: string[]) => { + let validtor = true; + if (value?.length > 1) { + return Promise.reject(new Error('仅可选择或输入一个节点')); + } + if (value && value.length) { + value.some((item) => { + validtor = serverReg.test(item.trim()); + return !serverReg.test(item.trim()); + }); + } + if (validtor) { + return Promise.resolve(); + } + return Promise.reject(new Error('请选择正确的 OCPExpress 节点')); + }; + + const serversValidator = (_: any, value: string[], type: string) => { + let validtor = true; + if (value && value.length) { + value.some((item) => { + validtor = serverReg.test(item.trim()); + return !serverReg.test(item.trim()); + }); + } + if (validtor) { + return Promise.resolve(); + } + if (type === 'OBServer') { + return Promise.reject(new Error('请输入正确的 IP 地址')); + } else { + return Promise.reject(new Error('请选择正确的 OBProxy 节点')); + } + }; + + const columns: ProColumns[] = [ + { + title: ( + <> + Zone 名称 + + + + + ), + dataIndex: 'name', + width: 224, + formItemProps: { + rules: [ + { required: true, whitespace: false, message: '此项是必填项' }, + { validator: nameValidator }, + ], + }, + }, + { + title: ( + <> + OBServer 节点 + + + + + ), + dataIndex: 'servers', + formItemProps: { + rules: [ + { required: true, message: '此项是必填项' }, + { + validator: (_: any, value: string[]) => + serversValidator(_, value, 'OBServer'), + }, + ], + }, + renderFormItem: (_: any, { isEditable, record }: any) => { + return isEditable ? ( + + ) : null; + }, + }, + { + title: ( + <> + RootServer 节点 + + + + + ), + dataIndex: 'rootservice', + formItemProps: { + rules: [ + { required: true, message: '此项是必选项' }, + { pattern: serverReg, message: '请选择正确的 RootServer 节点' }, + ], + }, + width: 224, + renderFormItem: (_: any, { isEditable, record }: any) => { + // rootservice options are items entered by the OBServer + const options = record?.servers ? formatOptions(record?.servers) : []; + return isEditable ? ( + + setParameterValue({ ...parameterValue, adaptive: value }) + } + disabled={!parameterValue?.auto} + > + {optionConfig.map((option) => ( + + {option.label} + + ))} + + + setParameterValue({ ...parameterValue, value: e.target.value }) + } + /> + + ); +} diff --git a/web/src/pages/components/PreCheck.tsx b/web/src/pages/components/PreCheck.tsx new file mode 100644 index 0000000..61c1acf --- /dev/null +++ b/web/src/pages/components/PreCheck.tsx @@ -0,0 +1,9 @@ +import { useModel } from 'umi'; +import CheckInfo from './CheckInfo'; +import PreCheckStatus from './PreCheckStatus'; + +export default function PreCheck() { + const { checkOK } = useModel('global'); + + return checkOK ? : ; +} diff --git a/web/src/pages/components/PreCheckStatus.tsx b/web/src/pages/components/PreCheckStatus.tsx new file mode 100644 index 0000000..def03a9 --- /dev/null +++ b/web/src/pages/components/PreCheckStatus.tsx @@ -0,0 +1,677 @@ +import { useEffect, useState } from 'react'; +import { useModel } from 'umi'; +import { + Space, + Button, + Progress, + Timeline, + Checkbox, + Typography, + Tooltip, + Tag, + Spin, + message, + Empty, +} from 'antd'; +import { ProCard } from '@ant-design/pro-components'; +import { + CloseOutlined, + QuestionCircleFilled, + ReadFilled, + CheckCircleFilled, + CloseCircleFilled, +} from '@ant-design/icons'; +import useRequest from '@/utils/useRequest'; +import { + preCheck, + preCheckStatus, + installDeployment, + createDeploymentConfig, + recover, +} from '@/services/ob-deploy-web/Deployments'; +import { handleQuit, handleResponseError } from '@/utils'; +import NP from 'number-precision'; +import styles from './index.less'; + +const { Text } = Typography; + +const statusColorConfig = { + PASSED: 'green', + PENDING: 'gray', + FAILED: 'red', +}; + +let timerScroll: NodeJS.Timer; +let timerFailed: NodeJS.Timer; +const initDuration = 3; +let durationScroll = initDuration; +let durationFailed = initDuration; + +const errCodeUrl = 'https://www.oceanbase.com/product/ob-deployer/error-codes'; + +export default function PreCheckStatus() { + const { + setCurrentStep, + configData, + setCheckOK, + handleQuitProgress, + getInfoByName, + setConfigData, + } = useModel('global'); + const oceanbase = configData?.components?.oceanbase; + const name = configData?.components?.oceanbase?.appname; + const [statusData, setStatusData] = useState({}); + const [failedList, setFailedList] = useState([]); + const [showFailedList, setShowFailedList] = useState([]); + const [hasAuto, setHasAuto] = useState(false); + const [hasManual, setHasManual] = useState(false); + const [onlyManual, setOnlyManual] = useState(false); + const [checkFinished, setCheckFinished] = useState(false); + const [isScroll, setIsScroll] = useState(false); + const [isScrollFailed, setIsScrollFailed] = useState(false); + const [loading, setLoading] = useState(false); + const [checkStatus, setCheckStatus] = useState(true); + const [lastError, setLastError] = useState(''); + const [currentPage, setCurrentPage] = useState(true); + const [firstErrorTimestamp, setFirstErrorTimestamp] = useState(); + + const { run: fetchPreCheckStatus } = useRequest(preCheckStatus, { + skipStatusError: true, + skipTypeError: true, + onSuccess: ({ success, data }: API.OBResponsePreCheckResult_) => { + if (success) { + let timer: NodeJS.Timer; + setStatusData(data || {}); + if (data?.status === 'RUNNING') { + timer = setTimeout(() => { + fetchPreCheckStatus({ name }); + }, 1000); + } + if (data?.status === 'FAILED') { + handleResponseError(data?.message); + setCheckStatus(false); + } else { + if (data?.all_passed) { + setFailedList([]); + setShowFailedList([]); + } else { + const newFailedList = + data?.info?.filter((item) => item.result === 'FAILED') || []; + newFailedList.forEach((item) => { + if (item.recoverable) { + setHasAuto(true); + } else { + setHasManual(true); + } + }); + setFailedList(newFailedList); + setShowFailedList(newFailedList); + } + const isFinished = !!data?.total && data?.finished === data?.total; + setCheckFinished(isFinished); + if (isFinished) { + clearTimeout(timer); + } + if (!isScroll && !isFinished) { + setTimeout(() => { + const timelineContainer = + document.getElementById('timeline-container'); + const runningItemDom = document.getElementById( + 'running-timeline-item', + ); + timelineContainer.scrollTop = NP.minus( + NP.strip(runningItemDom?.offsetTop), + 150, + ); + }, 10); + } + + if (!isScrollFailed && !isFinished && failedList) { + setTimeout(() => { + const failedContainer = + document.getElementById('failed-container'); + if (failedContainer) { + failedContainer.scrollTop = NP.strip( + failedContainer?.scrollHeight, + ); + } + }, 10); + } + setCheckStatus(true); + } + if (loading) { + setLoading(false); + } + } + }, + onError: ({ response, data, type }: any) => { + const handleError = () => { + const errorInfo = + data?.msg || + data?.detail || + response?.statusText || + '您的网络发生异常,无法连接服务器'; + const errorInfoStr = errorInfo ? JSON.stringify(errorInfo) : ''; + if (errorInfoStr && lastError !== errorInfoStr) { + setLastError(errorInfoStr); + handleResponseError(errorInfo); + } + }; + if (response?.status === 504 || (!response && type === 'TypeError')) { + const nowTime = new Date().getTime(); + if (!firstErrorTimestamp) { + setFirstErrorTimestamp(nowTime); + } + if (NP.divide(nowTime - firstErrorTimestamp) > 60000) { + handleError(); + setCheckStatus(false); + if (loading) { + setLoading(false); + } + } else { + if (currentPage) { + setTimeout(() => { + fetchPreCheckStatus({ name }); + }, 1000); + } + } + } else { + handleError(); + } + }, + }); + + const { run: handlePreCheck, loading: preCheckLoading } = useRequest( + preCheck, + { + onSuccess: ({ success }: API.OBResponse) => { + if (success) { + handleStartCheck(); + } + }, + onError: () => { + setCheckStatus(false); + if (loading) { + setLoading(false); + } + }, + }, + ); + + const { run: handleInstallConfirm } = useRequest(installDeployment); + + const handelCheck = async () => { + setLoading(true); + try { + await handlePreCheck({ name }); + } catch { + setLoading(false); + } + }; + + const { run: handleCreateConfig, loading: createLoading } = useRequest( + createDeploymentConfig, + { + onSuccess: ({ success }: API.OBResponse) => { + if (success) { + handelCheck(); + } + setLoading(false); + }, + onError: () => { + setCheckStatus(false); + if (loading) { + setLoading(false); + } + }, + }, + ); + + const handleRetryCheck = (newConfigData?: any) => { + setStatusData({}); + setFailedList([]); + setShowFailedList([]); + setCheckFinished(false); + let params = { ...configData }; + if (newConfigData) { + params = { ...newConfigData }; + } + setLoading(true); + handleCreateConfig({ name: oceanbase?.appname }, { ...params }); + }; + + const { run: handleRecover, loading: recoverLoading } = useRequest(recover, { + onSuccess: async ({ + success, + }: API.OBResponseDataListRecoverChangeParameter_) => { + if (success) { + message.success('自动修复成功'); + try { + const { success: nameSuccess, data: nameData } = await getInfoByName({ + name, + }); + if (nameSuccess) { + const { config } = nameData; + setConfigData(config || {}); + handleRetryCheck(config); + } else { + message.error('获取配置信息失败'); + } + } catch (e: any) { + const { response, data } = e; + handleResponseError( + data?.msg || data?.detail || response?.statusText, + ); + } + } + }, + }); + + const handleStartCheck = () => { + fetchPreCheckStatus({ name }); + }; + + const prevStep = () => { + setCheckOK(false); + setCurrentStep(3); + setCurrentPage(false); + }; + + const handleInstall = async () => { + const { success } = await handleInstallConfirm({ name }); + if (success) { + setCurrentStep(5); + setCurrentPage(false); + } + }; + + const handleScrollTimeline = () => { + if (!checkFinished) { + setIsScroll(true); + clearInterval(timerScroll); + durationScroll = initDuration; + timerScroll = setInterval(() => { + if (durationScroll === 0) { + clearInterval(timerScroll); + setIsScroll(false); + durationScroll = initDuration; + } else { + durationScroll -= 1; + } + }, 1000); + } + }; + + const handleScrollFailed = () => { + if (!checkFinished) { + setIsScrollFailed(true); + clearInterval(timerFailed); + durationFailed = initDuration; + timerFailed = setInterval(() => { + if (durationFailed === 0) { + clearInterval(timerFailed); + setIsScrollFailed(false); + durationFailed = initDuration; + } else { + durationFailed -= 1; + } + }, 1000); + } + }; + + const handleAutoRepair = () => { + setHasAuto(false); + handleRecover({ name }); + }; + + useEffect(() => { + if (onlyManual) { + const newShowFailedList = failedList.filter((item) => !item.recoverable); + setShowFailedList(newShowFailedList); + } else { + setShowFailedList(failedList); + } + }, [onlyManual]); + + useEffect(() => { + handelCheck(); + const timelineContainer = document.getElementById('timeline-container'); + timelineContainer.onmousewheel = handleScrollTimeline; // ie , chrome + timelineContainer?.addEventListener('DOMMouseScroll', handleScrollTimeline); // firefox + return () => { + timelineContainer.onmousewheel = () => {}; + timelineContainer?.removeEventListener( + 'DOMMouseScroll', + handleScrollTimeline, + ); + }; + }, []); + + useEffect(() => { + const addEventFailedContainer = () => { + const failedContainer = document.getElementById('failed-container'); + if (failedList?.length && failedContainer) { + if (!failedContainer.onmousewheel) { + failedContainer.onmousewheel = handleScrollFailed; // ie , chrome + failedContainer?.addEventListener( + 'DOMMouseScroll', + handleScrollFailed, + ); // firefox + } + } else { + setTimeout(() => { + addEventFailedContainer(); + }, 3000); + } + }; + + addEventFailedContainer(); + return () => { + const failedContainer = document.getElementById('failed-container'); + if (failedContainer) { + failedContainer.onmousewheel = () => {}; + failedContainer?.removeEventListener( + 'DOMMouseScroll', + handleScrollFailed, + ); + } + }; + }, [failedList]); + + let progressStatus = 'active'; + if (statusData?.status === 'FAILED') { + progressStatus = 'exception'; + } else if (checkFinished) { + if (statusData?.all_passed) { + progressStatus = 'success'; + } else { + progressStatus = 'exception'; + } + } + + const shape = ( +
+
+
+
+
+
+ ); + + return ( + + + handleRetryCheck()} + data-aspm-click="c307513.d317293" + data-aspm-desc="预检查结果-重新检查" + data-aspm-param={``} + data-aspm-expo + > + 重新检查 + + } + headStyle={{ paddingLeft: '16px', paddingRight: '16px' }} + > + + {loading ? null : ( + <> + + + {statusData?.info?.map( + (item: API.PreCheckInfo, index: number) => ( + + ) : ( + + ) + ) : null + } + > + {item?.name} {item?.server} + + ), + )} + + + )} + + + {hasManual ? ( + setOnlyManual(e.target.checked)} + disabled={!checkFinished || statusData?.all_passed} + > + 只看手动修复项 + + ) : null} + + + } + > + {showFailedList?.length ? ( +
+ {showFailedList?.map((item, index) => { + let reason = ''; + if (item?.description) { + const index = item?.description.indexOf(':'); + reason = item?.description.substring( + index, + item?.description.length, + ); + } + return ( + + + + + + {item.name} + + + + + 原因: + + OBD-{item.code} + {' '} + {reason} + + + + + + 建议: + {item.recoverable ? ( + 自动修复 + ) : ( + 手动修复 + )}{' '} + {item.advisement?.description} + +
+ + 了解更多方案 + +
+
+ ); + })} + {!checkFinished ? ( +
{shape}
+ ) : null} +
+ ) : checkFinished ? ( + 太棒了!无失败项 + } + /> + ) : ( +
+ {shape} +
+ 暂未发现失败项 +
+
+ )} + + +
+
+ + + + {!statusData?.all_passed ? ( + + + + ) : ( + + )} + +
+
+ + ); +} diff --git a/web/src/pages/components/ServerTags.tsx b/web/src/pages/components/ServerTags.tsx new file mode 100644 index 0000000..c06092c --- /dev/null +++ b/web/src/pages/components/ServerTags.tsx @@ -0,0 +1,151 @@ +import { useEffect, useState, useRef } from 'react'; +import { Select, Tooltip, Tag } from 'antd'; + +interface Props { + value?: string[]; + onChange?: (values?: string[]) => void; + name?: string; + setLastDeleteServer: (value: string) => void; +} + +export default ({ + value: values, + onChange, + name, + setLastDeleteServer, +}: Props) => { + const [visible, setVisible] = useState(false); + const [currentValues, setCurrentValues] = useState(values); + const getOverValues = (dataSource?: string[]) => { + return dataSource?.length > 3 ? dataSource.splice(3) : []; + }; + + const [overValuess, setOverValues] = useState( + getOverValues([...(values || [])]), + ); + + const open = useRef(); + open.current = { + input: false, + tooltip: false, + }; + + const onMouseEnterInput = () => { + open.current = { + ...(open?.current || {}), + input: true, + }; + setVisible(true); + }; + + const onMouseEnterTooltip = () => { + open.current = { + ...(open?.current || {}), + tooltip: true, + }; + setVisible(true); + }; + + const onMouseLeaveInput = () => { + setTimeout(() => { + if (!open?.current?.tooltip) { + setVisible(false); + } + }, 300); + }; + + const onMouseLeaveTooltip = () => { + setVisible(false); + }; + + const addEventTooltipOverlay = () => { + const tooltipOverlay = document.querySelector( + `.server-tooltip-overlay-${name}`, + ); + if (tooltipOverlay) { + tooltipOverlay?.addEventListener('mouseenter', onMouseEnterTooltip); + tooltipOverlay?.addEventListener('mouseleave', onMouseLeaveTooltip); + } else { + setTimeout(() => { + addEventTooltipOverlay(); + }, 500); + } + }; + + const addEventInputConatiner = () => { + const inputConatiner = document.querySelector(`.server-${name}`); + if (inputConatiner) { + inputConatiner?.addEventListener('mouseenter', onMouseEnterInput); + inputConatiner?.addEventListener('mouseleave', onMouseLeaveInput); + } else { + setTimeout(() => { + addEventInputConatiner(); + }, 500); + } + }; + + useEffect(() => { + const tooltipOverlay = document.querySelector( + `.server-tooltip-overlay-${name}`, + ); + const inputConatiner = document.querySelector(`.server-${name}`); + addEventTooltipOverlay(); + addEventInputConatiner(); + return () => { + tooltipOverlay?.removeEventListener('mouseenter', onMouseEnterTooltip); + tooltipOverlay?.removeEventListener('mouseleave', onMouseLeaveTooltip); + inputConatiner?.removeEventListener('mouseenter', onMouseEnterInput); + inputConatiner?.removeEventListener('mouseleave', onMouseLeaveInput); + }; + }, []); + + useEffect(() => { + setOverValues(getOverValues([...(currentValues || [])])); + if (onChange && currentValues?.length !== values?.length) { + onChange(currentValues); + } + }, [currentValues]); + + const onSelectChange = (changeValues?: string[]) => { + setCurrentValues(changeValues); + setLastDeleteServer(''); + }; + + const onClose = (value: string) => { + const newCurrentValues = currentValues?.filter((item) => item !== value); + setCurrentValues(newCurrentValues); + setLastDeleteServer(value); + }; + + const getOverContents = () => { + return overValuess?.map((item, index) => ( + onClose(item)}> + {item} + + )); + }; + + return ( + +