From 0e021f9ec99577c02d91663f7785e205cb990361 Mon Sep 17 00:00:00 2001 From: wunder957 Date: Wed, 30 Aug 2023 20:10:04 +0800 Subject: [PATCH 1/6] Improve CLI and monitor on error --- duetector/cli/main.py | 10 ++++++---- duetector/monitors/bcc_monitor.py | 10 ++++++++++ 2 files changed, 16 insertions(+), 4 deletions(-) diff --git a/duetector/cli/main.py b/duetector/cli/main.py index 295e515..9fce4df 100644 --- a/duetector/cli/main.py +++ b/duetector/cli/main.py @@ -143,16 +143,18 @@ def start( m.start_polling() def _shutdown(sig=None, frame=None): + logger.info("Exiting...") for m in monitors: m.shutdown() + for m in monitors: + logger.info(m.summary()) + exit(0) signal.signal(signal.SIGINT, _shutdown) signal.signal(signal.SIGTERM, _shutdown) logger.info("Waiting for KeyboardInterrupt or SIGTERM...") - signal.pause() - logger.info("Exiting...Get summary...") - for m in monitors: - logger.info(m.summary()) + while True: + signal.pause() @click.group() diff --git a/duetector/monitors/bcc_monitor.py b/duetector/monitors/bcc_monitor.py index a6d4dae..fc250c6 100644 --- a/duetector/monitors/bcc_monitor.py +++ b/duetector/monitors/bcc_monitor.py @@ -52,6 +52,8 @@ def init(self): # Prevrent ImportError for CI testing without bcc from bcc import BPF # noqa + err_tracers = [] + for tracer in self.tracers: try: bpf = BPF(text=tracer.prog) @@ -59,6 +61,10 @@ def init(self): logger.error(f"Failed to compile {tracer.__class__.__name__}") logger.exception(e) if self.continue_on_exception: + logger.info( + f"Continuing on exception. {tracer.__class__.__name__} will be disabled." + ) + err_tracers.append(tracer) continue else: raise e @@ -67,6 +73,10 @@ def init(self): self.bpf_tracers[tracer] = bpf logger.info(f"Tracer {tracer.__class__.__name__} attached") + # Remove tracers that failed to compile + for tracer in err_tracers: + self.tracers.remove(tracer) + def _set_callback(self, host, tracer): def _(data): for filter in self.filters: From e0c36fbf1312ca1b26adaed4c64bfd073626b854 Mon Sep 17 00:00:00 2001 From: wunder957 Date: Wed, 30 Aug 2023 23:24:46 +0800 Subject: [PATCH 2/6] Add dev-tools for debugging --- .gitignore | 4 +++- dev-tools/entrypoint.py | 15 +++++++++++++++ 2 files changed, 18 insertions(+), 1 deletion(-) create mode 100644 dev-tools/entrypoint.py diff --git a/.gitignore b/.gitignore index 712b8f6..8cbd122 100644 --- a/.gitignore +++ b/.gitignore @@ -570,4 +570,6 @@ dmypy.json cython_debug/ .idea/** -duetector-dbcollector.sqlite3 +duetector-dbcollector.sqlite3* +dev-tools/duetector-dbcollector.sqlite3* +dev-tools/config.toml diff --git a/dev-tools/entrypoint.py b/dev-tools/entrypoint.py new file mode 100644 index 0000000..fad1fba --- /dev/null +++ b/dev-tools/entrypoint.py @@ -0,0 +1,15 @@ +import os + +os.chdir(os.path.dirname(os.path.abspath(__file__))) + +import re +import sys + +from pkg_resources import load_entry_point + +if __name__ == "__main__": + os.unlink("./duetector-dbcollector.sqlite3") + sys.argv[0] = re.sub(r"(-script\.pyw?|\.exe)?$", "", sys.argv[0]) + sys.argv.append("start") + sys.argv.extend(["--config", "./config.toml"]) + sys.exit(load_entry_point("duetector", "console_scripts", "duectl")()) From d01443ca4b00d1c9a613284b4b89c8ef3277d0c5 Mon Sep 17 00:00:00 2001 From: wunder957 Date: Wed, 30 Aug 2023 23:33:59 +0800 Subject: [PATCH 3/6] Improve debug tools and log --- dev-tools/entrypoint.py | 9 +++++++-- duetector/managers/tracer.py | 2 +- duetector/monitors/base.py | 6 +++++- duetector/monitors/bcc_monitor.py | 3 --- duetector/monitors/sh_monitor.py | 3 --- 5 files changed, 13 insertions(+), 10 deletions(-) diff --git a/dev-tools/entrypoint.py b/dev-tools/entrypoint.py index fad1fba..e638dff 100644 --- a/dev-tools/entrypoint.py +++ b/dev-tools/entrypoint.py @@ -1,15 +1,20 @@ import os os.chdir(os.path.dirname(os.path.abspath(__file__))) +os.environ["DUETECTOR_LOG_LEVEL"] = "DEBUG" import re import sys +from pathlib import Path from pkg_resources import load_entry_point +db_file = Path("./duetector-dbcollector.sqlite3") +config_file = Path("./config.toml") + if __name__ == "__main__": - os.unlink("./duetector-dbcollector.sqlite3") + db_file.unlink(missing_ok=True) sys.argv[0] = re.sub(r"(-script\.pyw?|\.exe)?$", "", sys.argv[0]) sys.argv.append("start") - sys.argv.extend(["--config", "./config.toml"]) + sys.argv.extend(["--config", config_file.resolve().as_posix()]) sys.exit(load_entry_point("duetector", "console_scripts", "duectl")()) diff --git a/duetector/managers/tracer.py b/duetector/managers/tracer.py index ae480ed..dabbf5a 100644 --- a/duetector/managers/tracer.py +++ b/duetector/managers/tracer.py @@ -40,7 +40,7 @@ def init(self, tracer_type=Tracer, ignore_disabled=True) -> List[Tracer]: objs = [] for f in self.pm.hook.init_tracer(config=self.config.config_dict): if not f or (f.disabled and ignore_disabled): - logger.info(f"Tracer {f.__class__.__name__} is not available (None or Disabled)") + logger.debug(f"Tracer {f.__class__.__name__} is not available (None or Disabled)") continue if not isinstance(f, tracer_type): logger.debug( diff --git a/duetector/monitors/base.py b/duetector/monitors/base.py index a62489c..4437290 100644 --- a/duetector/monitors/base.py +++ b/duetector/monitors/base.py @@ -55,7 +55,11 @@ def poll(self, tracer: Tracer): raise NotImplementedError def summary(self) -> Dict: - return {collector.__class__.__name__: collector.summary() for collector in self.collectors} + return { + self.__class__.__name__: { + collector.__class__.__name__: collector.summary() for collector in self.collectors + } + } def start_polling(self): logger.info(f"Start polling {self.__class__.__name__}") diff --git a/duetector/monitors/bcc_monitor.py b/duetector/monitors/bcc_monitor.py index fc250c6..54fd315 100644 --- a/duetector/monitors/bcc_monitor.py +++ b/duetector/monitors/bcc_monitor.py @@ -91,9 +91,6 @@ def _(data): def poll(self, tracer: BccTracer): # type: ignore tracer.get_poller(self.bpf_tracers[tracer])(**tracer.poll_args) - def summary(self): - return {collector.__class__.__name__: collector.summary() for collector in self.collectors} - if __name__ == "__main__": m = BccMonitor() diff --git a/duetector/monitors/sh_monitor.py b/duetector/monitors/sh_monitor.py index 451192c..328c406 100644 --- a/duetector/monitors/sh_monitor.py +++ b/duetector/monitors/sh_monitor.py @@ -120,9 +120,6 @@ def poll_all(self): def poll(self, tracer: ShellTracer): # type: ignore return self.host.poll(tracer) - def summary(self): - return {collector.__class__.__name__: collector.summary() for collector in self.collectors} - if __name__ == "__main__": m = ShMonitor() From 162541bf0413547b1c8b73560b95e5a4d8e08f63 Mon Sep 17 00:00:00 2001 From: wunder957 Date: Wed, 30 Aug 2023 23:45:41 +0800 Subject: [PATCH 4/6] Fix testing --- tests/test_bcc_monitor.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_bcc_monitor.py b/tests/test_bcc_monitor.py index 920ea8e..8c0818d 100644 --- a/tests/test_bcc_monitor.py +++ b/tests/test_bcc_monitor.py @@ -110,7 +110,7 @@ def test_bcc_monitor(bcc_monitor: MockMonitor): bcc_monitor.poll_all() bcc_monitor.shutdown() assert bcc_monitor.summary() - bcc_monitor.summary()["DBCollector"]["BccMockTracer"]["last"] == Tracking( + bcc_monitor.summary()["MockMonitor"]["DBCollector"]["BccMockTracer"]["last"] == Tracking( tracer="BccMockTracer", pid=9999, uid=9999, From 9b3dcf9814bb65dad55ac90836589a5267d9510c Mon Sep 17 00:00:00 2001 From: wunder957 Date: Thu, 31 Aug 2023 00:35:23 +0800 Subject: [PATCH 5/6] Improve config and check repo config uptodate --- duetector/cli/main.py | 3 +++ duetector/config.py | 3 ++- duetector/tools/config_generator.py | 7 +++++-- tests/test_config.py | 1 - tests/test_repo.py | 32 +++++++++++++++++++++++++++++ 5 files changed, 42 insertions(+), 4 deletions(-) create mode 100644 tests/test_repo.py diff --git a/duetector/cli/main.py b/duetector/cli/main.py index 9fce4df..106f4ec 100644 --- a/duetector/cli/main.py +++ b/duetector/cli/main.py @@ -48,6 +48,9 @@ def generate_dynamic_config(load_current_config, path, load_env, dump_path): c = ConfigGenerator(load=load_current_config, path=path, load_env=load_env) if path.as_posix() == Path(dump_path).expanduser().absolute().as_posix(): + logger.info( + f"Dump path is same as origin path, rename {path} to {path.with_suffix('.old')}" + ) shutil.move(path, path.with_suffix(".old")) c.generate(dump_path) diff --git a/duetector/config.py b/duetector/config.py index 3f9c989..1263a9c 100644 --- a/duetector/config.py +++ b/duetector/config.py @@ -1,3 +1,4 @@ +import copy import os import shutil from pathlib import Path @@ -167,7 +168,7 @@ def __init__(self, config: Optional[Union[Config, Dict[str, Any]]] = None, *args if self.config_scope: for score in self.config_scope.split("."): config = config.get(score.lower(), {}) - c = self.default_config.copy() + c = copy.deepcopy(self.default_config) def _recursive_update(c, config): for k, v in config.items(): diff --git a/duetector/tools/config_generator.py b/duetector/tools/config_generator.py index 2973e7f..7a0efe0 100644 --- a/duetector/tools/config_generator.py +++ b/duetector/tools/config_generator.py @@ -10,6 +10,10 @@ def _recursive_load(config_scope: str, config_dict: dict, default_config: dict): + """ + Support .(dot) separated config_scope + + """ *prefix, config_scope = config_scope.lower().split(".") last = config_dict for p in prefix: @@ -48,7 +52,6 @@ def __init__(self, load=True, path=None, load_env=True): c.default_config, ) - # Support .(dot) separated config_scope for m in self.monitors: _recursive_load(m.config_scope, self.dynamic_config, m.default_config) @@ -80,6 +83,6 @@ def generate(self, dump_path): if __name__ == "__main__": _HERE = Path(__file__).parent - c = ConfigGenerator(load=False) + c = ConfigGenerator(load=False, load_env=False) config_path = _HERE / ".." / "static/config.toml" c.generate(config_path) diff --git a/tests/test_config.py b/tests/test_config.py index 0cd57c2..edefc03 100644 --- a/tests/test_config.py +++ b/tests/test_config.py @@ -69,7 +69,6 @@ def test_load_env(config_loader: ConfigLoader, monkeypatch): def test_generate(config_generator: ConfigGenerator, tmpdir): - tmpdir.join("config-generated.toml") generated_file = tmpdir.join("config-generated.toml") config_generator.generate(generated_file) assert generated_file.exists() diff --git a/tests/test_repo.py b/tests/test_repo.py new file mode 100644 index 0000000..56671ba --- /dev/null +++ b/tests/test_repo.py @@ -0,0 +1,32 @@ +from pathlib import Path + +import pytest +import toml + +from duetector.config import ConfigLoader +from duetector.tools.config_generator import ConfigGenerator + +_HERE = Path(__file__).parent.absolute() + + +def test_repo_config_uptodate(tmpdir): + # Check default config in repo is up to date + CONFIG_IN_REPO = _HERE / ".." / "duetector/static/config.toml" + GENERATED_CONFIG = tmpdir.join("g-default-config.toml") + config_generator = ConfigGenerator(load=False, load_env=False) + config_generator.generate(GENERATED_CONFIG) + assert GENERATED_CONFIG.exists() + + generated_config = toml.loads(GENERATED_CONFIG.read_text(encoding="utf-8")) + repo_config = ConfigLoader( + path=CONFIG_IN_REPO, + load_env=False, + dump_when_load=False, + config_dump_dir=None, + generate_config=False, + ).load_config() + assert generated_config == repo_config + + +if __name__ == "__main__": + pytest.main(["-vv", "-s", __file__]) From 47b5c642aea9416b7d93aac9313b983483a801c2 Mon Sep 17 00:00:00 2001 From: wunder957 Date: Thu, 31 Aug 2023 09:23:45 +0800 Subject: [PATCH 6/6] Fix testing --- tests/test_repo.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/test_repo.py b/tests/test_repo.py index 56671ba..cf47522 100644 --- a/tests/test_repo.py +++ b/tests/test_repo.py @@ -1,7 +1,7 @@ from pathlib import Path import pytest -import toml +import tomli from duetector.config import ConfigLoader from duetector.tools.config_generator import ConfigGenerator @@ -17,7 +17,7 @@ def test_repo_config_uptodate(tmpdir): config_generator.generate(GENERATED_CONFIG) assert GENERATED_CONFIG.exists() - generated_config = toml.loads(GENERATED_CONFIG.read_text(encoding="utf-8")) + generated_config = tomli.loads(GENERATED_CONFIG.read_text(encoding="utf-8")) repo_config = ConfigLoader( path=CONFIG_IN_REPO, load_env=False,