我爸电脑上有个加密压缩包,我用Python给解开了

老爸说他有个照片文件夹打不开了,让我过去看看,一瞅,好家伙,加密压缩包尘封老照片呀。

既然加密,没准还有意外收货。

在这里插入图片描述

作为一个“精灵鬼”,这么有价值的数据,我必须帮老爸解开呀。

寻找思路

解密压缩包的思路是什么?
答:通过各种密码去尝试解压文件。

用什么解压文件?
答:zip 使用 zipfilerar 使用 rarfile,已经有 Python 大佬给我们写好啦,只需要调用它们的方法即可。

密码从哪里找?
答:程序自行运算或者找密码本。

思路整理清楚之后,就可以开始了。

zipfile 与 rarfile

zipfile:Python 内置,无需安装;

rarfile:需要安装一下,文档参照:https://rarfile.readthedocs.io/api.html。

解压文件使用二者生成对象的 extractall 方法即可。

以下内容以 zipfile 库进行举例,举一反三即可。

先默默通过 zipfile 解压一个没有密码的文件,试试手感。

测试文件自己进行打包压缩即可,先尝试英文或者数字文件命名,在尝试中文命名。

import zipfile

try:
    # 创建 ZipFile 对象
    with zipfile.ZipFile('测试文件.zip') as zfile:
        # 解压文件
        zfile.extractall(path='./')
        print('文件解压成功')
except:
    print('失败啦!')

完成任务,成功解压文件,zip 文件和 python 代码放置在同一目录。

解压带密码的文件

下面假装不知道密码,通过密码进行解压,设置密码为 1234

import zipfile

try:
    # 创建 ZipFile 对象
    with zipfile.ZipFile('511.zip') as zfile:
        # 解压文件
        zfile.extractall(path='./', pwd=b"1234")
        print('文件解压成功')
except:
    print('失败啦!')

成功解压,后文就可以通过这种思路,对压缩包进行解压。

中文乱码问题
在测试的时候,还发现了如果文件名是中文,解压之后文件名出现乱码情况,修正它。

找到 Python 安装中的 Lib 文件夹,然后打开 zipfile.py 文件,直接修改源码。

搜索 fname_str = fname.decode("cp437"),再后面添加如下内容。

fname_str = fname_str.encode("cp437").decode('gbk')

搜索 filename = filename.decode('cp437'),再后面追加如下代码。

filename = filename.encode("cp437").decode('gbk')

保存该文件,再对测试文件进行解压,解压成功,文件名无错误。

解密文件

进入正式环节,接下来就要解密老爸的压缩包了,这么有价值的压缩包,可别打不开。

假设老爸的密码是 4 位,可以直接编写如下代码进行测试。

提前准备测试压缩包,密码为了好破解设置为 aaaf,代码如下:

import zipfile


def ext_file(pwd):

    try:
        # 创建 ZipFile 对象
        with zipfile.ZipFile('测试中文.zip') as zfile:
            # 解压文件
            zfile.extractall(path='./', pwd=pwd.encode('utf-8'))
            print('文件解压成功')
            return True
    except Exception as e:
        print('失败啦!', e)
        return False

# 先“细致入微”的实现一下,以后有好思路在修改
def get_pwds(my_password_str):
    for i1 in range(len(my_password_str)):
        for i2 in range(len(my_password_str)):
            for i3 in range(len(my_password_str)):
                for i4 in range(len(my_password_str)):
                    yield my_password_str[i1] + my_password_str[i2] + my_password_str[i3] + my_password_str[i4]


if __name__ == '__main__':
    my_password_str = "abcdefghijklmnopqrstuvwxyz0123456789"
    for pwd in get_pwds(my_password_str):
        print("正在测试密码:", pwd)
        yield_pwd = pwd
        ret = ext_file(yield_pwd)
        if ret:
            print("解密成功,密码是", yield_pwd)
            break

非常简单的运行几秒之后,文件解压成功。

正在测试密码:aaaa
失败啦!Bad password for file '测试中文.txt'
正在测试密码:aaab
失败啦!Bad password for file '测试中文.txt'
正在测试密码:aaac
失败啦!Bad password for file '测试中文.txt'
正在测试密码:aaad
失败啦!Bad password for file '测试中文.txt'
正在测试密码:aaae
失败啦!Bad password for file '测试中文.txt'
正在测试密码:aaaf
文件解压成功
解密成功,密码是 aaaf

此时相信大佬们已经发现问题,如果密码不是 4 位,是不确定的位数,那我那一处细致入微的循环操作,就要修改了。
而且我不知道老爸的密码是多少位数的,这就有点难度了。

更加优秀的解法
在 Python 中已经内置好了一个迭代器,可用于从一个字符串中固定的取出指定位数的密码,测试代码如下:

import itertools
my_pwdstr = 'abcdefghijklmnopqrstuvwxyz0123456789'
def ret_pwd():
    for x in itertools.permutations(my_pwdstr, 4):
        yield ''.join(x)

for item in ret_pwd():
    print(item)

上述代码用到了 itertools.permutations ,该函数的用法如下:

# 函数原型
itertools.permutations(iterable, r=None)

连续返回由 iterable 元素生成长度为 r 的排列。

如果 r 未指定或为 Noner 默认设置为 iterable 的长度,这种情况下,生成所有排列。

所以使用上述代码,可以快速的生成指定位数的密码,后续只需要传入老爸密码的位数,即可不变动代码就进行程序测试了。

import zipfile
import itertools


def ext_file(pwd):

    try:
        # 创建 ZipFile 对象
        with zipfile.ZipFile('测试中文.zip') as zfile:
            # 解压文件
            zfile.extractall(path='./', pwd=pwd.encode('utf-8'))
            print('文件解压成功')
            return True
    except Exception as e:
        print('失败啦!', e)
        return False


def get_pwds(my_password_str, nums):
    for x in itertools.permutations(my_password_str, nums):
        yield ''.join(x)

if __name__ == '__main__':
    my_password_str = "abcdefghijklmnopqrstuvwxyz0123456789"
    for pwd in get_pwds(my_password_str, 4):
        print(len(pwd))
        print("正在测试密码:", pwd)
        yield_pwd = pwd
        ret = ext_file(yield_pwd)
        if ret:
            print("解密成功,密码是", yield_pwd)
            break

经过改良之后,你可以动态控制密码的长度了,如果需要特殊符号也可以继续扩充 my_password_str 字符串。

扩展思路

密码的获取一定要自己生成吗?
答:有种文件叫做密码本,所以去找到一些,然后逐行读取就好了。

一个个的解密太慢了,有高效的办法吗?
答:多线程或者多进程破解,一个进程读一个密码本,每个进程下面在启用几个线程去分段解析密码。

事件后续

代码虽然写好了,但文件解密我用多线程轮询了 2 天,还是没有解开(相当真实的体验了)。

由于不知道密码组成和密码位数,也不知道算到何年何月去了,我决定还是从老爸那里在找找突破口,“逼问了”一下常用的密码,尤其是知道其银行卡密码之后,得到重要突破点。

解密结果:最终的最终尝试一周之后,文件解开了,密码不复杂,姓的首字母+123789。
本因为是我的姓名+生日,最后得一波感动,然而我还是多虑啦,哈哈哈。

感兴趣的小伙伴,赠送全套Python学习资料,包含面试题、简历资料等具体看下方。

一、Python所有方向的学习路线

Python所有方向的技术点做的整理,形成各个领域的知识点汇总,它的用处就在于,你可以按照下面的知识点去找对应的学习资源,保证自己学得较为全面。

img
img

二、Python必备开发工具

工具都帮大家整理好了,安装就可直接上手!img

三、最新Python学习笔记

当我学到一定基础,有自己的理解能力的时候,会去阅读一些前辈整理的书籍或者手写的笔记资料,这些笔记详细记载了他们对一些技术点的理解,这些理解是比较独到,可以学到不一样的思路。

img

四、Python视频合集

观看全面零基础学习视频,看视频学习是最快捷也是最有效果的方式,跟着视频中老师的思路,从基础到深入,还是很容易入门的。

img

五、实战案例

纸上得来终觉浅,要学会跟着视频一起敲,要动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。

img

六、面试宝典

在这里插入图片描述

在这里插入图片描述

简历模板在这里插入图片描述
若有侵权,请联系删除
  • 19
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值