DES算法的简单实现(python3)

2 篇文章 0 订阅
2 篇文章 1 订阅

Des算法的简单实现

# -*- coding:utf-8 -*- 
#Author: lengqiuqiu

# 字节转换
def dex_to_bit(data, n=0):
    output = []
    while data != 0:
        output.append(data % 2)
        data >>= 1
    if n > 0 and len(output) < n:
        for i in range(len(output), n):
            output.append(0)
    output.reverse()
    return output

# bytes转化为位列表
def bytes_to_list(bdata):
    if len(bdata) != 8:
        bdata = bdata + int(0).to_bytes(8 - len(bdata), byteorder='big', signed=False)
        # bdata = bdata + b'0x0' * (8 - len(bdata))
    return dex_to_bit(int.from_bytes(bdata, byteorder='big', signed=False), 64)

# 位列表转化为bytes
def list_to_bytes(ldata):
    temp=''
    while len(ldata) != 0:
        temp=temp + str(ldata.pop(0))
    output=int(temp, 2)
    return output.to_bytes(8, byteorder='big', signed=False)

# 置换算法
def _substitution(plaintext, ReplacementTable):
    ciphertext=[]
    for i in range(0, len(ReplacementTable)):
        ciphertext.append(plaintext[ReplacementTable[i]-1])
    return ciphertext

# 初始置换IP
def IP(plaintext):
    # 定义初始置换表
    ReplacementTable=[58, 50, 42, 34, 26, 18, 10, 2, 60, 52, 44, 36, 28, 20, 12, 4, 62, 54, 46, 38, 30, 22, 14, 6, 64, 26, 48, 40, 32, 24, 16, 8, 57, 49, 41,
                        33, 25, 17, 9, 1, 59, 51, 43, 35, 27, 19, 11, 3, 61, 53, 45, 37, 29, 21, 13, 5, 63, 55, 47, 39, 31, 23, 15, 7]
    return _substitution(plaintext, ReplacementTable)

# 逆初始置换IP
def re_IP(plaintext):
    # 定义逆初始置换表
    Re_ReplacementTable=[40, 8, 48, 16, 56, 24, 64, 32,
                           39, 7, 47, 15, 55, 23, 63, 31,
                           38, 6, 46, 14, 54, 22, 62, 30,
                           37, 5, 45, 13, 53, 21, 61, 29,
                           36, 4, 44, 12, 52, 20, 60, 28,
                           35, 3, 43, 11, 51, 19, 59, 27,
                           34, 2, 42, 10, 50, 18, 58, 26,
                           33, 1, 41, 9, 49, 17, 57, 25]
    return _substitution(plaintext, Re_ReplacementTable)

# E-拓展运算
def E(plaintext32):
    EBox=[32, 1, 2, 3, 4, 5,
            4, 5, 6, 7, 8, 9,
            8, 9, 10, 11, 12, 13,
            12, 13, 14, 15, 16, 17,
            16, 17, 18, 19, 20, 21,
            20, 21, 22, 23, 24, 25,
            24, 25, 26, 27, 28, 29,
            28, 29, 30, 31, 32, 1]
    plaintext48=_substitution(plaintext32, EBox)
    return plaintext48

# S-盒运算
def S(plaintext):
    outtext=[]
    s1=[[14, 4, 13, 1, 2, 15, 11, 8, 3, 10, 6, 12, 5, 9, 0, 7],
          [0, 15, 7, 4, 14, 2, 13, 1, 10, 6, 12, 11, 9, 5, 3, 8],
          [4, 1, 14, 8, 13, 6, 2, 11, 15, 12, 9, 7, 3, 10, 5, 0],
          [15, 12, 8, 2, 4, 9, 1, 7, 5, 11, 3, 14, 10, 0, 6, 13]]
    s2=[[15, 1, 8, 14, 6, 11, 3, 4, 9, 7, 2, 13, 12, 0, 5, 10],
          [3, 13, 4, 7, 15, 2, 8, 14, 12, 0, 1, 10, 6, 9, 11, 5],
          [0, 14, 7, 11, 10, 4, 13, 1, 5, 8, 12, 6, 9, 3, 2, 15],
          [13, 8, 10, 1, 3, 15, 4, 2, 11, 6, 7, 12, 0, 5, 14, 9]]
    s3=[[10, 0, 9, 14, 6, 3, 15, 5, 1, 13, 12, 7, 11, 4, 2, 8],
          [13, 7, 0, 9, 3, 4, 6, 10, 2, 8, 5, 14, 12, 11, 15, 1],
          [13, 6, 4, 9, 8, 15, 3, 0, 11, 1, 2, 12, 5, 10, 14, 7],
          [1, 10, 13, 0, 6, 9, 8, 7, 4, 15, 14, 3, 11, 5, 2, 12]]
    s4=[[7, 13, 14, 3, 0, 6, 9, 10, 1, 2, 8, 5, 11, 12, 4, 15],
          [13, 8, 11, 5, 6, 15, 0, 3, 4, 7, 2, 12, 1, 10, 14, 9],
          [10, 6, 9, 0, 12, 11, 7, 13, 15, 1, 3, 14, 5, 2, 8, 4],
          [3, 15, 0, 6, 10, 1, 13, 8, 9, 4, 5, 11, 12, 7, 2, 14]]
    s5=[[2, 12, 4, 1, 7, 10, 11, 6, 8, 5, 3, 15, 13, 0, 14, 9],
          [14, 11, 2, 12, 4, 7, 13, 1, 5, 0, 15, 10, 3, 9, 8, 6],
          [4, 2, 1, 11, 10, 13, 7, 8, 15, 9, 12, 5, 6, 3, 0, 14],
          [11, 8, 12, 7, 1, 14, 2, 13, 6, 15, 0, 9, 10, 4, 5, 3]]
    s6=[[12, 1, 10, 15, 9, 2, 6, 8, 0, 13, 3, 4, 14, 7, 5, 11],
          [10, 15, 4, 2, 7, 12, 9, 5, 6, 1, 13, 14, 0, 11, 3, 8],
          [9, 14, 15, 5, 2, 8, 12, 3, 7, 0, 4, 10, 1, 13, 11, 6],
          [4, 3, 2, 12, 9, 5, 15, 10, 11, 14, 1, 7, 6, 0, 8, 13]]
    s7=[[4, 11, 2, 14, 15, 0, 8, 13, 3, 12, 9, 7, 5, 10, 6, 1],
          [13, 0, 11, 7, 4, 9, 1, 10, 14, 3, 5, 12, 2, 15, 8, 6],
          [1, 4, 11, 13, 12, 3, 7, 14, 10, 15, 6, 8, 0, 5, 9, 2],
          [6, 11, 13, 8, 1, 4, 10, 7, 9, 5, 0, 15, 14, 2, 3, 12]]
    s8=[[13, 2, 8, 4, 6, 15, 11, 1, 10, 9, 3, 14, 5, 0, 12, 7],
          [1, 15, 13, 8, 10, 3, 7, 4, 12, 5, 6, 11, 0, 14, 9, 2],
          [7, 11, 4, 1, 9, 12, 14, 2, 0, 6, 10, 13, 15, 3, 5, 8],
          [2, 1, 14, 7, 4, 10, 8, 13, 15, 12, 9, 0, 3, 5, 6, 11]]
    SBox=[s1, s2, s3, s4, s5, s6, s7, s8]
    for i in range(0, 48, 6):
        col=int(str(plaintext[i])+str(plaintext[i+5]), 2)
        row=int(str(plaintext[i+1])+str(plaintext[i+2]) +
                  str(plaintext[i+3])+str(plaintext[i+4]), 2)
        temp=dex_to_bit(SBox[i // 6][col][row], 4)
        outtext.extend(temp)
    return outtext

# P-盒置换
def P(plaintext):
    PBox=[16, 7, 20, 21, 29, 12, 28, 17, 1, 15, 23, 26, 5, 18, 31,
            10, 2, 8, 24, 14, 32, 27, 3, 9, 19, 13, 30, 6, 22, 11, 4, 25]
    return _substitution(plaintext, PBox)

# 置换选择PC-1
def PC_1(keys):
    PC_1Box=[57, 49, 41, 33, 25, 17, 9,  1,
               58, 50, 42, 34, 26, 18, 10, 2,
               59, 51, 43, 35, 27, 19, 11, 3,
               60, 52, 44, 36, 63, 55, 47, 39,
               31, 23, 15, 7,  62, 54, 46, 38,
               30, 22, 14, 6,  61, 53, 45, 37,
               29, 21, 13, 5,  28, 20, 12, 4]
    return _substitution(keys, PC_1Box)

# 置换选择PC-2
def PC_2(keys):
    PC_2Box=[14, 17, 11, 24, 1, 5,
               3, 28, 15, 6, 21, 10,
               23, 19, 12, 4, 26, 8,
               16, 7, 27, 20, 13, 2,
               41, 52, 31, 37, 47, 55,
               30, 40, 51, 45, 33, 48,
               44, 49, 39, 56, 34, 53,
               46, 42, 50, 36, 29, 32]
    return _substitution(keys, PC_2Box)

#密钥初始化
def KeyInit(keys):
    keys56=PC_1(keys)
    lkeys28=keys56[0:28]
    rkeys28=keys56[28:56]
    return [lkeys28, rkeys28]

#轮密钥生成
def GenerateRoundKey(key, RoundNumber):
    ROL_TIMES=[1, 1, 2, 2, 2, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 1]
    #for i in range(0, RoundNumber):
    for j in range(0, ROL_TIMES[RoundNumber]):
        key[0].append(key[0].pop(0))
        key[1].append(key[1].pop(0))
    keys56=key[0]+key[1]
    keys48=PC_2(keys56)
    return keys48

#轮密钥生成
def DeGenerateRoundKey(key, RoundNumber):
    ROL_TIMES=[0, 1, 2, 2, 2, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 1]
    #for i in range(0, RoundNumber):
    for j in range(0, ROL_TIMES[RoundNumber]):
        key[0].insert(0,key[0].pop(len(key[0]) -1))
        key[1].insert(0,key[1].pop(len(key[1] )-1))
    keys56=key[0]+key[1]
    keys48=PC_2(keys56)
    return keys48

# 异或运算
def XOR(list1, list2):
    output=[]
    for i in range(0, len(list1)):
        output.append(list1[i] ^ list2[i])
    return output

#文件输入
def input_to_file(file_name):
    with open(file_name, 'rb') as fp:
        return fp.read()

#加密
def DES(filename,k):
    outname=filename+'.des'
    text=input_to_file(filename)
    key=k
    size = len(text) // 8 if len(text) % 8 == 0 else len(text) // 8 + 1
    keys=bytes_to_list(key)
    f=open(outname, 'bw')
    for i in range(0, size):
        temp_text=text[8*i:8*i+8]
        plaintext=bytes_to_list(temp_text)
        out = _des(plaintext,keys)
        bout = list_to_bytes(out)
        f.write(bout)
    f.close()

#加密
def _des(plaintext, keys):
    plaintext64=IP(plaintext)
    lplaintext32=plaintext64[0:32]
    rplaintext32=plaintext64[32:64]

    key=KeyInit(keys)
    for i in range(0, 16):
        plaintext48=E(rplaintext32)
        keys48=GenerateRoundKey(key[:], i)
        temp=XOR(plaintext48, keys48)
        plaintext32=S(temp)
        plaintext32=P(plaintext32)
        temp=XOR(plaintext32, lplaintext32)
        lplaintext32=rplaintext32
        rplaintext32=temp

    plaintext64=rplaintext32 + lplaintext32
    cryptytext=re_IP(plaintext64)
    return cryptytext

#解密,可优化,但是我懒
def DES_Decode(filename,k):
    outname=filename+'.out'
    text=input_to_file(filename)
    key=k
    size = len(text) // 8
    keys=bytes_to_list(key)
    f=open(outname, 'bw')
    for i in range(0, size - 1):
        temp_text=text[8*i:8*i+8]
        plaintext=bytes_to_list(temp_text)
        out = _des_decode(plaintext,keys)
        bout = list_to_bytes(out)
        f.write(bout)
    temp_text=text[8*(size - 1):8*(size - 1)+8]
    plaintext=bytes_to_list(temp_text)
    out = _des_decode(plaintext,keys)

    bout = list_to_bytes(out)
    f.write(bout)

    f.close()

#解密,可优化,但是我懒
def _des_decode(plaintext, keys):
    plaintext64=IP(plaintext)
    lplaintext32=plaintext64[0:32]
    rplaintext32=plaintext64[32:64]

    key=KeyInit(keys)

    for i in range(0, 16):
        plaintext48=E(rplaintext32)
        keys48=DeGenerateRoundKey(key[:], i)
        temp=XOR(plaintext48, keys48)
        plaintext32=S(temp)
        plaintext32=P(plaintext32)
        temp=XOR(plaintext32, lplaintext32)
        lplaintext32=rplaintext32
        rplaintext32=temp

    plaintext64=rplaintext32 + lplaintext32
    cryptytext=re_IP(plaintext64)
    return cryptytext

import time
import os
import sys

#简单界面
def Main():
    os.system('cls')
    print('********************欢迎使用DES加密解密工具********************')
    print('************************作者: lqq****************************')
    print('请选择功能:')
    print('1.加密   2.解密   3.退出')
    s = 0
    try:
        s = int(input())
        if s != 1 and s != 2 and s != 3:
            raise Exception("")
    except:
        os.system('cls')
        print('请输入正确的数字')
        time.sleep(3)
        sys.exit(0)
    if s == 3:
        sys.exit(0)
    elif s == 1:
        print('请输入要加密的文件路径:')
        filename = input()
        if filename == '':
            print('请输入路径')
            sys.exit(1)
        print('请输入密钥:')
        key = input()
        keys = key.encode()
        while len(keys) != 8:
            print('密钥长度为64bit,请重新输入')
            key = input()
            keys = key.encode()

        print('请选择加密模式:')
        print('1.ECB    2.CBC   3.CFB   4.OFB')
        try:
            ms = int(input())
            if ms < 1 and ms > 4:
                raise Exception("")
        except:
            os.system('cls')
            print('请输入正确的数字')
            time.sleep(3)
            sys.exit(0)
        if ms == 1:
            DES(filename,keys)
        elif ms == 2:
            pass
        elif ms == 3:
            pass
        elif ms == 4:
            pass
    if s == 3:
        sys.exit(0)
    elif s == 2:   
        print('请输入要解密的文件路径:')
        filename = input()
        if filename == '':
            print('请输入路径')
            sys.exit(1)
        print('请输入密钥:')
        key = input()
        keys = key.encode()
        while len(keys) != 8:
            print('密钥长度为64bit,请重新输入')
            key = input()
            keys = key.encode()

        print('请选择加密模式:')
        print('1.ECB    2.CBC   3.CFB   4.OFB')
        try:
            ms = int(input())
            if ms < 1 and ms > 4:
                raise Exception("")
        except:
            os.system('cls')
            print('请输入正确的数字')
            time.sleep(3)
            sys.exit(0)
        if ms == 1:
            DES_Decode(filename,keys)
        elif ms == 2:
            pass
        elif ms == 3:
            pass
        elif ms == 4:
            pass

Main()

时间有限,能力有限,为了应付作业随便写的代码,可能会存在许多的问题,如果小伙伴们在参考的时候发现存在问题也希望可以反馈一下。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值