File: process_monitor.py

package info (click to toggle)
python-azure 20250603%2Bgit-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 851,724 kB
  • sloc: python: 7,362,925; ansic: 804; javascript: 287; makefile: 195; sh: 145; xml: 109
file content (85 lines) | stat: -rw-r--r-- 2,965 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
# --------------------------------------------------------------------------------------------
# Copyright (c) Microsoft Corporation. All rights reserved.
# Licensed under the MIT License. See License.txt in the project root for license information.
# --------------------------------------------------------------------------------------------
import os
import sys
import logging
import psutil
import threading
import time


def get_base_logger(logger_name: str, log_file_name: str = ""):
    logger = logging.getLogger(logger_name)
    logger.setLevel(logging.INFO)
    formatter = logging.Formatter("%(asctime)s %(name)-12s %(levelname)-8s %(message)s")
    handers = [logging.StreamHandler(sys.stdout)]
    if log_file_name:
        handers.append(logging.FileHandler(log_file_name, mode="w"))
    for handler in handers:
        handler.setFormatter(formatter)
        logger.addHandler(handler)

    return logger


class ProcessMonitor:
    def __init__(self, logger_name, log_file_name: str, log_interval: int):
        """
        Process Monitor monitors the CPU usage, memory usage of a specific process.
        :param logger_name: The name for the logger.
        :param log_interval: The interval of logging.
        """
        self._monitor_thread = None
        self._logger = get_base_logger(
            logger_name=logger_name,
            log_file_name=log_file_name,
        )
        self._pid = os.getpid()
        self._process_instance = psutil.Process(self._pid)
        self._log_interval = log_interval
        self.running = False

    def __enter__(self):
        print("Process monitor start working.")
        self.start()
        return self

    def __exit__(self, *args):
        self.stop()
        print("Process monitor stop working.")

    def _monitor_work(self):
        while self.running:
            log_content = (
                "process status: {},"
                "process cpu usage percent: {},"
                "process memory usage percent: {:.3f}".format(
                    self._process_instance.status(),
                    self._process_instance.cpu_percent(),
                    self._process_instance.memory_percent(),
                )
            )
            self._logger.info(log_content)
            time.sleep(self._log_interval)

    @property
    def memory_usage_percent(self):
        return self._process_instance.memory_percent() * 100

    @property
    def cpu_usage_percent(self):
        return self._process_instance.cpu_percent()

    def start(self):
        self.running = True
        self._monitor_thread = threading.Thread(target=self._monitor_work, daemon=True)
        self._monitor_thread.start()
        self._logger.info("Start monitoring process id:{}".format(self._pid))

    def stop(self):
        self.running = False
        self._monitor_thread.join()
        self._logger.info("Stop monitoring process id:{}".format(self._pid))
        self._monitor_thread = None