返回列表 发布新帖

[教程] 某Q阅读网页版正文JS+CSS解密教程

225 7
发表于 2025-1-31 13:34:46 来自手机 | 查看全部 阅读模式

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

×
本帖最后由 洛璃 于 2025-1-31 13:36 编辑

该教程仅供技术交流,并且无法获得正常正文(字体解密无解),可以参考一下JS+CSS解密流程(比较简单)
某些原因,该教程无法写过于详细,但是核心步骤都会给出
获取网页源代码,可以在末尾找到fkconfig和content
content通过js加密,具体加密js可通过浏览器网络抓包获取,得到:
  1. window = global;
  2. window.self = window;
  3. window.outerHeight = 1030
  4. window.innerHeight = 940
  5. location = {
  6.     protocol: "https:",
  7.     hostname: "book.qq.com",
  8. }
  9. navigator = {
  10.     userAgent: 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36',
  11. }
  12. window.Fock = require('./work')
  13. var beimu = {
  14.     initFock: function initFock(fkConfig) {
  15.         var fuid = fkConfig.fuid;
  16.         window.Fock.setupUserKey(fuid);
  17.         try {
  18.             eval(atob(fkConfig.fkp))
  19.         } catch (e) {
  20.             console.error("fkp errpr", e)
  21.         }
  22.     },
  23.     showContent: function (fkConfig, data, e) {
  24.         this.initFock(fkConfig);
  25.         window.Fock.unlock(data, String(e), (function () {
  26.             debugger;
  27.             var res = arguments[1];
  28.             console.log(btoa(encodeURI(res)))
  29.         }))

  30.     }
  31. }
  32. var fkConfig = __fk_config__
  33. var content = '__page_content__'
  34. var page_num = "__page_num__"
  35. beimu.showContent(fkConfig, content, page_num)
复制代码

然后Python调用js:
  1. import subprocess
  2. import json
  3. import base64
  4. from urllib.parse import quote,unquote

  5. with open('main_module.js','r',encoding='utf-8') as f:
  6.     js_code = f.read()
  7. page_num = 84

  8. js_code = js_code.replace('__fk_config__',json.dumps(fkConfig))
  9. js_code = js_code.replace('__page_content__',content)
  10. js_code = js_code.replace('__page_num__',str(page_num))


  11. with open('main.js','w',encoding='utf-8') as f:
  12.     f.write(js_code)

  13. res = subprocess.check_output(['node','main.js'],text=True).strip()

  14. res = base64.b64decode(res).decode('utf-8')
  15. res = unquote(res)

  16. print(res)
复制代码

然后得到JS解密后的内容,类似于:
<p class="p0"><y4v>㺹</y4v><ykz>㐃</ykz><yht>妻</yht><yni>择</yni><yku>㐂</yku><y86>迅</y86>
接着看抓包后的style.css,进行css解密
核心代码:
  1. # 定义正则表达式,适配多行内容
  2. order_pattern = re.compile(r"(\S+)\s*\{\s*order:\s*(\d+);?\s*\}")
  3. xpath_pattern = re.compile(r"\.(\S+)\s(\S+)\s+\{")

  4. css_file = cssutils.parseFile('style.css')


  5. selector = etree.HTML(text=html_text)
  6. name_list = set()

  7. for rule in css_file:
  8.     x_matches = xpath_pattern.findall(rule.cssText)
  9.     if len(x_matches)>=1:
  10.         class_name = x_matches[0][0]
  11.         tag_name = x_matches[0][1]

  12.         if "::" in tag_name:
  13.             tag_a = tag_name.split("::")[1]
  14.             tag_name = tag_name.split("::")[0]
  15.             # print(tag_a)
  16.             name_list.add(tag_a)
  17.             xpath_ste = '//*[@class="{}"]/{}'.format(class_name, tag_name)
  18.             tag = selector.xpath(xpath_ste)[0]
  19.             if tag_a =='before':
  20.                 t = tag.text
  21.                 if t is None:
  22.                     t = ''
  23.                 c = re.search('content:\s(\S+)\\n',rule.cssText)

  24.                 if c:
  25.                     c = c.group(1)
  26.                     if c.find('attr')!=-1:
  27.                         a = re.search('attr\((\S+)\)',c).group(1)
  28.                         v = tag.attrib.get(a)
  29.                         t = "{}{}".format(v,t)
  30.                     else:
  31.                         t = "{}{}".format(c.replace('"',''),t)
  32.                     tag.text = t
  33.                         # print('debugger')
  34.             elif tag_a =='first-letter':
  35.                 if rule.cssText.find('font-size: 0') != -1:
  36.                     t = tag.text[1:]
  37.                     tag.text = t
  38.                     # print(t)
  39.             elif tag_a =='after':
  40.                 t = tag.text
  41.                 if t is None:
  42.                     t = ''
  43.                 c = re.search('content:\s(\S+)\\n', rule.cssText)

  44.                 if c:
  45.                     c = c.group(1)
  46.                     if c.find('attr') != -1:
  47.                         a = re.search('attr\((\S+)\)', c).group(1)
  48.                         v = tag.attrib.get(a)
  49.                         t = "{}{}".format(t,v)
  50.                     else:
  51.                         t = "{}{}".format(t, c.replace('"',''))
  52.                     tag.text = t
  53.                         # print('debugger')

  54.     else:

  55.         if rule.cssText.find('font-size: 0') != -1:
  56.             class_name = re.search('\.(\S+)\s\{',rule.cssText).group(1)
  57.             xpath_ste = '//*[@class="{}"]'.format(class_name)
  58.             tag_list = selector.xpath(xpath_ste)
  59.             for tag in tag_list:
  60.                 tag.text = ''
  61.         
  62.     if rule.cssText.find('scalex') != -1:
  63.         # 这里是字符反转 未作指定使用
  64.         pass
复制代码

接着,遍历CSS文件中的规则并提取键值对
  1. order_key_value_pairs = []
  2. for rule in css_file:
  3.     order_matches = order_pattern.findall(rule.cssText)
  4.     if order_matches:
  5.         order_key_value_pairs.extend(order_matches)
复制代码

最后的处理:
  1. order_key_value_dict = {key: int(value) for key, value in order_key_value_pairs}

  2. selector = etree.HTML(text=modified_html_string)
  3. p_list = selector.xpath('/html/body/p')
  4. html_text_list = []
  5. for p in p_list:
  6.     if p.attrib.get('class'):
  7.         name_list = p.xpath('./*')
  8.         # 按照 order_key_value_dict 中的值对 name_list 进行排序
  9.         name_list_sorted = sorted(name_list, key=lambda x: order_key_value_dict.get(x.tag, float('inf')))
  10.         text = "".join([etree.tostring(_, pretty_print=True).decode('utf-8') for _ in name_list_sorted])
  11.         # 创建一个新的 <p> 元素,保留原来的属性
  12.         new_p = etree.Element(p.tag, attrib=p.attrib)
  13.         for element in name_list_sorted:
  14.             new_p.append(element)

  15.         # 将新的 <p> 元素序列化为字符串
  16.         result = etree.tostring(new_p, pretty_print=True).decode('utf-8')
  17.     else:
  18.         result = etree.tostring(p, pretty_print=True).decode('utf-8')
  19.     # print(result)
  20.     html_text_list.append(result)
  21. html_text = "".join(html_text_list)
  22. # print(html_text)
  23. #
  24. old_woff = UniversalFontRecognition('fixed.m4mlmb6d.woff2')
  25. random_woff = UniversalFontRecognition('font.ttf')
  26. old_recognition_result = old_woff.crack()
  27. random_recognition_result = random_woff.crack()

  28. def replace_with_upper(match):

  29.     r = old_recognition_result.get(int(match.group(1)))
  30.     if r:
  31.         return r
  32.     else:
  33.         return random_recognition_result.get(int(match.group(1)))
  34. new_text = re.sub("&#(\d+?)\;", replace_with_upper, html_text)
  35. print(new_text)
复制代码

得到最终内容,使用了自定义字体,但由于字体是随机的,无法进行打表解密
不过可以参考以上的JS+CSS解密流程,仅供交流学习使用

评论7

kissxhhLv.4 发表于 2025-1-31 13:40:15 | 查看全部
路҈̣͉̮͍̩̜̞̘̥̟̜̠͔̭̭̙̯̘̞̥̣̓́̒̓̋͌͗̊̅̓̓͌͆̑͐̀ͅ过҉̲̤̖̳̙̠̫̘̯̤̦͓̘̰̐͐̋̍̉̓͑̃̏͌͂͌̍͗́̊̍͒͆̚ͅ

没个性,不签名!
回复

使用道具 举报

永恒唯一Lv.4星光赞助(未赞助不可申请) 发表于 2025-1-31 15:43:10 来自手机 | 查看全部
小白 路过。
回复

使用道具 举报

HMuSeaBLv.3 发表于 2025-1-31 18:13:52 来自手机 | 查看全部
路过
回复

使用道具 举报

1050148346 发表于 2025-1-31 18:47:01 来自手机 | 查看全部
感谢分享
回复

使用道具 举报

xiaoaiai919Lv.4星光赞助(未赞助不可申请) 发表于 2025-1-31 20:30:43 来自手机 | 查看全部
路过
回复

使用道具 举报

怣疯Lv.9星光赞助(未赞助不可申请) 发表于 2025-2-2 22:30:34 来自手机 | 查看全部
看看
回复

使用道具 举报

风中追枫星光赞助(未赞助不可申请) 发表于 3 天前 来自手机 | 查看全部
感谢分享
回复

使用道具 举报

回复

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

投诉/建议联系

ysqbbs@outlook.com

未经授权禁止转载,复制和建立镜像,
如有违反,追究法律责任
  • 关注公众号
  • 添加QQ群
Copyright © 2001-2025 源社区 版权所有 All Rights Reserved.
关灯 在本版发帖
扫一扫添加QQ群
返回顶部
快速回复 返回顶部 返回列表