diff --git a/.gitignore b/.gitignore index 56e5fc05b4734dccfcbdcb88bf8106e7ff7bdb9f..c3578b3243f54cc9d94b09718cbe5a3de1cca002 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 6ad96f0cc351909997a2fb2cd67eecfa5dd8234e..4fe111533bbc56849969ba4f8d6d350dc7c58aef 100644 --- a/_cmd.py +++ b/_cmd.py @@ -24,18 +24,16 @@ from __future__ import absolute_import, division, print_function 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 @@ VERSION = '' 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 @@ class BaseCommand(object): 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 @@ class ObdCommand(BaseCommand): 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 @@ class ObdCommand(BaseCommand): @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 @@ class ObdCommand(BaseCommand): 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 @@ class ObdCommand(BaseCommand): 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 @@ class MajorCommand(BaseCommand): 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 @@ class DevModeEnableCommand(HiddenObdCommand): 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 @@ class DevModeDisableCommand(HiddenObdCommand): 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 @@ class MirrorCloneCommand(ObdCommand): 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 @@ class MirrorCreateCommand(ObdCommand): 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 @@ class MirrorListCommand(ObdCommand): 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 @@ class MirrorListCommand(ObdCommand): 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 @@ class MirrorEnableCommand(ObdCommand): 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 @@ class MirrorDisableCommand(ObdCommand): 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 @@ class MirrorMajorCommand(MajorCommand): 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 @@ class ClusterConfigStyleChange(ClusterMirrorCommand): 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 @@ class ClusterCheckForOCPChange(ClusterMirrorCommand): 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 @@ class DemoCommand(ClusterMirrorCommand): 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 @@ class ClusterAutoDeployCommand(ClusterMirrorCommand): 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 @@ class ClusterAutoDeployCommand(ClusterMirrorCommand): 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 @@ class ClusterDeployCommand(ClusterMirrorCommand): 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 @@ class ClusterDestroyCommand(ClusterMirrorCommand): 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 @@ class ClusterRedeployCommand(ClusterMirrorCommand): 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 @@ class ClusterChangeRepositoryCommand(ClusterMirrorCommand): 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 @@ class CLusterUpgradeCommand(ClusterMirrorCommand): 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 @@ class ClusterTenantCreateCommand(ClusterMirrorCommand): 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 @@ class ClusterTenantDropCommand(ClusterMirrorCommand): 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 @@ class ClusterTenantDropCommand(ClusterMirrorCommand): 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 @@ class SysBenchCommand(TestMirrorCommand): 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 @@ class DbConnectCommand(HiddenObdCommand): 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 @@ class DoobaCommand(HiddenObdCommand): 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 @@ class DoobaCommand(HiddenObdCommand): 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 @@ class UpdateCommand(ObdCommand): 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 @@ if __name__ == '__main__': 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 7f5158754b0c18534202eb116e33d768dadcd809..9e713813c548693f218cbd69c651cc8859056e6e 100644 --- a/_deploy.py +++ b/_deploy.py @@ -23,7 +23,6 @@ from __future__ import absolute_import, division, print_function import os import re import sys -import pickle import getpass import hashlib from copy import deepcopy @@ -31,9 +30,9 @@ from enum import Enum 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 @@ class ClusterConfig(object): 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 a544ac8f4f0a125d6a65acfd573f65b0c338b838..8d022d0da69983b09401053552c5336fc8fff785 100644 --- a/_environ.py +++ b/_environ.py @@ -23,6 +23,9 @@ from __future__ import absolute_import, division, print_function # 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 @@ ENV_REPO_INSTALL_MODE = "OBD_REPO_INSTALL_MODE" # 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 b1c09e2622ec398128bb78ea24df74ada34b0e48..4c92c3282633d480b15edc2cb8aee9dfc64a63a4 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 a15de07747fe2035a6f46c492b8b1ac1a2c13ce3..19e33fde40a99d6062c7e98e074bfa84d52821ef 100644 --- a/_lock.py +++ b/_lock.py @@ -188,6 +188,12 @@ class EXLock(Lock): 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 @@ class LockManager(Manager): @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 @@ class LockManager(Manager): 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 a1ed7ffb6e350a388114555bdd7c47e02f3efb0e..b3c41cf58f74f245a378ae3cab5aca5cc03ae171 100644 --- a/_mirror.py +++ b/_mirror.py @@ -24,6 +24,7 @@ from __future__ import absolute_import, division, print_function import re import os import sys +import tempfile import time import pickle import string @@ -33,6 +34,7 @@ from glob import glob 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 @@ class RemoteMirrorRepository(MirrorRepository): 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 @@ class RemoteMirrorRepository(MirrorRepository): 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 @@ class RemoteMirrorRepository(MirrorRepository): 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 @@ class RemoteMirrorRepository(MirrorRepository): 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 @@ class LocalMirrorRepository(MirrorRepository): 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 @@ class MirrorRepositoryManager(Manager): 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 7d714a21e556d5cb955ff0f0792eacbcb604c8da..08149447b4030672132e8cd8ace2ef8184700cc3 100644 --- a/_plugin.py +++ b/_plugin.py @@ -25,7 +25,7 @@ import re 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 @@ class Plugin(object): 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 @@ class PluginReturn(object): @property def value(self): return self._return_value - + @property def args(self): return self._return_args @@ -91,11 +118,9 @@ class PluginReturn(object): @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 @@ class PluginReturn(object): 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 @@ class PluginReturn(object): 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 @@ class SubIO(object): def __del__(self): self.before_close() - + def _temp_function(self, *arg, **kwargs): pass @@ -192,13 +240,21 @@ class ScriptPlugin(Plugin): 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 @@ class ScriptPlugin(Plugin): 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 pyScriptPluginExec(func): 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 @@ class ParamPlugin(Plugin): raise Exception('Invalid Value') else: self._value = 0 - + class Time(ConfigItemType): UNITS = { @@ -384,7 +463,7 @@ class ParamPlugin(Plugin): 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 @@ class ParamPlugin(Plugin): 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 ParamPlugin(Plugin): 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 @@ class ParamPlugin(Plugin): 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 @@ class ParamPlugin(Plugin): 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 @@ class ParamPlugin(Plugin): 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 @@ class ParamPlugin(Plugin): 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 @@ class ParamPlugin(Plugin): 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 @@ class InstallPlugin(Plugin): def var_replace(cls, string, var): if not var: return string - done = [] + done = [] while string: m = cls._KEYCRE.search(string) @@ -830,7 +926,7 @@ class ComponentPluginLoader(object): 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 @@ class PyScriptPluginLoader(ComponentPluginLoader): 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 @@ class %s(PyScriptPlugin): %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 7830a4bcfba24d2cf5c7f3365473b7adda79ff15..64c5757f02dcadb1309b0f44708488cc58c7f741 100644 --- a/_repository.py +++ b/_repository.py @@ -30,7 +30,8 @@ from multiprocessing.pool import Pool 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 @@ class ExtractFileInfo(object): 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 @@ class ParallerExtractor(object): 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 @@ class ParallerExtractor(object): 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 @@ class ParallerExtractor(object): 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 @@ class Repository(PackageInfo): 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 @@ class Repository(PackageInfo): 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 67ac4c3408a4e7ba7937a61175bf53afbffe90b9..13d4080d5b9a94a9b46f2086bfe073f2a5d1986f 100644 --- a/_stdio.py +++ b/_stdio.py @@ -48,24 +48,61 @@ if sys.version_info.major == 3: 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 @@ class SysStdin(object): 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 @@ class IOHalo(Halo): 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 @@ class IOHalo(Halo): 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 @@ class IOProgressBar(ProgressBar): 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 @@ class IOProgressBar(ProgressBar): 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 @@ class IOProgressBar(ProgressBar): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class IO(object): 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 @@ class StdIO(object): 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 d83a203ffcdf90e875e6094e3424d054654258c0..25183e29bdc7c78045ec569a461babf24476819c 100644 --- a/config_parser/oceanbase/cluster_config_parser.py +++ b/config_parser/oceanbase/cluster_config_parser.py @@ -56,6 +56,12 @@ class ClusterConfigParser(ConfigParser): 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 @@ class ClusterConfigParser(ConfigParser): 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 4bcc5d3a2ed015c0a3d73de2dd3ea675d5cc245a..1316bb20b0bb887384332478a5d92aed9b827cd5 100644 --- a/core.py +++ b/core.py @@ -24,21 +24,21 @@ import re 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): ), 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): _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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): # 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 @@ class ObdHome(object): 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 19c534946345d34b941545fa0d60fd8a540df3c8..9d8e69ff68cedb1a0cf8d0107be98e3c0ac13bc3 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 17806bd73682aec40e91780c936d2676fac0f321..a993ed529eb704cbee924c9eb8a4948b9f607628 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 270cf60b6b537a23bf1a63ac06829a0252999da6..d7ad5e0b68391e6677d51abb61b2379313aa34bd 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 fe925968f998815b3ce07a3bb4357f445f57c619..d7ad5e0b68391e6677d51abb61b2379313aa34bd 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 2eec916043ab4e764de066884528a859cc2146af..3999ddef35022777b1e8ca162764ac73d3db3c55 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 37d0d537afd1181a4b2cde5d3dec0c017ff10915..f9691b442a2a63f62d700ad0b34cd737e1f67f1f 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 6b53be45a0fb7bd5691b556337876ffa10febeec..203e9d816fe2b16f5d44a39d1740ba1eb8cf5c6b 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 1526ba847dc9e5965940dea010e32ccf88f99df2..7a943115076cf0b8e8b4108f276c89f5bb2dafab 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 8c0b846e9c2be4c1eae4e42831df446432d0e9d2..185688dcc042a3513c3c13866c4eb1f4174d9c9b 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 ebc05289cc4b185a9fd3371aef1e3eca7904baf1..b6acd56720e6a3c3083efea9fca88d5fa2d182bb 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 f8d556fb1435f723ca98163233b35cecc415c19c..447237ad41c10e0f92a80868158369c6f0fad01b 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 0000000000000000000000000000000000000000..f48d675516fb9ca31d6520f1033c8102da92d4b4 --- /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 ea254f12ddb102e857787e7165c3059b1059afd8..dd15e635387d4c96ec5f850ff5b9cd3c47156b9c 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 e366c854b7f41ae0e8f752426eadecf38a9bc1e0..3cf262e11ce7aaa0b15bedc953ec13f1fce16ed4 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 7d4b097dc57be8aee3e1004f01b3ffc58b91c7a8..3301a88ffacb79c99c75575c23ac00490e8ad030 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 4a6bef82bd1c8e1dcc11411c2c3afb66e26e7e94..9d8055a933a10bba0e99609a6d2eefe57e8703c2 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 d96c623d0971c1f822a9ed0f4cfa5ba5a34e1e85..24faa425509db2279d87534363ca7584dca19109 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 da967aafe0c0baff0dc19fbb232d0f36d601bd88..dd5ee0ee5c6852138abe3b259a5ee5bde28ffe51 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 b0cebceec251981e02e325760ff42771b5378cc2..55d1b30c05f55f2029646255ba08a5fbe97ee7c8 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 0000000000000000000000000000000000000000..d6a95b461a4ba622944855eb9e5ad77adb8deeb4 --- /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 3419f51eae89ab9b162b67b10a2c987c1fb352d8..fc45667050d18de02478350ac0012b3b3bfb2695 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 f8d556fb1435f723ca98163233b35cecc415c19c..447237ad41c10e0f92a80868158369c6f0fad01b 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 0000000000000000000000000000000000000000..300c798282be585f84a96a82668ed18a900c755b --- /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 0000000000000000000000000000000000000000..3608cc2039a8b2cc87f767872d7c6670cf5d11bb --- /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 fa4fea98748e6833bca2d4f35fcb0bc038123c91..1eff5af75e9f7824d600bfbd412b4874210d69b1 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 5e00e6ec2868222890e6cc883b58ea1fbb691ee2..4c5cecfcd72cd566ba550c27553b4fdad66d9ecc 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 fe041f7db76f2c83909e4bb9fe0f7189b9d3149e..b39a4ac4a03cff3236c6f3c81dca300954d5b027 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 0000000000000000000000000000000000000000..41eef71b20c117170ae0fb3960a84195304cbc78 --- /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 0000000000000000000000000000000000000000..c7cb45cc8a8b220282fc124ca1771653814b43be --- /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 0000000000000000000000000000000000000000..2d83eacca4009e827b4476988c92ae78f8d1e06f --- /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 0000000000000000000000000000000000000000..3edbfc03040c61a303d877a75f2a934bb236f690 --- /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 0000000000000000000000000000000000000000..b72d27ae0b983692fe597280873b2fe29320d605 --- /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 0000000000000000000000000000000000000000..2d83eacca4009e827b4476988c92ae78f8d1e06f --- /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 0000000000000000000000000000000000000000..3edbfc03040c61a303d877a75f2a934bb236f690 --- /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 0000000000000000000000000000000000000000..b72d27ae0b983692fe597280873b2fe29320d605 --- /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 3022d0534eb45b1a4e1930619d0340a0c0fd014f..0ecb4fc591914c23efd84e5d2841be4cd38774e1 100644 --- a/optimize/optimize_parser/0.1/optimize_parser.py +++ b/optimize/optimize_parser/0.1/optimize_parser.py @@ -148,7 +148,7 @@ class OptimizeItem(object): 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 @@ class CursorOptimizer(SafeStdio): 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 @@ class CursorOptimizer(SafeStdio): 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 0b3f8d5a11a06d27d124f70a9c4444b52ffaceb2..c413c9ba8614a65d23ad0356c980ddba197f96aa 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 51ce5194bf10ded41c39046093f32cc314520b30..d1730055ee069782ace2252d5b95b9f24d79280d 100644 --- a/plugins/general/0.1/install_repo.py +++ b/plugins/general/0.1/install_repo.py @@ -93,7 +93,7 @@ def install_repo(plugin_context, obd_home, install_repository, install_plugin, c 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 425de37d35ba71ebb013d05dd70328e5635e0027..4a7373af1e6b606aa39bdbc70accfd182e3612f3 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 46d41753a1a67187d57137c1582cb1b0968635ea..22b5a145c48c49160ea8eb57f0be562f88708d13 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 40ea1f25fa30e1fc5323cab302b516c98b0632b7..febb2ba3d1f8aab0f5e81ec2146c0a7b4fb3a022 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 a2c2986a23e0d1179816152b2c60302fe8812d6a..9c08961ff1df5e5582587151aac6d126daa7394a 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 fecf1cd4c0671325726df01cb6a8be358d247bd7..6970449c6cfab50cbd0bc7c31b2fc20e6affe94f 100644 --- a/plugins/grafana/7.5.17/reload.py +++ b/plugins/grafana/7.5.17/reload.py @@ -27,7 +27,7 @@ import re 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 ddc8811b909e04c4b2acce1826721e3b1366baad..9606c5454fa9dbddda1ce890a5e1e8fddfc15411 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 5703040ff6e314167ee3a8914a4fb0c7d51fd4cb..5def012dad46e366f20635818b795536a18ed611 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 start(plugin_context, local_home_path, repository_dir, need_bootstrap=False, 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 start(plugin_context, local_home_path, repository_dir, need_bootstrap=False, 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 dfc8c0b007eee3dc638545cdcbc72ade87204488..0b01aa1b7d6d37d98e4e937dea036f3dd4036e76 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 start_check(plugin_context, strict_check=False, *args, **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 193ec670711a30cfc16856ec4c31cec4821c9452..9c3f9a78cdb413861bf25d18f1f43bc9a304b9dd 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 cd33d15468db8250a5aec39803b0246445ef33cc..5bb056127117dc674ade0f5c132a52f387fb2daf 100644 --- a/plugins/mysqltest/3.1.0/check_opt.py +++ b/plugins/mysqltest/3.1.0/check_opt.py @@ -25,7 +25,8 @@ import os 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 555adfb7dd59c8c791c9388167a4b7fd2f9f0df6..4501249f833a215fcfa511442457b78b58a7433e 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 0ab059312f79ecbc6e40c96c6fffa73e404bc021..6ddfe8e2b76272fa8bedaa77441831acdb8e43aa 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 6a2777c99fd5961ce779386f9f22896ac35bb266..c407f06c7ffb89aedac8a36368fe876ca1a5a425 100644 --- a/plugins/mysqltest/3.1.0/run_test.py +++ b/plugins/mysqltest/3.1.0/run_test.py @@ -205,7 +205,6 @@ def run_test(plugin_context, env, *args, **kwargs): 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 run_test(plugin_context, env, *args, **kwargs): 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 b87dd34e12ab056762eb1f025449d11027c59d92..7f2ed99565b7b789633c7e18735e0ee7441af1fb 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 ff25721f967aa6f71b27dc0d21c8ee971ed0fe89..2b7701133ac7d20bee3804af5f4effeec6137440 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 1d973e8fcb93790bb677e7689c206ac18ed86391..f04f76d282799943239751f31099aeddb236966c 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 61c1bbf3ec1e699e6364309693dd195af767ced8..e46b1a10bfd7837444644ed380f8da07cf6a4d44 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 14456ec404118d2e1dd9b2f70aedf8e25ceea407..379dd1ed097c34f4a5925b659723862aa21df061 100644 --- a/plugins/obagent/0.1/init.py +++ b/plugins/obagent/0.1/init.py @@ -23,7 +23,7 @@ from __future__ import absolute_import, division, print_function 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 ddd98066d4c49203843360d6aad87e81fd10479f..eb35d078778db4f91010e13cba6f1de8b9ec70a8 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 0034d3c056dad732dc0ba2e389b7408a6d1ee2f5..fa3f9c148e575697849de79618acfeba8537e2f3 100644 --- a/plugins/obagent/0.1/reload.py +++ b/plugins/obagent/0.1/reload.py @@ -29,11 +29,17 @@ from tool import YamlLoader 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 79cbbe52730cc83a2e349f6f85f21556dbbf879d..fc6815f0c08bda92bc0ed4e8da88343bf7641991 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 ede6e5f65d958eea24f3496981c72245f8f00c1e..ac4e42bf31b5650fd1df3914c53932c1f4a675fd 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 d4d755cb81a6ad1a27cf81364104bad1cb1f6d23..ca627458fb8d5c246d11b737f144492fd6280f6c 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 @@ success = True 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 start_check(plugin_context, strict_check=False, *args, **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 3f4fd9f85a369e2da1eefa3673d020c76197249d..8f520d9024b20eb681567ff0a9b591b28acc436f 100644 --- a/plugins/obagent/0.1/stop.py +++ b/plugins/obagent/0.1/stop.py @@ -28,7 +28,7 @@ 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) 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 ded202c030da33c340301909862d5811dcd2913c..17bf275379e0f37403430ed6eddd9221614336c4 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 bf8897784331e81b4a9605f1955954b7a20b1c5c..c54944a37e11553674e6427e710947745bc9a05c 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 d2fbef3735e6afb607b6e5039e8d8203c56354f4..d15693004a01f08d31b12b4f7be9195616693ab7 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 3d83b34cc56ff4bfc5179d4e706dc0c68c9decba..e3e5ad69f352e607d3ca18a19334d78769778956 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 @@ success = True 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 start_check(plugin_context, strict_check=False, *args, **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 0000000000000000000000000000000000000000..eab4e2846f8febe78185c576f912e63d2d58ce61 --- /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 0000000000000000000000000000000000000000..2b82766b7562665ca8b20a83bfbe283c61394f8a --- /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 0000000000000000000000000000000000000000..03e84e0c7539c5451a049651bf3596b27404caa7 --- /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 0000000000000000000000000000000000000000..f92aff85438cceccefd8d0f1320bcc715703a4b0 --- /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 0000000000000000000000000000000000000000..8cafaa3657c8fcd5e4862212306f68a25bcb1174 --- /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 0000000000000000000000000000000000000000..886a9e728472fff321f5f24e2ff41cf76e7d42f4 --- /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 0000000000000000000000000000000000000000..4ecf445b10c2482f65b25fb302d6fb5f37819b93 --- /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 0000000000000000000000000000000000000000..d11c25a17a50fdba2aa757f77066136535ef69b0 --- /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 0000000000000000000000000000000000000000..0373a0e0526f54317b66ee2638b42ac2121df4a7 --- /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 0000000000000000000000000000000000000000..7a3f63741c6ae775fe6c911f75669564bc545421 --- /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 0000000000000000000000000000000000000000..6cbd26ebcf933f7f12433c6605a5b1aaeb648194 --- /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 0000000000000000000000000000000000000000..7c4d391b58cda6b77d8147547671996c81a41a4f --- /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 4125841e421205a45b7129b6859230e709b793ef..56318782b4d6f36f83179aea3f7f3bb01caa2505 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 9cbb6c66b057299c79414eaedc416c5d4a19b786..e4f2c62bc67738099ba682aea7566cf02034aa51 100644 --- a/plugins/obproxy/3.1.0/connect.py +++ b/plugins/obproxy/3.1.0/connect.py @@ -22,43 +22,101 @@ from __future__ import absolute_import, division, print_function 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 2310ca7a25cbee8a6c636ec1b47b5f41712c8809..2e20633e7a7825138c042ef36481f02f48a0e9d2 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 08def5951a57f8c027a80d69c55ce97765da4368..17059791d98c88f79f9a3fc6b79af698cffb4e68 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 1ea7ee954aa5793bd40677b0eeb30f03861fa6ab..ae019a9b44a140fb51ed9c6dfb16e644add861d8 100644 --- a/plugins/obproxy/3.1.0/init.py +++ b/plugins/obproxy/3.1.0/init.py @@ -22,7 +22,7 @@ from __future__ import absolute_import, division, print_function 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 0f40a1bad97b103265000315eeb3a0973b6f945c..37c75769ae3ddc453741a42407064167d4907be6 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 291a55aad1001c7451283c6d4dfb2d6634528803..1a2acc87aa14566894ea141a0b16f684bd5be2ae 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 0a646e6f4307235a20d35fb9388c8fbf112c9fc6..c550fad823f076d59c35f86368e590da0b7c2871 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 81352668e8e166071661b92510b8b7db577a3dbe..a3bc5c75a6be0fe27cc26d0eb0088406453301cc 100644 --- a/plugins/obproxy/3.1.0/start.py +++ b/plugins/obproxy/3.1.0/start.py @@ -24,12 +24,14 @@ import os 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 @@ class EnvVariables(object): 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 68b2189dc1a892b5c4fd2448d642f90e695915e6..bca4c0c240077185aa50b901a82b40c97d1da5af 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 @@ success = True 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 start_check(plugin_context, strict_check=False, *args, **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 0a748c6a577981ac630c4446df25cf71f5cc888e..169960d441ab915eacc7abc933020c8eb6bd0baf 100644 --- a/plugins/obproxy/3.1.0/stop.py +++ b/plugins/obproxy/3.1.0/stop.py @@ -28,7 +28,7 @@ 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) 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 af4ea3eb5d4891adbcd5354f2f87913b1d23966e..eec38aeccf5cb1e41841e09e49127439fa7b308a 100644 --- a/plugins/obproxy/3.1.0/upgrade.py +++ b/plugins/obproxy/3.1.0/upgrade.py @@ -22,11 +22,18 @@ from __future__ import absolute_import, division, print_function 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 2f30640d8f40459f1eca898c91d839349eb43a68..1070ac4e05159a489c9263899eedcbb9e514b1d3 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 289dc266d4b558d39f6ea257800c868f4b32ed0f..893b414c2c7f23eefa870e0970959d254ccc0422 100644 --- a/plugins/oceanbase/3.1.0/connect.py +++ b/plugins/oceanbase/3.1.0/connect.py @@ -22,31 +22,102 @@ from __future__ import absolute_import, division, print_function 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 f3e16d4a1a8098a942afc8a40e331391fa7831ad..e828234a818e2ba717657105bf5368b4b9019465 100644 --- a/plugins/oceanbase/3.1.0/create_tenant.py +++ b/plugins/oceanbase/3.1.0/create_tenant.py @@ -26,6 +26,8 @@ import time 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 create_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 - 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 create_tenant(plugin_context, cursor, *args, **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 a2a12dbd5d396ea6b9ef42b60f4560f8cdb24add..e04bcfeb690b5089531c2d782d29caa9b7756aaa 100644 --- a/plugins/oceanbase/3.1.0/display.py +++ b/plugins/oceanbase/3.1.0/display.py @@ -23,6 +23,10 @@ from __future__ import absolute_import, division, print_function 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 7ff28e705ee72728a432a83274de32e8753ec756..fdf36d2c5cd91d17eb0cf41a3fff5f492c09b59b 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 drop_tenant(plugin_context, cursor, *args, **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 5e453f75df0fbede47e4eeb88d7043ccad377740..3f350db89943d346518237cceab70a0b3bae2ebd 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 e89aea7d42881c8ec0274790be805565c261f990..f7e17421dc21ae27bc73337a02c377462aa05bbe 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 0000000000000000000000000000000000000000..a86f7cc403485b140d74300d6710cd82f15e1d75 --- /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 f61c76b771c058ca1e0500efe5f468fc6b49f7f7..b788fed51d64924c3eda476d55bedeb06fcbe0ad 100644 --- a/plugins/oceanbase/3.1.0/major_freeze.py +++ b/plugins/oceanbase/3.1.0/major_freeze.py @@ -25,34 +25,32 @@ import time 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 0523778ffc0e71bfc88da358054384a567670696..d0af70b078c8aa60885142a68e19e58bfc3a4331 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 b10bffae9f6c3ade2d4c1fb9a2856aeb9fa4d3df..1f55c1d062a2e295a4c13754235cbdc0b187c550 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 d0021af6efb27cbf3f830ac3eadaeb128544d4f7..385ed9f45f7c2d483650e3da2297ffd997f2883e 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 22a57759f376074bef896a032bc4936a9bc9268e..a36f71fceef1e4f4173f0ab775c37fd19fe3dc8c 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 62d3441baf10fde8902a9beec2d8786851f4950f..c401db31660bb5edb573f725b23cc23552b063b7 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 @@ class EnvVariables(object): 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 fe757f86a373d179fd3b06a4d122a235b775868c..c5e9f47b9f042e8417d233c05bf3a552151085fc 100644 --- a/plugins/oceanbase/3.1.0/start_check.py +++ b/plugins/oceanbase/3.1.0/start_check.py @@ -24,11 +24,7 @@ import os 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 @@ success = True 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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 740aa1cab38ab8f064ec53a87dd2c6c9851f85c0..ec70b9bfc9d8a39484d2b4473e19d5245cd82540 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 911b60c7e787f2f59cd11ea3a19823e7a488ce41..03a6132b451ac2bba58fb18ae6e9a88ec66527a1 100644 --- a/plugins/oceanbase/3.1.0/upgrade.py +++ b/plugins/oceanbase/3.1.0/upgrade.py @@ -205,11 +205,18 @@ class Upgrader(object): 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 @@ class Upgrader(object): 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 @@ class Upgrader(object): 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 @@ class Upgrader(object): 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 @@ class Upgrader(object): 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 @@ class Upgrader(object): 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 42ca225f2faf58e1109529bc65baa32b2a0d2af1..4fdb3c587119c7bc8c841a7125288688d08e940e 100644 --- a/plugins/oceanbase/3.1.0/upgrade_check.py +++ b/plugins/oceanbase/3.1.0/upgrade_check.py @@ -23,21 +23,9 @@ from __future__ import absolute_import, division, print_function 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 upgrade_check(plugin_context, current_repository, repositories, route, curso 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 1b909da181ced94b59bd0f5813cd76f30dd9f354..bebcd0c4ea9cff6178d1eab49f50d141c6cfb3de 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 @@ from __future__ import absolute_import, division, print_function 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 6ebfa141b348a9ee749159ac19e8bd8a41faa701..111cd5b2c74e18d1a6fb8d6844a5d521ba2ef98b 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 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) + 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 6315ccd3ab201ac65a2e8d46a443e7bd8f5a1efd..09cf244e88c8d7219496047647cd7ebd44e2c571 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 @@ import time 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 create_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 - 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 create_tenant(plugin_context, cursor, *args, **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 create_tenant(plugin_context, cursor, *args, **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 2ae16b11e06a750e1716b00fa6baf2b33f4c61bf..2d4261e494ee4cd7cce27add17caa326fdb39fc2 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 drop_tenant(plugin_context, cursor, *args, **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 03271cca94a9241bf951a62a5d626f4bafce12f7..8b4d13c0b494d2d98cfcc1abf65372554920c164 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 @@ from __future__ import absolute_import, division, print_function 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 e396fccd88a874c021800b835313bc0d664e6401..668b6013b1efabf98eafb013ca896b31ed9e535a 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 0000000000000000000000000000000000000000..a200012172864076c367139b1b7b61cf167c758a --- /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 55844f5adc158ade09f648fd3fe689cbd139ab76..6e6b94c2ea3f8357bbbafb976f4d9c5faafb47d3 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 @@ import time 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 b45643153a75df6a54995fe8506d07355d4a0990..44eef6c3be3378e3fc1e95bd3fc46bd60190d948 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 fc00ca6c1e9d77650d2f239dab5259376fed6ff9..e449f100a27a330a9fc3ce94b524db677d266c1f 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 605d8c240c9e1daaaad05340abd8cb48d574360c..08d64cda31fa4ec0735dc0fa23196d5d38a1c3fe 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 @@ class EnvVariables(object): 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 98b975844213d2844ef314799b307133667d9850..f3df2ae4fc68b0092f53857d439d7d56a5b7262b 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 @@ from __future__ import absolute_import, division, print_function 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 @@ success = True 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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 _start_check(plugin_context, strict_check=False, *args, **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 0000000000000000000000000000000000000000..3b146dd63e4e947ccf8a1957c661846ce00ef407 --- /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 0000000000000000000000000000000000000000..891b92f786b06f0c6fae9d337714f959e00a7245 --- /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 0000000000000000000000000000000000000000..f73e406a06635f6cb27dca8ff77ddc0082e0018a --- /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 0000000000000000000000000000000000000000..8c6bda34b329160b4a6b1bb632de07a556963d8c --- /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 0000000000000000000000000000000000000000..ffd77eff178e2e0d2f65a8b31864dba8f0ed7a96 --- /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 0000000000000000000000000000000000000000..e4cb51f659f12df6af2e029c0ffa7d0edd23448f --- /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 0000000000000000000000000000000000000000..41844a1ff7ea1b6ef08f4c71cb0c3bae5813748e --- /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 0000000000000000000000000000000000000000..f43e51aeae8226d54bc47b60bffc6e7fcbea1544 --- /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 0000000000000000000000000000000000000000..0f3757a19ec020e32b5eb204d71a0e0db41f3aa8 --- /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 0000000000000000000000000000000000000000..5006a19e06b357dea89fe86f2f6b785ab72f9c3f --- /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 0000000000000000000000000000000000000000..071289471acce2b0150f22bb94c5d1aaba0a5f52 --- /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 0000000000000000000000000000000000000000..9de2d40aa524ae8e593d8bba47e644986a3c399d --- /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 0000000000000000000000000000000000000000..b2d245cd6954aad416ab004f05bfe3dff5b24637 --- /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 0000000000000000000000000000000000000000..d654a134b5c2d19644c33976754f4a08edd2e635 --- /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 0000000000000000000000000000000000000000..72b7aabf99f130f4a3c36f0ad27f3ad2ccdc9d04 --- /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 0000000000000000000000000000000000000000..c7e98bf92f38612a62e64f8289c1c9edb10491b7 --- /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 1940f8210eaa74ebdd4eb9f56799d2cbdbed6d95..bff435aaaa3dded1dc750e893ee7fa289eb5a31e 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 c4ddebc891d11b300b9ee140847d11c328ace89c..f2eef6eefaa0622b7fcb72b5461f08bef57fa27d 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 2b0828c2ce7c25891fa9b47901876bb9ea8d1e89..61cc0b98d45e5a4b2594c9c9516a58c76d8abc4e 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 cb45acaf2b7d7685a7d665560216a76f35c6df57..e7d070d15d3134ef5856c224c73db579fa21ee4d 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 0be45a3eeaad4d51aac4ab53368bd183d4a76de7..5c6ccc49eb6fbc827b25c31ff5b83ee9f8e562ee 100644 --- a/plugins/prometheus/2.37.1/restart.py +++ b/plugins/prometheus/2.37.1/restart.py @@ -29,11 +29,22 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 @@ class Restart(object): 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 042aa606ef258eef6ba49ed4c4789b5882cb8745..878ec1af792547e299e19042945123c2b9cc9d8b 100644 --- a/plugins/prometheus/2.37.1/start.py +++ b/plugins/prometheus/2.37.1/start.py @@ -28,6 +28,7 @@ from copy import deepcopy 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 start(plugin_context, local_home_path, repository_dir, repository_dir_map=No 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 start(plugin_context, local_home_path, repository_dir, repository_dir_map=No 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 edb915ecbd05f2e21bd0def1985a815bf1359bbb..9192687564dea71cc29fb699c59626ef95f49a18 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 @@ success = True 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 start_check(plugin_context, strict_check=False, *args, **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 start_check(plugin_context, strict_check=False, *args, **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 start_check(plugin_context, strict_check=False, *args, **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 cb75c84a087156e1f9914d01a8e80b20f3d0f9e7..d8a4f7d013cc8799eb177bee870add3bcc80db1c 100644 --- a/plugins/prometheus/2.37.1/stop.py +++ b/plugins/prometheus/2.37.1/stop.py @@ -26,7 +26,7 @@ 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 + 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 a7a7832692b32d39cb2e2e890371158b9f4d3b3e..3098562ab0ed1f181c442ad1dc4fc6f6c8068180 100644 --- a/plugins/sysbench/3.1.0/pre_test.py +++ b/plugins/sysbench/3.1.0/pre_test.py @@ -65,18 +65,6 @@ def pre_test(plugin_context, cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 5f03528a2efeef32f2b4280cb8e605ecd4c48559..6c1a52f3975d1ab42d367f44f3a47b648d52eb5d 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 85443477f0407eeb372f17d6cc187896ac262c35..dfc02d2ce5fa0334b5e893b0f50498a4989e3181 100644 --- a/plugins/tpcc/3.1.0/build.py +++ b/plugins/tpcc/3.1.0/build.py @@ -43,18 +43,6 @@ def build(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 build(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 cee1c0c642ebcaca5d454796b3cdd43d0bf1d6e7..30ac96811e1759fe6467be59d889c3728bc0dc87 100644 --- a/plugins/tpcc/3.1.0/pre_test.py +++ b/plugins/tpcc/3.1.0/pre_test.py @@ -59,18 +59,6 @@ def pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 1a58ba8d95d8ea1836281bbb886806cd822d4226..e6526fcabe02fc6f6669b5e3b6e533642a5b319b 100644 --- a/plugins/tpcc/3.1.0/run_test.py +++ b/plugins/tpcc/3.1.0/run_test.py @@ -43,18 +43,6 @@ def run_test(plugin_context, cursor, odp_cursor=None, *args, **kwargs): 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 run_test(plugin_context, cursor, odp_cursor=None, *args, **kwargs): 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 e180f5b416185c9198020ddd43c98a268b9a701a..d75e11a28253a6fe74027190b182d85744465686 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 ef80663a42f64d45caa7856de60fbe4536cd6070..38214dfcc6bceb5ad7968bfe473d7c2507f14aad 100644 --- a/plugins/tpcc/4.0.0.0/build.py +++ b/plugins/tpcc/4.0.0.0/build.py @@ -42,18 +42,6 @@ def build(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 build(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 build(plugin_context, cursor, odp_cursor, *args, **kwargs): # 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 85848a1ec44ae280626f7a2b2be72b6ddebc5c57..6ea230bc5c020451729f23da0132a5467459fd97 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, odp_cursor, *args, **kwargs): 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 290f60c006e992a84340cac559cad14229cc64e5..57f46960a647b5220f2844d0052bc52d60227cb5 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 run_test(plugin_context, cursor, odp_cursor=None, *args, **kwargs): 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 run_test(plugin_context, cursor, odp_cursor=None, *args, **kwargs): # 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 run_test(plugin_context, cursor, odp_cursor=None, *args, **kwargs): 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 f6707b41725d4d318935a01b90a1bea52f9e26e4..def7161366eb2833de667d4d36081ec9594aec5a 100644 --- a/plugins/tpch/3.1.0/pre_test.py +++ b/plugins/tpch/3.1.0/pre_test.py @@ -68,18 +68,6 @@ def pre_test(plugin_context, cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 5dcfcbde1a194d4b32a42fbf6099b3a6be3de183..bc50c5b278a524d34749813575cdc07b8240f59a 100644 --- a/plugins/tpch/3.1.0/run_test.py +++ b/plugins/tpch/3.1.0/run_test.py @@ -62,17 +62,6 @@ def run_test(plugin_context, db, cursor, *args, **kwargs): 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 run_test(plugin_context, db, cursor, *args, **kwargs): 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 run_test(plugin_context, db, cursor, *args, **kwargs): 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 99946b3d1a293d91116f75471c2b837b30a36b53..1e2c7587c3e7921e88443d52c300737016b109f7 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 2fa8a5b5bb35e2824ba9d56acac7b93b38886ed2..f42457a327d787eceddce449a88c9f0a5133f1f9 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 pre_test(plugin_context, cursor, *args, **kwargs): 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 64264fb09509acb31e9772996dbd323ff92a2066..749253e2ab6175fd4a513bc784e6366cb6dfd94a 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 run_test(plugin_context, db, cursor, *args, **kwargs): 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 run_test(plugin_context, db, cursor, *args, **kwargs): 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 run_test(plugin_context, db, cursor, *args, **kwargs): # 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 run_test(plugin_context, db, cursor, *args, **kwargs): 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 1aa5028e9242bdd3b0f20bdc9997fae56b6da464..24cdb0937d9369b739b90a3d73c90003794f8ef2 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 71ef69408219d163b4b90293d2e295ea2175c0be..3216a2adc9f585efb62f55725d5709dce529a396 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 4fb1551c4f53154fc7359bca97c502243727807a..99be94e5c5d42aebbc1506a8bdeaf8cc83d4f24d 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 46fd4e218a0973187ac4992b18bf2419222c2d67..1bd2c186c9d34ff018d7ecdeb5d534cdd4d491a4 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 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/service/api/__init__.py b/service/api/__init__.py new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/service/api/response.py b/service/api/response.py new file mode 100644 index 0000000000000000000000000000000000000000..8447d76ec9267c4899c3a967ff8eec24f8dee178 --- /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 0000000000000000000000000000000000000000..629bf419147501b06d2e8095835eb99c4d802082 --- /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 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/service/api/v1/components.py b/service/api/v1/components.py new file mode 100644 index 0000000000000000000000000000000000000000..a47d3252093693e9b1cd0dea1f6b7bda54b0ed58 --- /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 0000000000000000000000000000000000000000..2194d210367431537d11e6c7c21e2c5dec90241f --- /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 0000000000000000000000000000000000000000..043ae7d2a2b3732bde7fef398fed90efe5a3eabc --- /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 0000000000000000000000000000000000000000..a10d2f3c4fbe3c460f4411b8cf530144cd2d02ac --- /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 0000000000000000000000000000000000000000..0ffe2003d559e89bb37419623327c0d6e32b254e --- /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 0000000000000000000000000000000000000000..bd9f0d3013c198f7045f0d41bb06aaf55c52a8bc --- /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 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/service/common/const.py b/service/common/const.py new file mode 100644 index 0000000000000000000000000000000000000000..f47c33464c3989e4a62b1b5525b43ef74921b282 --- /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 0000000000000000000000000000000000000000..e94f8c94918536cf3cb1cda45df1aacaa85ff706 --- /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 0000000000000000000000000000000000000000..a2fd67b4204b1b715a694cc8cb4c195d572ea78a --- /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 0000000000000000000000000000000000000000..a22968fac21c5c6a472369d3c2be5dc41fb65683 --- /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 0000000000000000000000000000000000000000..9b9151bfec7b66a094c2eaee86aba1095f37b3b5 --- /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 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/service/handler/base_handler.py b/service/handler/base_handler.py new file mode 100644 index 0000000000000000000000000000000000000000..56993a51e3aefb69a5b842ae03d8a1eaef204f2f --- /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 0000000000000000000000000000000000000000..1d55e6afaf287ebb2f78d07493bfa3a9263f2878 --- /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 0000000000000000000000000000000000000000..93e6e6a03b18e1c84e7b0fd4e60f48027204df67 --- /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 0000000000000000000000000000000000000000..b95fa3f553f838bac76f2b49cff5b30421ca19f2 --- /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 0000000000000000000000000000000000000000..ab8a0ffc096fe390752c9f95b92436c7ed2db7df --- /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 0000000000000000000000000000000000000000..784f70863886cfe7e71b1cae22ad70963bcbb105 --- /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 0000000000000000000000000000000000000000..0dc0687f614849c86526d2cf4af665c74da3736e --- /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 0000000000000000000000000000000000000000..fcd662c8539dbd646ce80c309e9420f618d4b5a3 --- /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 0000000000000000000000000000000000000000..2230ccd34622fb2849f05287d6cffbf0f360e527 --- /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 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/service/model/components.py b/service/model/components.py new file mode 100644 index 0000000000000000000000000000000000000000..4f5a702754ec2544b3429ba4099a20003e715045 --- /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 0000000000000000000000000000000000000000..ee733337366e1a10e205b0f28cb3e3d38a98165b --- /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 0000000000000000000000000000000000000000..c9b711ba9493ee91a473d4cbb732cf342b1f4382 --- /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 0000000000000000000000000000000000000000..327d98f746e2cb5e63763340b3beb175f903851d --- /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 0000000000000000000000000000000000000000..13b97c5136ffb3e294841043b2c6cdb844bcdaea --- /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 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/ssh.py b/ssh.py index a076b5b50d593ae4f8e951bb0369b2cbd50d1443..5eacbce59bd5fee6f9b73bef6a17ccfe4d777557 100644 --- a/ssh.py +++ b/ssh.py @@ -42,6 +42,7 @@ from multiprocessing.pool import ThreadPool 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 @@ class SshClient(SafeStdio): 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 @@ class SshClient(SafeStdio): 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 @@ class SshClient(SafeStdio): 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 @@ class SshClient(SafeStdio): 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 15f9941fb2e4a44db7110d978ecc34ed3f3e33bb..a09fa59ae3dfe2f9d180542938ea904198d55ba5 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 shutil import re import json import hashlib +import socket from io import BytesIO from ruamel.yaml import YAML, YAMLContextManager, representer @@ -132,7 +133,7 @@ class DynamicLoading(object): 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 @@ class DynamicLoading(object): 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 @@ class ConfigUtil(object): 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 @@ class FileUtil(object): return False else: raise IOError(info) - + try: if os.path.islink(src): FileUtil.symlink(os.readlink(src), dst) @@ -438,7 +443,7 @@ class FileUtil(object): 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 @@ class YamlLoader(YAML): 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 @@ class CommandEnv(SafeStdio): 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 0000000000000000000000000000000000000000..7e3649acc2c165b62750e2ca02b80f8ee0da6c4d --- /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 0000000000000000000000000000000000000000..4594b11a3dd72ab3d99652ed60910e470f90773b --- /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 0000000000000000000000000000000000000000..25b3bfc83e7fe1f4e84ee2889f0e83723923f650 --- /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 0000000000000000000000000000000000000000..94beb148408f6876d8b8c8f960f7dcc45c63e936 --- /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 0000000000000000000000000000000000000000..07afeb7fd6d717d5a30fc7453c91a0f7a8ee71ba --- /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 0000000000000000000000000000000000000000..4447dfc7fce81a4c2491c2267cce898d065b2da1 --- /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 0000000000000000000000000000000000000000..83b53b87b0c296e0da6cf2888839de1d22b2387f --- /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 0000000000000000000000000000000000000000..ee15b71a2c9cfecae20ee0a5e5004842a7dd23dc --- /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 0000000000000000000000000000000000000000..41b8198e6ad9d9e0732cb6eb525f0be37124f9fd --- /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 0000000000000000000000000000000000000000..6f38273ec59b8197bb9d3e6ca25eede262fbe973 --- /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 0000000000000000000000000000000000000000..b6436992ac2e41b16ef077399e38747bc5318b8e --- /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 0000000000000000000000000000000000000000..181a339493c79e5211909dbdd8be674b805bcbd7 --- /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 0000000000000000000000000000000000000000..a220a96e8a3dbd499f4b2e92ea1ac25e4ec377c1 --- /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 0000000000000000000000000000000000000000..02e073303226514e216e19868c7936f6de2ac3d1 --- /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 0000000000000000000000000000000000000000..2d42b1fde6a5e9195fc7f9b7b2d07cf59d8feffb --- /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 0000000000000000000000000000000000000000..e1c33a4e4383bf948deafca6404bc1979058919d --- /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 0000000000000000000000000000000000000000..16723dd9566ad9eb834be53e2afaff2aa89642a5 --- /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 0000000000000000000000000000000000000000..28e29ba3571943274d05fc793a0063306f4ee215 --- /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 0000000000000000000000000000000000000000..d147803b8bfb6ca3150b488a3257035cfcb75312 --- /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 0000000000000000000000000000000000000000..670f393f79172d707ad31c2f791f841c14779915 --- /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 0000000000000000000000000000000000000000..acbeefa2c2f2589e3f829b915b0813349aede977 --- /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 0000000000000000000000000000000000000000..b1e56e5cbf48c9ef409a92afd7a54cee1b9b5d81 --- /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 0000000000000000000000000000000000000000..b1b15c20e76ab2f365fa3aaca654d0d658876daa --- /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 0000000000000000000000000000000000000000..0cfc5c88ab4c75edf2b3e78637b8e6dd4de56215 --- /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 0000000000000000000000000000000000000000..abe51f331485052aa493aeee48a55e0fba569e36 --- /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 0000000000000000000000000000000000000000..4f6d9684b9a8d5a9649d7d2788f0ea858492bd0c --- /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 0000000000000000000000000000000000000000..2daeceede68fd09a4c859be9b6c00c8b6baf4a11 --- /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 0000000000000000000000000000000000000000..c9eaacbfd2a03b61aac15775656e93ac1e9c8f35 --- /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 0000000000000000000000000000000000000000..40c542a83c3672c4875d332601282289a6d61691 --- /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 Binary files /dev/null and b/web/public/assets/empty.png differ diff --git a/web/public/assets/failed.png b/web/public/assets/failed.png new file mode 100644 index 0000000000000000000000000000000000000000..0ba7db0952ec7fa71158ae0a287d5e8f2bd37b89 Binary files /dev/null and b/web/public/assets/failed.png differ diff --git a/web/public/assets/logo.png b/web/public/assets/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..f4436f966e5e78a949d533cde4f201ab80950672 Binary files /dev/null and b/web/public/assets/logo.png differ diff --git a/web/public/assets/oceanbase.png b/web/public/assets/oceanbase.png new file mode 100644 index 0000000000000000000000000000000000000000..b79b525f6b664678fe22a176d979daa8b73d599d Binary files /dev/null and b/web/public/assets/oceanbase.png differ diff --git a/web/public/assets/progress/data.json b/web/public/assets/progress/data.json new file mode 100644 index 0000000000000000000000000000000000000000..28baca6aa953025349cf70777dbe6ea5c73b9a87 --- /dev/null +++ b/web/public/assets/progress/data.json @@ -0,0 +1,5269 @@ +{ + "v": "5.9.6", + "fr": 30, + "ip": 0, + "op": 201, + "w": 500, + "h": 125, + "nm": "seq_0", + "ddd": 0, + "assets": [ + { + "id": "imgSeq_0", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_1", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_2", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_3", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_4", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_5", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_6", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_7", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_8", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_9", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_10", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_11", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_12", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_13", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_14", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_15", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_16", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_17", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_18", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_19", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_20", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_21", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_22", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_23", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_24", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_25", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_26", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_27", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_28", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_29", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_30", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_31", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_32", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_33", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_34", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_35", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_36", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_37", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_38", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_39", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_40", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_41", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_42", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_43", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_44", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_45", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_46", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_47", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_48", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_49", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_50", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_51", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_52", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_53", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_54", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_55", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_56", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_57", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_58", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_59", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_60", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_61", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_62", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_63", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_64", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_65", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_66", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_67", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_68", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_69", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_70", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_71", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_72", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_73", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_74", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_75", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_76", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_77", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_78", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_79", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_80", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_81", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_82", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_83", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_84", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_85", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_86", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_87", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_88", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_89", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_90", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_91", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_92", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_93", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_94", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_95", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_96", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_97", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_98", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_99", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_100", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_101", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_102", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_103", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_104", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_105", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_106", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_107", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_108", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_109", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_110", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_111", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_112", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_113", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_114", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_115", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_116", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_117", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_118", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_119", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_120", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_121", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_122", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_123", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_124", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_125", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_126", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_127", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_128", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_129", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_130", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_131", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_132", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_133", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_134", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_135", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_136", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_137", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_138", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_139", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_140", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_141", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_142", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_143", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_144", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_145", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_146", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_147", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_148", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_149", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_150", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_151", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_152", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_153", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_154", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_155", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_156", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_157", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_158", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_159", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_160", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_161", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_162", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_163", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_164", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_165", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_166", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_167", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_168", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_169", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_170", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_171", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_172", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_173", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_174", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_175", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_176", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_177", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_178", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_179", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_180", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_181", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_182", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_183", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_184", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_185", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_186", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_187", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_188", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_189", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_190", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_191", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_192", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_193", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_194", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_195", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_196", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_197", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_198", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_199", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_200", + "w": 500, + "h": 125, + "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": 80, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_80", + "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": 80, + "st": 80, + "op": 81, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_81", + "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": 81, + "st": 81, + "op": 82, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_82", + "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": 82, + "st": 82, + "op": 83, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_83", + "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": 83, + "st": 83, + "op": 84, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_84", + "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": 84, + "st": 84, + "op": 85, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_85", + "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": 85, + "st": 85, + "op": 86, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_86", + "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": 86, + "st": 86, + "op": 87, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_87", + "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": 87, + "st": 87, + "op": 88, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_88", + "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": 88, + "st": 88, + "op": 89, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_89", + "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": 89, + "st": 89, + "op": 90, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_90", + "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": 90, + "st": 90, + "op": 91, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_91", + "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": 91, + "st": 91, + "op": 92, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_92", + "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": 92, + "st": 92, + "op": 93, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_93", + "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": 93, + "st": 93, + "op": 94, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_94", + "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": 94, + "st": 94, + "op": 95, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_95", + "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": 95, + "st": 95, + "op": 96, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_96", + "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": 96, + "st": 96, + "op": 97, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_97", + "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": 97, + "st": 97, + "op": 98, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_98", + "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": 98, + "st": 98, + "op": 99, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_99", + "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": 99, + "st": 99, + "op": 100, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_100", + "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": 100, + "st": 100, + "op": 101, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_101", + "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": 101, + "st": 101, + "op": 102, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_102", + "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": 102, + "st": 102, + "op": 103, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_103", + "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": 103, + "st": 103, + "op": 104, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_104", + "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": 104, + "st": 104, + "op": 105, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_105", + "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": 105, + "st": 105, + "op": 106, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_106", + "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": 106, + "st": 106, + "op": 107, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_107", + "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": 107, + "st": 107, + "op": 108, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_108", + "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": 108, + "st": 108, + "op": 109, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_109", + "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": 109, + "st": 109, + "op": 110, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_110", + "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": 110, + "st": 110, + "op": 111, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_111", + "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": 111, + "st": 111, + "op": 112, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_112", + "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": 112, + "st": 112, + "op": 113, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_113", + "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": 113, + "st": 113, + "op": 114, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_114", + "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": 114, + "st": 114, + "op": 115, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_115", + "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": 115, + "st": 115, + "op": 116, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_116", + "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": 116, + "st": 116, + "op": 117, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_117", + "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": 117, + "st": 117, + "op": 118, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_118", + "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": 118, + "st": 118, + "op": 119, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_119", + "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": 119, + "st": 119, + "op": 120, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_120", + "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": 120, + "st": 120, + "op": 121, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_121", + "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": 121, + "st": 121, + "op": 122, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_122", + "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": 122, + "st": 122, + "op": 123, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_123", + "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": 123, + "st": 123, + "op": 124, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_124", + "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": 124, + "st": 124, + "op": 125, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_125", + "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": 125, + "st": 125, + "op": 126, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_126", + "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": 126, + "st": 126, + "op": 127, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_127", + "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": 127, + "st": 127, + "op": 128, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_128", + "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": 128, + "st": 128, + "op": 129, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_129", + "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": 129, + "st": 129, + "op": 130, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_130", + "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": 130, + "st": 130, + "op": 131, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_131", + "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": 131, + "st": 131, + "op": 132, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_132", + "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": 132, + "st": 132, + "op": 133, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_133", + "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": 133, + "st": 133, + "op": 134, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_134", + "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": 134, + "st": 134, + "op": 135, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_135", + "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": 135, + "st": 135, + "op": 136, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_136", + "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": 136, + "st": 136, + "op": 137, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_137", + "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": 137, + "st": 137, + "op": 138, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_138", + "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": 138, + "st": 138, + "op": 139, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_139", + "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": 139, + "st": 139, + "op": 140, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_140", + "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": 140, + "st": 140, + "op": 141, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_141", + "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": 141, + "st": 141, + "op": 142, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_142", + "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": 142, + "st": 142, + "op": 143, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_143", + "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": 143, + "st": 143, + "op": 144, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_144", + "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": 144, + "st": 144, + "op": 145, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_145", + "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": 145, + "st": 145, + "op": 146, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_146", + "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": 146, + "st": 146, + "op": 147, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_147", + "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": 147, + "st": 147, + "op": 148, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_148", + "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": 148, + "st": 148, + "op": 149, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_149", + "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": 149, + "st": 149, + "op": 150, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_150", + "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": 150, + "st": 150, + "op": 151, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_151", + "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": 151, + "st": 151, + "op": 152, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_152", + "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": 152, + "st": 152, + "op": 153, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_153", + "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": 153, + "st": 153, + "op": 154, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_154", + "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": 154, + "st": 154, + "op": 155, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_155", + "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": 155, + "st": 155, + "op": 156, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_156", + "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": 156, + "st": 156, + "op": 157, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_157", + "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": 157, + "st": 157, + "op": 158, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_158", + "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": 158, + "st": 158, + "op": 159, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_159", + "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": 159, + "st": 159, + "op": 160, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_160", + "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": 160, + "st": 160, + "op": 161, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_161", + "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": 161, + "st": 161, + "op": 162, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_162", + "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": 162, + "st": 162, + "op": 163, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_163", + "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": 163, + "st": 163, + "op": 164, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_164", + "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": 164, + "st": 164, + "op": 165, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_165", + "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": 165, + "st": 165, + "op": 166, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_166", + "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": 166, + "st": 166, + "op": 167, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_167", + "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": 167, + "st": 167, + "op": 168, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_168", + "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": 168, + "st": 168, + "op": 169, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_169", + "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": 169, + "st": 169, + "op": 170, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_170", + "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": 170, + "st": 170, + "op": 171, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_171", + "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": 171, + "st": 171, + "op": 172, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_172", + "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": 172, + "st": 172, + "op": 173, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_173", + "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": 173, + "st": 173, + "op": 174, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_174", + "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": 174, + "st": 174, + "op": 175, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_175", + "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": 175, + "st": 175, + "op": 176, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_176", + "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": 176, + "st": 176, + "op": 177, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_177", + "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": 177, + "st": 177, + "op": 178, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_178", + "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": 178, + "st": 178, + "op": 179, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_179", + "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": 179, + "st": 179, + "op": 180, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_180", + "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": 180, + "st": 180, + "op": 181, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_181", + "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": 181, + "st": 181, + "op": 182, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_182", + "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": 182, + "st": 182, + "op": 183, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_183", + "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": 183, + "st": 183, + "op": 184, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_184", + "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": 184, + "st": 184, + "op": 185, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_185", + "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": 185, + "st": 185, + "op": 186, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_186", + "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": 186, + "st": 186, + "op": 187, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_187", + "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": 187, + "st": 187, + "op": 188, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_188", + "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": 188, + "st": 188, + "op": 189, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_189", + "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": 189, + "st": 189, + "op": 190, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_190", + "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": 190, + "st": 190, + "op": 191, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_191", + "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": 191, + "st": 191, + "op": 192, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_192", + "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": 192, + "st": 192, + "op": 193, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_193", + "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": 193, + "st": 193, + "op": 194, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_194", + "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": 194, + "st": 194, + "op": 195, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_195", + "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": 195, + "st": 195, + "op": 196, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_196", + "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": 196, + "st": 196, + "op": 197, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_197", + "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": 197, + "st": 197, + "op": 198, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_198", + "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": 198, + "st": 198, + "op": 199, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_199", + "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": 199, + "st": 199, + "op": 200, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_200", + "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": 200, + "st": 200, + "op": 202, + "sr": 1, + "bm": 0 + } + ] + } + ], + "layers": [ + { + "ddd": 0, + "ind": 1, + "ty": 0, + "nm": "seq_0_[0-200].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": [250, 62.5, 0], "ix": 2, "l": 2 }, + "a": { "a": 0, "k": [250, 62.5, 0], "ix": 1, "l": 2 }, + "s": { "a": 0, "k": [100, 100, 100], "ix": 6, "l": 2 } + }, + "ao": 0, + "w": 500, + "h": 125, + "ip": 0, + "op": 201, + "st": 0, + "bm": 0 + } + ], + "markers": [] +} diff --git a/web/public/assets/spaceman/data.json b/web/public/assets/spaceman/data.json new file mode 100644 index 0000000000000000000000000000000000000000..b070000589a5a16a2b24ef5e6b189cbc5467b2b1 --- /dev/null +++ b/web/public/assets/spaceman/data.json @@ -0,0 +1,6075 @@ +{ + "v": "5.9.6", + "fr": 30, + "ip": 0, + "op": 232, + "w": 500, + "h": 125, + "nm": "seq_0", + "ddd": 0, + "assets": [ + { + "id": "imgSeq_0", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_1", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_2", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_3", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_4", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_5", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_6", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_7", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_8", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_9", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_10", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_11", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_12", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_13", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_14", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_15", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_16", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_17", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_18", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_19", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_20", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_21", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_22", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_23", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_24", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_25", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_26", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_27", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_28", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_29", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_30", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_31", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_32", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_33", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_34", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_35", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_36", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_37", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_38", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_39", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_40", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_41", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_42", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_43", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_44", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_45", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_46", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_47", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_48", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_49", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_50", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_51", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_52", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_53", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_54", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_55", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_56", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_57", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_58", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_59", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_60", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_61", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_62", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_63", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_64", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_65", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_66", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_67", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_68", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_69", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_70", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_71", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_72", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_73", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_74", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_75", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_76", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_77", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_78", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_79", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_80", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_81", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_82", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_83", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_84", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_85", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_86", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_87", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_88", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_89", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_90", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_91", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_92", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_93", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_94", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_95", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_96", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_97", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_98", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_99", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_100", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_101", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_102", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_103", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_104", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_105", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_106", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_107", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_108", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_109", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_110", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_111", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_112", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_113", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_114", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_115", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_116", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_117", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_118", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_119", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_120", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_121", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_122", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_123", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_124", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_125", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_126", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_127", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_128", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_129", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_130", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_131", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_132", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_133", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_134", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_135", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_136", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_137", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_138", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_139", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_140", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_141", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_142", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_143", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_144", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_145", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_146", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_147", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_148", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_149", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_150", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_151", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_152", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_153", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_154", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_155", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_156", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_157", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_158", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_159", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_160", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_161", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_162", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_163", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_164", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_165", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_166", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_167", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_168", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_169", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_170", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_171", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_172", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_173", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_174", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_175", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_176", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_177", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_178", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_179", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_180", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_181", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_182", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_183", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_184", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_185", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_186", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_187", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_188", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_189", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_190", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_191", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_192", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_193", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_194", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_195", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_196", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_197", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_198", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_199", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_200", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_201", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_202", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_203", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_204", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_205", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_206", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_207", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_208", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_209", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_210", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_211", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_212", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_213", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_214", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_215", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_216", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_217", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_218", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_219", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_220", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_221", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_222", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_223", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_224", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_225", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_226", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_227", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_228", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_229", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_230", + "w": 500, + "h": 125, + "t": "seq", + "u": "", + "p": "", + "e": 1 + }, + { + "id": "imgSeq_231", + "w": 500, + "h": 125, + "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": 80, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_80", + "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": 80, + "st": 80, + "op": 81, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_81", + "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": 81, + "st": 81, + "op": 82, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_82", + "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": 82, + "st": 82, + "op": 83, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_83", + "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": 83, + "st": 83, + "op": 84, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_84", + "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": 84, + "st": 84, + "op": 85, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_85", + "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": 85, + "st": 85, + "op": 86, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_86", + "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": 86, + "st": 86, + "op": 87, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_87", + "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": 87, + "st": 87, + "op": 88, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_88", + "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": 88, + "st": 88, + "op": 89, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_89", + "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": 89, + "st": 89, + "op": 90, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_90", + "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": 90, + "st": 90, + "op": 91, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_91", + "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": 91, + "st": 91, + "op": 92, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_92", + "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": 92, + "st": 92, + "op": 93, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_93", + "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": 93, + "st": 93, + "op": 94, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_94", + "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": 94, + "st": 94, + "op": 95, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_95", + "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": 95, + "st": 95, + "op": 96, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_96", + "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": 96, + "st": 96, + "op": 97, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_97", + "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": 97, + "st": 97, + "op": 98, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_98", + "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": 98, + "st": 98, + "op": 99, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_99", + "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": 99, + "st": 99, + "op": 100, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_100", + "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": 100, + "st": 100, + "op": 101, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_101", + "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": 101, + "st": 101, + "op": 102, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_102", + "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": 102, + "st": 102, + "op": 103, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_103", + "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": 103, + "st": 103, + "op": 104, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_104", + "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": 104, + "st": 104, + "op": 105, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_105", + "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": 105, + "st": 105, + "op": 106, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_106", + "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": 106, + "st": 106, + "op": 107, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_107", + "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": 107, + "st": 107, + "op": 108, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_108", + "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": 108, + "st": 108, + "op": 109, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_109", + "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": 109, + "st": 109, + "op": 110, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_110", + "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": 110, + "st": 110, + "op": 111, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_111", + "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": 111, + "st": 111, + "op": 112, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_112", + "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": 112, + "st": 112, + "op": 113, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_113", + "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": 113, + "st": 113, + "op": 114, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_114", + "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": 114, + "st": 114, + "op": 115, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_115", + "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": 115, + "st": 115, + "op": 116, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_116", + "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": 116, + "st": 116, + "op": 117, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_117", + "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": 117, + "st": 117, + "op": 118, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_118", + "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": 118, + "st": 118, + "op": 119, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_119", + "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": 119, + "st": 119, + "op": 120, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_120", + "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": 120, + "st": 120, + "op": 121, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_121", + "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": 121, + "st": 121, + "op": 122, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_122", + "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": 122, + "st": 122, + "op": 123, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_123", + "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": 123, + "st": 123, + "op": 124, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_124", + "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": 124, + "st": 124, + "op": 125, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_125", + "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": 125, + "st": 125, + "op": 126, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_126", + "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": 126, + "st": 126, + "op": 127, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_127", + "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": 127, + "st": 127, + "op": 128, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_128", + "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": 128, + "st": 128, + "op": 129, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_129", + "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": 129, + "st": 129, + "op": 130, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_130", + "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": 130, + "st": 130, + "op": 131, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_131", + "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": 131, + "st": 131, + "op": 132, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_132", + "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": 132, + "st": 132, + "op": 133, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_133", + "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": 133, + "st": 133, + "op": 134, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_134", + "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": 134, + "st": 134, + "op": 135, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_135", + "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": 135, + "st": 135, + "op": 136, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_136", + "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": 136, + "st": 136, + "op": 137, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_137", + "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": 137, + "st": 137, + "op": 138, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_138", + "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": 138, + "st": 138, + "op": 139, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_139", + "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": 139, + "st": 139, + "op": 140, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_140", + "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": 140, + "st": 140, + "op": 141, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_141", + "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": 141, + "st": 141, + "op": 142, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_142", + "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": 142, + "st": 142, + "op": 143, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_143", + "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": 143, + "st": 143, + "op": 144, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_144", + "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": 144, + "st": 144, + "op": 145, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_145", + "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": 145, + "st": 145, + "op": 146, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_146", + "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": 146, + "st": 146, + "op": 147, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_147", + "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": 147, + "st": 147, + "op": 148, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_148", + "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": 148, + "st": 148, + "op": 149, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_149", + "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": 149, + "st": 149, + "op": 150, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_150", + "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": 150, + "st": 150, + "op": 151, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_151", + "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": 151, + "st": 151, + "op": 152, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_152", + "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": 152, + "st": 152, + "op": 153, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_153", + "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": 153, + "st": 153, + "op": 154, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_154", + "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": 154, + "st": 154, + "op": 155, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_155", + "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": 155, + "st": 155, + "op": 156, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_156", + "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": 156, + "st": 156, + "op": 157, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_157", + "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": 157, + "st": 157, + "op": 158, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_158", + "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": 158, + "st": 158, + "op": 159, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_159", + "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": 159, + "st": 159, + "op": 160, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_160", + "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": 160, + "st": 160, + "op": 161, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_161", + "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": 161, + "st": 161, + "op": 162, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_162", + "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": 162, + "st": 162, + "op": 163, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_163", + "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": 163, + "st": 163, + "op": 164, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_164", + "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": 164, + "st": 164, + "op": 165, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_165", + "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": 165, + "st": 165, + "op": 166, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_166", + "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": 166, + "st": 166, + "op": 167, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_167", + "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": 167, + "st": 167, + "op": 168, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_168", + "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": 168, + "st": 168, + "op": 169, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_169", + "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": 169, + "st": 169, + "op": 170, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_170", + "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": 170, + "st": 170, + "op": 171, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_171", + "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": 171, + "st": 171, + "op": 172, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_172", + "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": 172, + "st": 172, + "op": 173, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_173", + "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": 173, + "st": 173, + "op": 174, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_174", + "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": 174, + "st": 174, + "op": 175, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_175", + "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": 175, + "st": 175, + "op": 176, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_176", + "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": 176, + "st": 176, + "op": 177, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_177", + "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": 177, + "st": 177, + "op": 178, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_178", + "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": 178, + "st": 178, + "op": 179, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_179", + "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": 179, + "st": 179, + "op": 180, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_180", + "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": 180, + "st": 180, + "op": 181, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_181", + "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": 181, + "st": 181, + "op": 182, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_182", + "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": 182, + "st": 182, + "op": 183, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_183", + "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": 183, + "st": 183, + "op": 184, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_184", + "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": 184, + "st": 184, + "op": 185, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_185", + "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": 185, + "st": 185, + "op": 186, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_186", + "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": 186, + "st": 186, + "op": 187, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_187", + "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": 187, + "st": 187, + "op": 188, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_188", + "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": 188, + "st": 188, + "op": 189, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_189", + "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": 189, + "st": 189, + "op": 190, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_190", + "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": 190, + "st": 190, + "op": 191, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_191", + "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": 191, + "st": 191, + "op": 192, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_192", + "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": 192, + "st": 192, + "op": 193, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_193", + "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": 193, + "st": 193, + "op": 194, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_194", + "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": 194, + "st": 194, + "op": 195, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_195", + "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": 195, + "st": 195, + "op": 196, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_196", + "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": 196, + "st": 196, + "op": 197, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_197", + "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": 197, + "st": 197, + "op": 198, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_198", + "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": 198, + "st": 198, + "op": 199, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_199", + "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": 199, + "st": 199, + "op": 200, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_200", + "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": 200, + "st": 200, + "op": 201, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_201", + "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": 201, + "st": 201, + "op": 202, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_202", + "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": 202, + "st": 202, + "op": 203, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_203", + "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": 203, + "st": 203, + "op": 204, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_204", + "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": 204, + "st": 204, + "op": 205, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_205", + "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": 205, + "st": 205, + "op": 206, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_206", + "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": 206, + "st": 206, + "op": 207, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_207", + "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": 207, + "st": 207, + "op": 208, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_208", + "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": 208, + "st": 208, + "op": 209, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_209", + "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": 209, + "st": 209, + "op": 210, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_210", + "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": 210, + "st": 210, + "op": 211, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_211", + "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": 211, + "st": 211, + "op": 212, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_212", + "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": 212, + "st": 212, + "op": 213, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_213", + "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": 213, + "st": 213, + "op": 214, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_214", + "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": 214, + "st": 214, + "op": 215, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_215", + "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": 215, + "st": 215, + "op": 216, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_216", + "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": 216, + "st": 216, + "op": 217, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_217", + "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": 217, + "st": 217, + "op": 218, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_218", + "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": 218, + "st": 218, + "op": 219, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_219", + "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": 219, + "st": 219, + "op": 220, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_220", + "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": 220, + "st": 220, + "op": 221, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_221", + "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": 221, + "st": 221, + "op": 222, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_222", + "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": 222, + "st": 222, + "op": 223, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_223", + "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": 223, + "st": 223, + "op": 224, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_224", + "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": 224, + "st": 224, + "op": 225, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_225", + "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": 225, + "st": 225, + "op": 226, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_226", + "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": 226, + "st": 226, + "op": 227, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_227", + "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": 227, + "st": 227, + "op": 228, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_228", + "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": 228, + "st": 228, + "op": 229, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_229", + "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": 229, + "st": 229, + "op": 230, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_230", + "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": 230, + "st": 230, + "op": 231, + "sr": 1, + "bm": 0 + }, + { + "ty": 2, + "sc": "#00ffff", + "refId": "imgSeq_231", + "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": 231, + "st": 231, + "op": 233, + "sr": 1, + "bm": 0 + } + ] + } + ], + "layers": [ + { + "ddd": 0, + "ind": 1, + "ty": 0, + "nm": "seq_0_[0-231].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": [250, 62.5, 0], "ix": 2, "l": 2 }, + "a": { "a": 0, "k": [250, 62.5, 0], "ix": 1, "l": 2 }, + "s": { "a": 0, "k": [100, 100, 100], "ix": 6, "l": 2 } + }, + "ao": 0, + "w": 500, + "h": 125, + "ip": 0, + "op": 232, + "st": 0, + "bm": 0 + } + ], + "markers": [] +} diff --git a/web/public/assets/successful.png b/web/public/assets/successful.png new file mode 100644 index 0000000000000000000000000000000000000000..0a94b90d6d9b410c03907725ea428cc31ecb9fd8 Binary files /dev/null and b/web/public/assets/successful.png differ diff --git a/web/public/assets/welcome/cover.jpg b/web/public/assets/welcome/cover.jpg new file mode 100644 index 0000000000000000000000000000000000000000..5f24e1f8fbd7d4190374d1e6331bc9e2d829e740 Binary files /dev/null and b/web/public/assets/welcome/cover.jpg differ diff --git a/web/public/assets/welcome/data.mp4 b/web/public/assets/welcome/data.mp4 new file mode 100644 index 0000000000000000000000000000000000000000..d4b05c01c7ebba0e787a166012d9c795486129e8 Binary files /dev/null and b/web/public/assets/welcome/data.mp4 differ diff --git a/web/src/app.ts b/web/src/app.ts new file mode 100644 index 0000000000000000000000000000000000000000..93129704e86dc5579b7dd41bf6874a234882342b --- /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 0000000000000000000000000000000000000000..7ea066c97fe3850472a1dfadc6fec2a6ff0aa79d --- /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 0000000000000000000000000000000000000000..41a7ab2a8faa688006cc58c9c84824ccc52af9dc --- /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 0000000000000000000000000000000000000000..09b496f733e6926cab1f9b7a69403215d706ba60 --- /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 0000000000000000000000000000000000000000..f0765a0b43d8a90fdb18a4edac64cb8093c5e899 --- /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 0000000000000000000000000000000000000000..3b9e3ec3a7d4b556a1c397f6e32e25c9e3f88a99 --- /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 0000000000000000000000000000000000000000..c46758997631efb8023e4b936c23ecfcd200ed62 --- /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 0000000000000000000000000000000000000000..7669bdeea9f2ad8a5585fcbfdacc86ac496f4ebc --- /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 0000000000000000000000000000000000000000..b70dcb4a308332459c4d35e042bfa221e3efa026 --- /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 0000000000000000000000000000000000000000..fbb2560a5d01d0ff0b3788fa6609d0daa4c76546 --- /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 0000000000000000000000000000000000000000..0a9984e7c7f35973c61610814ebba544df49d6d4 --- /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 0000000000000000000000000000000000000000..a9328477303a47b024316b56eefab47b42e55d4b --- /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 0000000000000000000000000000000000000000..b5983e62c5db9e09c0da495c8b30a801b9bcda5a --- /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 0000000000000000000000000000000000000000..61c1acf31f08f66b6a1813961af0e11486abd76e --- /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 0000000000000000000000000000000000000000..def03a9bfdd64a19e5fa4c1a2513ce2dadd2b691 --- /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 0000000000000000000000000000000000000000..c06092c646dd6d4ae0d900e59b5d51a2679ab81c --- /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 ( + +