Development Artifact Cleanup: ✅ BROTHER_NODE REORGANIZATION: Moved development test node to appropriate location - dev/test-nodes/brother_node/: Moved from root directory for better organization - Contains development configuration, test logs, and test chain data - No impact on production systems - purely development/testing artifact ✅ DEVELOPMENT ARTIFACTS IDENTIFIED: - Chain ID: aitbc-brother-chain (test/development chain) - Ports: 8010 (P2P) and 8011 (RPC) - different from production - Environment: .env file with test configuration - Logs: rpc.log and node.log from development testing session (March 15, 2026) ✅ ROOT DIRECTORY CLEANUP: Removed development clutter from production directory - brother_node/ moved to dev/test-nodes/brother_node/ - Root directory now contains only production-ready components - Development artifacts properly organized in dev/ subdirectory DIRECTORY STRUCTURE IMPROVEMENT: 📁 dev/test-nodes/: Development and testing node configurations 🏗️ Root Directory: Clean production structure with only essential components 🧪 Development Isolation: Test environments separated from production BENEFITS: ✅ Clean Production Directory: No development artifacts in root ✅ Better Organization: Development nodes grouped in dev/ subdirectory ✅ Clear Separation: Production vs development environments clearly distinguished ✅ Maintainability: Easier to identify and manage development components RESULT: Successfully moved brother_node development artifact to dev/test-nodes/ subdirectory, cleaning up the root directory while preserving development testing environment for future use.
164 lines
4.4 KiB
Python
Executable File
164 lines
4.4 KiB
Python
Executable File
import contextlib
|
|
import ctypes
|
|
import os
|
|
|
|
from ctypes.wintypes import (
|
|
BOOL,
|
|
CHAR,
|
|
DWORD,
|
|
HANDLE,
|
|
LONG,
|
|
LPWSTR,
|
|
MAX_PATH,
|
|
PDWORD,
|
|
ULONG,
|
|
)
|
|
|
|
from shellingham._core import SHELL_NAMES
|
|
|
|
|
|
INVALID_HANDLE_VALUE = HANDLE(-1).value
|
|
ERROR_NO_MORE_FILES = 18
|
|
ERROR_INSUFFICIENT_BUFFER = 122
|
|
TH32CS_SNAPPROCESS = 2
|
|
PROCESS_QUERY_LIMITED_INFORMATION = 0x1000
|
|
|
|
|
|
kernel32 = ctypes.windll.kernel32
|
|
|
|
|
|
def _check_handle(error_val=0):
|
|
def check(ret, func, args):
|
|
if ret == error_val:
|
|
raise ctypes.WinError()
|
|
return ret
|
|
|
|
return check
|
|
|
|
|
|
def _check_expected(expected):
|
|
def check(ret, func, args):
|
|
if ret:
|
|
return True
|
|
code = ctypes.GetLastError()
|
|
if code == expected:
|
|
return False
|
|
raise ctypes.WinError(code)
|
|
|
|
return check
|
|
|
|
|
|
class ProcessEntry32(ctypes.Structure):
|
|
_fields_ = (
|
|
("dwSize", DWORD),
|
|
("cntUsage", DWORD),
|
|
("th32ProcessID", DWORD),
|
|
("th32DefaultHeapID", ctypes.POINTER(ULONG)),
|
|
("th32ModuleID", DWORD),
|
|
("cntThreads", DWORD),
|
|
("th32ParentProcessID", DWORD),
|
|
("pcPriClassBase", LONG),
|
|
("dwFlags", DWORD),
|
|
("szExeFile", CHAR * MAX_PATH),
|
|
)
|
|
|
|
|
|
kernel32.CloseHandle.argtypes = [HANDLE]
|
|
kernel32.CloseHandle.restype = BOOL
|
|
|
|
kernel32.CreateToolhelp32Snapshot.argtypes = [DWORD, DWORD]
|
|
kernel32.CreateToolhelp32Snapshot.restype = HANDLE
|
|
kernel32.CreateToolhelp32Snapshot.errcheck = _check_handle( # type: ignore
|
|
INVALID_HANDLE_VALUE,
|
|
)
|
|
|
|
kernel32.Process32First.argtypes = [HANDLE, ctypes.POINTER(ProcessEntry32)]
|
|
kernel32.Process32First.restype = BOOL
|
|
kernel32.Process32First.errcheck = _check_expected( # type: ignore
|
|
ERROR_NO_MORE_FILES,
|
|
)
|
|
|
|
kernel32.Process32Next.argtypes = [HANDLE, ctypes.POINTER(ProcessEntry32)]
|
|
kernel32.Process32Next.restype = BOOL
|
|
kernel32.Process32Next.errcheck = _check_expected( # type: ignore
|
|
ERROR_NO_MORE_FILES,
|
|
)
|
|
|
|
kernel32.GetCurrentProcessId.argtypes = []
|
|
kernel32.GetCurrentProcessId.restype = DWORD
|
|
|
|
kernel32.OpenProcess.argtypes = [DWORD, BOOL, DWORD]
|
|
kernel32.OpenProcess.restype = HANDLE
|
|
kernel32.OpenProcess.errcheck = _check_handle( # type: ignore
|
|
INVALID_HANDLE_VALUE,
|
|
)
|
|
|
|
kernel32.QueryFullProcessImageNameW.argtypes = [HANDLE, DWORD, LPWSTR, PDWORD]
|
|
kernel32.QueryFullProcessImageNameW.restype = BOOL
|
|
kernel32.QueryFullProcessImageNameW.errcheck = _check_expected( # type: ignore
|
|
ERROR_INSUFFICIENT_BUFFER,
|
|
)
|
|
|
|
|
|
@contextlib.contextmanager
|
|
def _handle(f, *args, **kwargs):
|
|
handle = f(*args, **kwargs)
|
|
try:
|
|
yield handle
|
|
finally:
|
|
kernel32.CloseHandle(handle)
|
|
|
|
|
|
def _iter_processes():
|
|
f = kernel32.CreateToolhelp32Snapshot
|
|
with _handle(f, TH32CS_SNAPPROCESS, 0) as snap:
|
|
entry = ProcessEntry32()
|
|
entry.dwSize = ctypes.sizeof(entry)
|
|
ret = kernel32.Process32First(snap, entry)
|
|
while ret:
|
|
yield entry
|
|
ret = kernel32.Process32Next(snap, entry)
|
|
|
|
|
|
def _get_full_path(proch):
|
|
size = DWORD(MAX_PATH)
|
|
while True:
|
|
path_buff = ctypes.create_unicode_buffer("", size.value)
|
|
if kernel32.QueryFullProcessImageNameW(proch, 0, path_buff, size):
|
|
return path_buff.value
|
|
size.value *= 2
|
|
|
|
|
|
def get_shell(pid=None, max_depth=10):
|
|
proc_map = {
|
|
proc.th32ProcessID: (proc.th32ParentProcessID, proc.szExeFile)
|
|
for proc in _iter_processes()
|
|
}
|
|
pid = pid or os.getpid()
|
|
|
|
for _ in range(0, max_depth + 1):
|
|
try:
|
|
ppid, executable = proc_map[pid]
|
|
except KeyError: # No such process? Give up.
|
|
break
|
|
|
|
# The executable name would be encoded with the current code page if
|
|
# we're in ANSI mode (usually). Try to decode it into str/unicode,
|
|
# replacing invalid characters to be safe (not thoeratically necessary,
|
|
# I think). Note that we need to use 'mbcs' instead of encoding
|
|
# settings from sys because this is from the Windows API, not Python
|
|
# internals (which those settings reflect). (pypa/pipenv#3382)
|
|
if isinstance(executable, bytes):
|
|
executable = executable.decode("mbcs", "replace")
|
|
|
|
name = executable.rpartition(".")[0].lower()
|
|
if name not in SHELL_NAMES:
|
|
pid = ppid
|
|
continue
|
|
|
|
key = PROCESS_QUERY_LIMITED_INFORMATION
|
|
with _handle(kernel32.OpenProcess, key, 0, pid) as proch:
|
|
return (name, _get_full_path(proch))
|
|
|
|
return None
|