| 1 |  |  | """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 2 |  |  | Low-level tools (e.g. memory management). | 
            
                                                                                                            
                            
            
                                    
            
            
                | 3 |  |  | """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 4 |  |  | import atexit | 
            
                                                                                                            
                            
            
                                    
            
            
                | 5 |  |  | import importlib | 
            
                                                                                                            
                            
            
                                    
            
            
                | 6 |  |  | import locale | 
            
                                                                                                            
                            
            
                                    
            
            
                | 7 |  |  | import logging | 
            
                                                                                                            
                            
            
                                    
            
            
                | 8 |  |  | import os | 
            
                                                                                                            
                            
            
                                    
            
            
                | 9 |  |  | import platform | 
            
                                                                                                            
                            
            
                                    
            
            
                | 10 |  |  | import socket | 
            
                                                                                                            
                            
            
                                    
            
            
                | 11 |  |  | import signal | 
            
                                                                                                            
                            
            
                                    
            
            
                | 12 |  |  | import struct | 
            
                                                                                                            
                            
            
                                    
            
            
                | 13 |  |  | import sys | 
            
                                                                                                            
                            
            
                                    
            
            
                | 14 |  |  | import traceback | 
            
                                                                                                            
                            
            
                                    
            
            
                | 15 |  |  | from collections import Callable | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 16 |  |  | from dataclasses import dataclass, asdict | 
            
                                                                                                            
                            
            
                                    
            
            
                | 17 |  |  | from datetime import timezone, datetime | 
            
                                                                                                            
                            
            
                                    
            
            
                | 18 |  |  | from getpass import getuser | 
            
                                                                                                            
                            
            
                                    
            
            
                | 19 |  |  | from typing import Any, Union, Sequence, Mapping, Optional, NamedTuple | 
            
                                                                                                            
                            
            
                                    
            
            
                | 20 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 21 |  |  | from pocketutils.core.input_output import Writeable | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 22 |  |  | from pocketutils.tools.base_tools import BaseTools | 
            
                                                                                                            
                            
            
                                    
            
            
                | 23 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 24 |  |  | logger = logging.getLogger("pocketutils") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 25 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 26 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 27 |  |  | @dataclass(frozen=True, repr=True, order=True) | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 28 |  |  | class Frame: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 29 |  |  |     depth: int | 
            
                                                                                                            
                            
            
                                    
            
            
                | 30 |  |  |     filename: str | 
            
                                                                                                            
                            
            
                                    
            
            
                | 31 |  |  |     line: int | 
            
                                                                                                            
                            
            
                                    
            
            
                | 32 |  |  |     name: str | 
            
                                                                                                            
                            
            
                                    
            
            
                | 33 |  |  |     repeats: int | 
            
                                                                                                            
                            
            
                                    
            
            
                | 34 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 35 |  |  |     def as_dict(self) -> Mapping[str, Union[int, str]]: | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 36 |  |  |         return asdict(self) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 37 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 38 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 39 |  |  | class SerializedException(NamedTuple): | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 40 |  |  |     message: Sequence[str] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 41 |  |  |     stacktrace: Sequence[Frame] | 
            
                                                                                                            
                                                                
            
                                    
            
            
                | 42 |  |  |  | 
            
                                                                        
                            
            
                                    
            
            
                | 43 |  |  |  | 
            
                                                                        
                            
            
                                    
            
            
                | 44 |  |  | @dataclass(frozen=True, repr=True) | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                        
                            
            
                                    
            
            
                | 45 |  |  | class SignalHandler: | 
            
                                                                        
                            
            
                                    
            
            
                | 46 |  |  |     name: str | 
            
                                                                        
                            
            
                                    
            
            
                | 47 |  |  |     code: int | 
            
                                                                        
                            
            
                                    
            
            
                | 48 |  |  |     desc: str | 
            
                                                                        
                            
            
                                    
            
            
                | 49 |  |  |     sink: Union[Writeable, Callable[[str], Any]] | 
            
                                                                        
                            
            
                                    
            
            
                | 50 |  |  |  | 
            
                                                                        
                            
            
                                    
            
            
                | 51 |  |  |     def __call__(self): | 
            
                                                                        
                            
            
                                    
            
            
                | 52 |  |  |         sys.stderr.write(f"~~{self.name}[{self.code}] ({self.desc})~~") | 
            
                                                                        
                            
            
                                    
            
            
                | 53 |  |  |         traceback.print_stack(file=sys.stderr) | 
            
                                                                        
                            
            
                                    
            
            
                | 54 |  |  |         for line in traceback.format_stack(): | 
            
                                                                        
                            
            
                                    
            
            
                | 55 |  |  |             sys.stderr.write(line) | 
            
                                                                        
                            
            
                                    
            
            
                | 56 |  |  |  | 
            
                                                                        
                            
            
                                    
            
            
                | 57 |  |  |  | 
            
                                                                        
                            
            
                                    
            
            
                | 58 |  |  | @dataclass(frozen=True, repr=True) | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                        
                            
            
                                    
            
            
                | 59 |  |  | class ExitHandler: | 
            
                                                                        
                            
            
                                    
            
            
                | 60 |  |  |     sink: Writeable | 
            
                                                                                                            
                            
            
                                    
            
            
                | 61 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 62 |  |  |     def __call__(self): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 63 |  |  |         self.sink.write(f"~~EXIT~~") | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 64 |  |  |         traceback.print_stack(file=sys.stderr) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 65 |  |  |         for line in traceback.format_stack(): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 66 |  |  |             self.sink.write(line) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 67 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 68 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 69 |  |  | class SystemTools(BaseTools): | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 70 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 71 |  |  |     def get_env_info(cls, *, include_insecure: bool = False) -> Mapping[str, str]: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 72 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 73 |  |  |         Get a dictionary of some system and environment information. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 74 |  |  |         Includes os_release, hostname, username, mem + disk, shell, etc. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 75 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 76 |  |  |         Args: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 77 |  |  |             include_insecure: Include data like hostname and username | 
            
                                                                                                            
                            
            
                                    
            
            
                | 78 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 79 |  |  |         .. caution :: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 80 |  |  |             Even with ``include_insecure=False``, avoid exposing this data to untrusted | 
            
                                                                                                            
                            
            
                                    
            
            
                | 81 |  |  |             sources. For example, this includes the specific OS release, which could | 
            
                                                                                                            
                            
            
                                    
            
            
                | 82 |  |  |             be used in attack. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 83 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 84 |  |  |         try: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 85 |  |  |             import psutil | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 86 |  |  |         except ImportError: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 87 |  |  |             psutil = None | 
            
                                                                                                            
                            
            
                                    
            
            
                | 88 |  |  |             logger.warning("psutil is not installed, so cannot get extended env info") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 89 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 90 |  |  |         now = datetime.now(timezone.utc).astimezone().isoformat() | 
            
                                                                                                            
                            
            
                                    
            
            
                | 91 |  |  |         uname = platform.uname() | 
            
                                                                                                            
                            
            
                                    
            
            
                | 92 |  |  |         language_code, encoding = locale.getlocale() | 
            
                                                                                                            
                            
            
                                    
            
            
                | 93 |  |  |         # build up this dict: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 94 |  |  |         data = {} | 
            
                                                                                                            
                            
            
                                    
            
            
                | 95 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 96 |  |  |         def _try(os_fn, k: str, *args): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 97 |  |  |             if any((a is None for a in args)): | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 98 |  |  |                 return None | 
            
                                                                                                            
                            
            
                                    
            
            
                | 99 |  |  |             try: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 100 |  |  |                 v = os_fn(*args) | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 101 |  |  |                 data[k] = v | 
            
                                                                                                            
                            
            
                                    
            
            
                | 102 |  |  |                 return v | 
            
                                                                                                            
                            
            
                                    
            
            
                | 103 |  |  |             except (OSError, ImportError): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 104 |  |  |                 return None | 
            
                                                                                                            
                            
            
                                    
            
            
                | 105 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 106 |  |  |         data.update( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 107 |  |  |             dict( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 108 |  |  |                 platform=platform.platform(), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 109 |  |  |                 python=".".join(str(i) for i in sys.version_info), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 110 |  |  |                 os=uname.system, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 111 |  |  |                 os_release=uname.release, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 112 |  |  |                 os_version=uname.version, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 113 |  |  |                 machine=uname.machine, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 114 |  |  |                 byte_order=sys.byteorder, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 115 |  |  |                 processor=uname.processor, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 116 |  |  |                 build=sys.version, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 117 |  |  |                 python_bits=8 * struct.calcsize("P"), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 118 |  |  |                 environment_info_capture_datetime=now, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 119 |  |  |                 encoding=encoding, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 120 |  |  |                 lang_code=language_code, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 121 |  |  |                 recursion_limit=sys.getrecursionlimit(), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 122 |  |  |                 float_info=sys.float_info, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 123 |  |  |                 int_info=sys.int_info, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 124 |  |  |                 flags=sys.flags, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 125 |  |  |                 hash_info=sys.hash_info, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 126 |  |  |                 implementation=sys.implementation, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 127 |  |  |                 switch_interval=sys.getswitchinterval(), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 128 |  |  |                 filesystem_encoding=sys.getfilesystemencoding(), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 129 |  |  |             ) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 130 |  |  |         ) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 131 |  |  |         if "LANG" in os.environ: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 132 |  |  |             data["lang"] = os.environ["LANG"] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 133 |  |  |         if "SHELL" in os.environ: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 134 |  |  |             data["shell"] = os.environ["SHELL"] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 135 |  |  |         if "LC_ALL" in os.environ: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 136 |  |  |             data["lc_all"] = os.environ["LC_ALL"] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 137 |  |  |         if hasattr(sys, "winver"): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 138 |  |  |             data["win_ver"] = sys.getwindowsversion() | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 139 |  |  |         if hasattr(sys, "mac_ver"): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 140 |  |  |             data["mac_ver"] = sys.mac_ver() | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 141 |  |  |         if hasattr(sys, "linux_distribution"): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 142 |  |  |             data["linux_distribution"] = sys.linux_distribution() | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 143 |  |  |         if include_insecure: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 144 |  |  |             _try(getuser, "username") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 145 |  |  |             _try(os.getlogin, "login") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 146 |  |  |             _try(socket.gethostname, "hostname") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 147 |  |  |             _try(os.getcwd, "cwd") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 148 |  |  |             pid = _try(os.getpid, "pid") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 149 |  |  |             ppid = _try(os.getppid, "parent_pid") | 
            
                                                                                                            
                            
            
                                    
            
            
                | 150 |  |  |             if hasattr(os, "getpriority"): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 151 |  |  |                 _try(os.getpriority, "priority", os.PRIO_PROCESS, pid) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 152 |  |  |                 _try(os.getpriority, "parent_priority", os.PRIO_PROCESS, ppid) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 153 |  |  |         if psutil is not None: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 154 |  |  |             data.update( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 155 |  |  |                 dict( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 156 |  |  |                     disk_used=psutil.disk_usage(".").used, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 157 |  |  |                     disk_free=psutil.disk_usage(".").free, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 158 |  |  |                     memory_used=psutil.virtual_memory().used, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 159 |  |  |                     memory_available=psutil.virtual_memory().available, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 160 |  |  |                 ) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 161 |  |  |             ) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 162 |  |  |         return {k: str(v) for k, v in dict(data).items()} | 
            
                                                                                                            
                            
            
                                    
            
            
                | 163 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 164 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 165 |  |  |     def list_package_versions(cls) -> Mapping[str, str]: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 166 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 167 |  |  |         Returns installed packages and their version numbers. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 168 |  |  |         Reliable; uses importlib (Python 3.8+). | 
            
                                                                                                            
                            
            
                                    
            
            
                | 169 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 170 |  |  |         # calling .metadata reads the metadata file | 
            
                                                                                                            
                            
            
                                    
            
            
                | 171 |  |  |         # and .version is an alias to .metadata["version"] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 172 |  |  |         # so make sure to only read once | 
            
                                                                                                            
                            
            
                                    
            
            
                | 173 |  |  |         # TODO: get installed extras? | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 174 |  |  |         dct = {} | 
            
                                                                                                            
                            
            
                                    
            
            
                | 175 |  |  |         for d in importlib.metadata.distributions(): | 
                            
                    |  |  |  | 
                                                                                        
                                                                                            
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 176 |  |  |             meta = d.metadata | 
            
                                                                                                            
                            
            
                                    
            
            
                | 177 |  |  |             dct[meta["name"]] = meta["version"] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 178 |  |  |         return dct | 
            
                                                                                                            
                            
            
                                    
            
            
                | 179 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 180 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 181 |  |  |     def serialize_exception(cls, e: Optional[BaseException]) -> SerializedException: | 
                            
                    |  |  |  | 
                                                                                        
                                                                                            
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 182 |  |  |         tbe = traceback.TracebackException.from_exception(e) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 183 |  |  |         msg = [] if e is None else list(tbe.format_exception_only()) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 184 |  |  |         tb = SystemTools.build_traceback(e) | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 185 |  |  |         return SerializedException(msg, tb) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 186 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 187 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 188 |  |  |     def serialize_exception_msg(cls, e: Optional[BaseException]) -> Sequence[str]: | 
                            
                    |  |  |  | 
                                                                                        
                                                                                            
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 189 |  |  |         tbe = traceback.TracebackException.from_exception(e) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 190 |  |  |         return [] if e is None else list(tbe.format_exception_only()) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 191 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 192 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 193 |  |  |     def build_traceback(cls, e: Optional[BaseException]) -> Sequence[Frame]: | 
                            
                    |  |  |  | 
                                                                                        
                                                                                            
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 194 |  |  |         if e is None: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 195 |  |  |             return [] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 196 |  |  |         tb = [] | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 197 |  |  |         current = None | 
            
                                                                                                            
                            
            
                                    
            
            
                | 198 |  |  |         tbe = traceback.TracebackException.from_exception(e) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 199 |  |  |         last, repeats = None, 0 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 200 |  |  |         for i, s in enumerate(tbe.stack): | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 201 |  |  |             current = Frame(depth=i, filename=s.filename, line=s.line, name=s.name, repeats=-1) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 202 |  |  |             if current == last: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 203 |  |  |                 repeats += 1 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 204 |  |  |             else: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 205 |  |  |                 current = Frame( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 206 |  |  |                     depth=current.depth, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 207 |  |  |                     filename=current.filename, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 208 |  |  |                     line=current.line, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 209 |  |  |                     name=current.name, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 210 |  |  |                     repeats=repeats, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 211 |  |  |                 ) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 212 |  |  |                 tb.append(current) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 213 |  |  |                 repeats = 0 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 214 |  |  |             last = current | 
            
                                                                                                            
                            
            
                                    
            
            
                | 215 |  |  |         if current is not None and current == last: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 216 |  |  |             tb.append(current) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 217 |  |  |         return tb | 
            
                                                                                                            
                            
            
                                    
            
            
                | 218 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 219 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 220 |  |  |     def trace_signals(cls, sink: Writeable = sys.stderr) -> None: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 221 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 222 |  |  |         Registers signal handlers for all signals that log the traceback. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 223 |  |  |         Uses ``signal.signal``. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 224 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 225 |  |  |         for sig in signal.valid_signals(): | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 226 |  |  |             handler = SignalHandler(sig.name, sig.value, signal.strsignal(sig), sink) | 
                            
                    |  |  |  | 
                                                                                        
                                                                                     | 
            
                                                                                                            
                            
            
                                    
            
            
                | 227 |  |  |             signal.signal(sig.value, handler) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 228 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 229 |  |  |     @classmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 230 |  |  |     def trace_exit(cls, sink: Writeable = sys.stderr) -> None: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 231 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 232 |  |  |         Registers an exit handler via ``atexit.register`` that logs the traceback. | 
            
                                                                                                            
                            
            
                                    
            
            
                | 233 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 234 |  |  |         atexit.register(ExitHandler(sink)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 235 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 236 |  |  |  | 
            
                                                                                                            
                                                                
            
                                    
            
            
                | 237 |  |  | __all__ = ["SignalHandler", "ExitHandler", "SystemTools"] | 
            
                                                        
            
                                    
            
            
                | 238 |  |  |  |