import os import hashlib import sqlite3 import sys import unittest import zipfile import importlib from pathlib import Path from tempfile import TemporaryDirectory ROOT = Path(__file__).resolve().parents[1] sys.path.insert(0, str(ROOT / "src")) class TestChatExportChatHistoryModal(unittest.TestCase): _MD5 = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" def _reload_export_modules(self): import wechat_decrypt_tool.app_paths as app_paths import wechat_decrypt_tool.chat_helpers as chat_helpers import wechat_decrypt_tool.media_helpers as media_helpers import wechat_decrypt_tool.chat_export_service as chat_export_service importlib.reload(app_paths) importlib.reload(chat_helpers) importlib.reload(media_helpers) importlib.reload(chat_export_service) return chat_export_service def _seed_contact_db(self, path: Path, *, account: str, username: str) -> None: conn = sqlite3.connect(str(path)) try: conn.execute( """ CREATE TABLE contact ( username TEXT, remark TEXT, nick_name TEXT, alias TEXT, local_type INTEGER, verify_flag INTEGER, big_head_url TEXT, small_head_url TEXT ) """ ) conn.execute( """ CREATE TABLE stranger ( username TEXT, remark TEXT, nick_name TEXT, alias TEXT, local_type INTEGER, verify_flag INTEGER, big_head_url TEXT, small_head_url TEXT ) """ ) conn.execute( "INSERT INTO contact VALUES (?, ?, ?, ?, ?, ?, ?, ?)", (account, "", "我", "", 1, 0, "", ""), ) conn.execute( "INSERT INTO contact VALUES (?, ?, ?, ?, ?, ?, ?, ?)", (username, "", "测试好友", "", 1, 0, "", ""), ) conn.commit() finally: conn.close() def _seed_session_db(self, path: Path, *, username: str) -> None: conn = sqlite3.connect(str(path)) try: conn.execute( """ CREATE TABLE SessionTable ( username TEXT, is_hidden INTEGER, sort_timestamp INTEGER ) """ ) conn.execute( "INSERT INTO SessionTable VALUES (?, ?, ?)", (username, 0, 1735689600), ) conn.commit() finally: conn.close() def _seed_message_db(self, path: Path, *, account: str, username: str) -> None: conn = sqlite3.connect(str(path)) try: conn.execute("CREATE TABLE Name2Id (rowid INTEGER PRIMARY KEY, user_name TEXT)") conn.execute("INSERT INTO Name2Id(rowid, user_name) VALUES (?, ?)", (1, account)) conn.execute("INSERT INTO Name2Id(rowid, user_name) VALUES (?, ?)", (2, username)) table_name = f"msg_{hashlib.md5(username.encode('utf-8')).hexdigest()}" conn.execute( f""" CREATE TABLE {table_name} ( local_id INTEGER, server_id INTEGER, local_type INTEGER, sort_seq INTEGER, real_sender_id INTEGER, create_time INTEGER, message_content TEXT, compress_content BLOB ) """ ) record_item = ( "" "" "" "2" f"{self._MD5}" "" "" "" ) chat_history_xml = ( "" "19" "聊天记录" "记录预览" f"" "" ) conn.execute( f"INSERT INTO {table_name} (local_id, server_id, local_type, sort_seq, real_sender_id, create_time, message_content, compress_content) VALUES (?, ?, ?, ?, ?, ?, ?, ?)", (1, 1001, 49, 1, 2, 1735689601, chat_history_xml, None), ) conn.commit() finally: conn.close() def _seed_media_files(self, account_dir: Path) -> None: resource_root = account_dir / "resource" (resource_root / "aa").mkdir(parents=True, exist_ok=True) (resource_root / "aa" / f"{self._MD5}.jpg").write_bytes(b"\xff\xd8\xff\xd9") def _prepare_account(self, root: Path, *, account: str, username: str) -> Path: account_dir = root / "output" / "databases" / account account_dir.mkdir(parents=True, exist_ok=True) self._seed_contact_db(account_dir / "contact.db", account=account, username=username) self._seed_session_db(account_dir / "session.db", username=username) self._seed_message_db(account_dir / "message_0.db", account=account, username=username) self._seed_media_files(account_dir) return account_dir def _create_job(self, manager, *, account: str, username: str): job = manager.create_job( account=account, scope="selected", usernames=[username], export_format="html", start_time=None, end_time=None, include_hidden=False, include_official=False, include_media=True, media_kinds=["image"], message_types=["chatHistory", "image"], output_dir=None, allow_process_key_extract=False, download_remote_media=False, privacy_mode=False, file_name=None, ) for _ in range(200): latest = manager.get_job(job.export_id) if latest and latest.status in {"done", "error", "cancelled"}: return latest import time as _time _time.sleep(0.05) self.fail("export job did not finish in time") def test_chat_history_modal_has_media_index_and_record_item(self): with TemporaryDirectory() as td: root = Path(td) account = "wxid_test" username = "wxid_friend" self._prepare_account(root, account=account, username=username) prev_data = os.environ.get("WECHAT_TOOL_DATA_DIR") try: os.environ["WECHAT_TOOL_DATA_DIR"] = str(root) svc = self._reload_export_modules() job = self._create_job(svc.CHAT_EXPORT_MANAGER, account=account, username=username) self.assertEqual(job.status, "done", msg=job.error) with zipfile.ZipFile(job.zip_path, "r") as zf: names = set(zf.namelist()) self.assertIn(f"media/images/{self._MD5}.jpg", names) html_path = next((n for n in names if n.endswith("/messages.html")), "") self.assertTrue(html_path) html_text = zf.read(html_path).decode("utf-8") self.assertIn('id="chatHistoryModal"', html_text) self.assertIn('data-wce-chat-history="1"', html_text) self.assertIn('data-record-item-b64="', html_text) self.assertIn('id="wceMediaIndex"', html_text) self.assertIn(self._MD5, html_text) finally: if prev_data is None: os.environ.pop("WECHAT_TOOL_DATA_DIR", None) else: os.environ["WECHAT_TOOL_DATA_DIR"] = prev_data