#!/usr/bin/env python # -*- coding: utf-8 -*- """ @Time : 2025/1/10 2:36 @Author : SiYuan @Email : 863909694@qq.com @File : wxManager-wx_info_v4.py @Description : 部分思路参考:https://github.com/0xlane/wechat-dump-rs """ import ctypes import multiprocessing import os.path import hmac import os import struct import time from ctypes import wintypes from multiprocessing import freeze_support import pymem from Crypto.Protocol.KDF import PBKDF2 from Crypto.Hash import SHA512 import yara from wxManager.decrypt.common import WeChatInfo from wxManager.decrypt.common import get_version # 定义必要的常量 PROCESS_ALL_ACCESS = 0x1F0FFF PAGE_READWRITE = 0x04 MEM_COMMIT = 0x1000 MEM_PRIVATE = 0x20000 # Constants IV_SIZE = 16 HMAC_SHA256_SIZE = 64 HMAC_SHA512_SIZE = 64 KEY_SIZE = 32 AES_BLOCK_SIZE = 16 ROUND_COUNT = 256000 PAGE_SIZE = 4096 SALT_SIZE = 16 finish_flag = False # 定义 MEMORY_BASIC_INFORMATION 结构 class MEMORY_BASIC_INFORMATION(ctypes.Structure): _fields_ = [ ("BaseAddress", ctypes.c_void_p), ("AllocationBase", ctypes.c_void_p), ("AllocationProtect", ctypes.c_ulong), ("RegionSize", ctypes.c_size_t), ("State", ctypes.c_ulong), ("Protect", ctypes.c_ulong), ("Type", ctypes.c_ulong), ] # Windows API Constants PROCESS_VM_READ = 0x0010 PROCESS_QUERY_INFORMATION = 0x0400 # Load Windows DLLs kernel32 = ctypes.windll.kernel32 # 打开目标进程 def open_process(pid): return ctypes.windll.kernel32.OpenProcess(PROCESS_ALL_ACCESS, False, pid) # 读取目标进程内存 def read_process_memory(process_handle, address, size): buffer = ctypes.create_string_buffer(size) bytes_read = ctypes.c_size_t(0) success = ctypes.windll.kernel32.ReadProcessMemory( process_handle, ctypes.c_void_p(address), buffer, size, ctypes.byref(bytes_read) ) if not success: return None return buffer.raw # 获取所有内存区域 def get_memory_regions(process_handle): regions = [] mbi = MEMORY_BASIC_INFORMATION() address = 0 while ctypes.windll.kernel32.VirtualQueryEx( process_handle, ctypes.c_void_p(address), ctypes.byref(mbi), ctypes.sizeof(mbi) ): if mbi.State == MEM_COMMIT and mbi.Type == MEM_PRIVATE: regions.append((mbi.BaseAddress, mbi.RegionSize)) address += mbi.RegionSize return regions rules_v4 = r''' rule GetDataDir { strings: $a = /[a-zA-Z]:\\(.{1,100}?\\){0,1}?xwechat_files\\[0-9a-zA-Z_-]{6,24}?\\db_storage\\/ condition: $a } rule GetPhoneNumberOffset { strings: $a = /[\x01-\x20]\x00{7}(\x0f|\x1f)\x00{7}[0-9]{11}\x00{5}\x0b\x00{7}\x0f\x00{7}/ condition: $a } rule GetKeyAddrStub { strings: $a = /.{6}\x00{2}\x00{8}\x20\x00{7}\x2f\x00{7}/ condition: all of them } ''' def read_string(data: bytes, offset, size): try: return data[offset:offset + size].decode('utf-8') except: # print(data[offset:offset + size]) # print(traceback.format_exc()) return '' def read_num(data: bytes, offset, size): # 构建格式字符串,根据 size 来选择相应的格式 if size == 1: fmt = ' WeChatInfo | None: wechat_info = WeChatInfo() wechat_info.pid = pid wechat_info.version = get_version(pid) process_handle = open_process(pid) if not process_handle: print(f"无法打开进程 {pid}") return wechat_info queue = multiprocessing.Queue() process = multiprocessing.Process(target=worker, args=(pid, queue)) process.start() wechat_info.wx_dir = get_wx_dir(process_handle) # print(wx_dir_cnt) if not wechat_info.wx_dir: return wechat_info db_file_path = os.path.join(wechat_info.wx_dir, 'favorite', 'favorite_fts.db') if not os.path.exists(db_file_path): db_file_path = os.path.join(wechat_info.wx_dir, 'head_image', 'head_image.db') with open(db_file_path, 'rb') as f: buf = f.read() wechat_info.key = get_key(pid, process_handle, buf) ctypes.windll.kernel32.CloseHandle(process_handle) wechat_info.wxid = '_'.join(wechat_info.wx_dir.split('\\')[-3].split('_')[0:-1]) wechat_info.wx_dir = '\\'.join(wechat_info.wx_dir.split('\\')[:-2]) process.join() # 等待子进程完成 if not queue.empty(): nickname_info = queue.get() wechat_info.nick_name = nickname_info.get('nick_name', '') wechat_info.phone = nickname_info.get('phone', '') wechat_info.account_name = nickname_info.get('account_name', '') if not wechat_info.key: wechat_info.errcode = 404 else: wechat_info.errcode = 200 return wechat_info if __name__ == '__main__': freeze_support() st = time.time() pm = pymem.Pymem("Weixin.exe") pid = pm.process_id w = dump_wechat_info_v4(pid) print(w) et = time.time() print(et - st)