Shamir秘密共享方案 (Python)
Shamir’s Secret Sharing scheme is an important cryptographic algorithm that allows private information— “secrets” — to be distributed securely amongst an untrusted network.
Shamir’s method for secret sharing relies on polynomial interpolation, which is an algebraic method of estimating unknown values in a gap between two known data points — without needing to know anything about what is on either side of those points.
SSS encodes a “secret” into a polynomial, then splits it into pieces and distributes it It’s possible to use polynomial interpolation to efficiently reconstruct that secret without requiring every single share. Instead only the threshold is needed, which provides enough points of data to correctly estimate the values between gaps in the encrypted shares.
REF: https://medium.com/keylesstech/a-beginners-guide-to-shamir-s-secret-sharing-e864efbf3648
Shamir秘密共享方案,叫做Shamir Secret Sharing, SSS。是由Shamir提出的一个分享密钥(本文秘密和“密钥”同义)的局部、并共同计算密钥的方法。
设计目的
假设公司A,B,C有一个密钥可以打开共同的仓库房门,为了防止保管人不在,或者被侵害,或者钥匙被偷,或者监守自盗。需要设计一个分享秘密的方案。
一个直观的方案就是将秘密分开为3份,给ABC各自钥匙的一部分,他们的子密钥合起来才能打开房门。
数学定义
引入一个临界点(Threshold,也叫门槛)的概念。N个分享秘密的人,只要凑够k个人(k<=N)就可以重建秘密。k就是这个临界点。
k<N,大大增加了使用的便利性和可扩展性。任意的k个人合起来都可以重建秘密,少于k个人无法重建秘密。
k=N,表明方案中所有的参与者需要贡献出自己的子秘密,才能合成所需的秘密。
举例
一个(k,N) 临界点方案,其共享秘密是 S.
对一个k-1次的多项式,取 N个不一样的点(i,f(i))。那么只要凑够 k个点就可以接出系数(a0,a1,……ak-1)。

只要把N个点分给N个人,设某个系数为共同秘密(如a0是秘密),那么就等于实现了SSS算法。
秘密分配及还原过程
首先介绍一个小费马定理:
引申为:
那么有:
秘密碎片生成:
构造一个多项式

其中,S为我们的秘密,p为素数,且S < p
取w个不相等的x,带入F(x)中,得到w组(xi,yi),分配给w个人
公开p,销毁多项式,每个人负责保密自己的(xi,yi)
秘密恢复:
当x=0时,F(0)=S,即可恢复出S
将t组(xi,yi)带入下式即可

其中,负一次方为该项模p的逆
将t组(xi,yi)带入即可得到S
示例:
假设我们有w=4个人,设定至少t=3人才能恢复秘密。
秘密S=2,p=23
构造
取x1=1,x2=2,x3=3,x4=4
带入得y1=7,y2=16, y3=6,y4=0
利用3组进行恢复(1,7) (3,6) (4,0)

计算可得到S=2
一个简单的恢复脚本
1 # coding:utf-8
2
3
4 def oj(a, n): # 求逆的函数
5 a = a % n
6 s = [0, 1]
7 while a != 1:
8 if a == 0:
9 return 0
10 q = n/a
11 t = n % a
12 n = a
13 a = t
14 s += [s[-2] - q * s[-1]]
15 return s[-1]
16
17 p = 23
18 m = ((4, 0),
19 (3, 6),
20 (2, 16))
21 r = (
22 m[0][1] * (0 - m[1][0]) * (0 - m[2][0]) * oj((m[0][0] - m[1][0]) * (m[0][0] - m[2][0]), p) +
23 m[1][1] * (0 - m[0][0]) * (0 - m[2][0]) * oj((m[1][0] - m[0][0]) * (m[1][0] - m[2][0]), p) +
24 m[2][1] * (0 - m[0][0]) * (0 - m[1][0]) * oj((m[2][0] - m[0][0]) * (m[2][0] - m[1][0]), p)
25 ) % p
26 print r
另外一个脚本
1 import Crypto.Util.number as numb
2 import random
3
4
5 # 求逆的函数,之前的版本用python2写的,这次用的python3,只把整除符号改了一下
6 def oj(a, n):
7 a = a % n
8 s = [0, 1]
9 while a != 1:
10 if a == 0:
11 return 0
12 q = n // a
13 t = n % a
14 n = a
15 a = t
16 s += [s[-2] - q * s[-1]]
17 return s[-1]
18
19
20 # max_length 为p的长度,同时也是秘密的最大长度
21 # secret_is_text =0 默认输入时文本, 非0时认为是数字
22 # p 默认为0, 会根据max_length 自动生成,不为0时直接使用,需要保证p为素数, 函数内没有素性检验
23 def create(max_length=513, secret_is_text=0, p=0):
24 if not p:
25 p = numb.getPrime(max_length)
26
27 w = int(input("请输入秘密保存人数:"))
28 t = int(input("请输入秘密恢复所需人数:"))
29 while not (t > 0 and t <= w):
30 t = int(input("请重新输入:"))
31 s = input("请输入你的秘密:")
32
33 if secret_is_text:
34 s = numb.bytes_to_long(s.encode("utf-8"))
35 else:
36 try:
37 s = int(s)
38 except Exception as e:
39 s = numb.bytes_to_long(s.encode("utf-8"))
40
41 x_list = list()
42 a_list = list()
43 i = w
44 while i > 0:
45 x = random.randint(p // 2, p) # 该范围没有特定限制,如果想让xi,yi取小一点儿的话可把范围写小点儿,但是要大于w
46 if x not in x_list:
47 x_list.append(x)
48 i -= 1
49
50 for a in range(t):
51 a_list.append(random.randint(p // 2, p)) # 同上
52
53 result = list()
54 for x in x_list:
55 y = s
56 for a_n in range(t):
57 a = a_list[i]
58 y += a * pow(x, i + 1, p)
59 result.append((x, y))
60 return t, p, result
61
62
63 # get_text=1 默认恢复为字符串,若想得到数字填0
64 def restore(p, information, get_text=1):
65
66 x_list = list()
67 y_list=list()
68 for x, y in information:
69 x_list.append(x)
70 y_list.append(y)
71
72 s = 0
73 for x_i in range(len(x_list)):
74 tmp_num = y_list[x_i]
75 x_i_j = 1
76 for x_j in range(len(x_list)):
77 if x_i != x_j:
78 tmp_num = tmp_num * (0 - x_list[x_j]) % p
79 x_i_j *= x_list[x_i] - x_list[x_j]
80 tmp_num = tmp_num * oj(x_i_j, p) % p
81 s += tmp_num
82
83 s = s % p
84 print(s)
85 if get_text:
86 try:
87 s = numb.long_to_bytes(s)
88 s = s.decode("utf-8")
89 except Exception as e:
90 print(e)
91
92 return s
93
94
95 t, p, result = create() #result为秘密碎片的列表
96 print(result)
97 print(restore(p, result[:t])) #这里我取了result的前t个,实际中可以取任意t个。
Shamir秘密共享方案 (Python)的更多相关文章
- 二级域名session 共享方案
二级域名session 共享方案 1.利用COOKIE存放session_id(); 实例: 域名一文件php代码: <?php session_start(); setcookie(&qu ...
- 第7.12节 可共享的Python类变量
第7.12节 可共享的Python类变量 一. 引言 在上节已经引入介绍了类变量和实例变量,类体中定义的变量为类变量,默认属于类本身,实例变量是实例方法中定义的self对象的变量,对于每个实例都 ...
- 多War项目中静态文件的共享方案
[原创申明:文章为原创,欢迎非盈利性转载,但转载必须注明来源] 在互联网产品中,一般会有多个项目(Jar.WAR)组成一个产品线.这些WAR项目,因为使用相同的前端架构(jQuery.easyui等) ...
- 项目分布式部署那些事(2):基于OCS(Memcached)的Session共享方案
在不久之前发布了一篇"项目分布式部署那些事(1):ONS消息队列.基于Redis的Session共享,开源共享",因为一些问题我们使用了阿里云的OCS,下面就来简单的介绍和分享下相 ...
- 细说tomcat之集群session共享方案
1. Tomcat Cluster官网:http://tomcat.apache.org/tomcat-7.0-doc/cluster-howto.htmlTomcat原生支持的集群方案,通过组播消息 ...
- nginx+php负载均衡集群环境中的session共享方案梳理
在网站使用nginx+php做负载均衡情况下,同一个IP访问同一个页面会被分配到不同的服务器上,如果session不同步的话,就会出现很多问题,比如说最常见的登录状态. 下面罗列几种nginx负载均衡 ...
- Tomcat利用MSM实现Session共享方案解说
Session共享有多种解决方法,常用的有四种:1)客户端Cookie保存2)服务器间Session同步3)使用集群管理Session(如MSM) 4)把Session持久化到数据库 针对上面Sess ...
- netty做集群 channel共享 方案
netty做集群 channel如何共享? 方案一: netty 集群,通过rocketmq等MQ 推送到所有netty服务端, channel 共享无非是要那个通道都可以发送消息向客户端, MQ广播 ...
- 西南大学网络实现路由器WIFI共享方案(一号多用户共享)
背景: 学校更换网页认证,限制多台设备的登录,后台记录发现会将账号封30min禁止登陆,于是想办法冲破这个限制.看到马丁大神的博客知道了学校的检测机制,只需要定时对账号进行认证下线即可实现,不被学校检 ...
随机推荐
- Graph-GraphSage
MPNN很好地概括了空域卷积的过程,但定义在这个框架下的所有模型都有一个共同的缺陷: 1. 卷积操作针对的对象是整张图,也就意味着要将所有结点放入内存/显存中,才能进行卷积操作.但对实际场景中的大规模 ...
- 01 . Go框架之Gin框架从入门到熟悉(路由和上传文件)
Gin框架简介 Gin是使用Go/Golang语言实现的HTTP Web框架, 接口简洁, 性能极高,截止1.4.0版本,包含测试代码,仅14K, 其中测试代码9K, 也就是说测试源码仅5k左右, 具 ...
- vue 404
问题描述:前端同事使用Vue.js框架,利用vue-route结合webpack编写了一个单页路由项目,运维协助在服务器端配置nginx.部署完成后,访问首页没问题,从首页里打开二级页面没问题,但是所 ...
- ORACLE 数据泵 expdp/impdp
ORACLE 数据泵 expdp/impdp 一.概念 Oracle Database 10g 引入了最新的数据泵(Data Dump)技术,数据泵导出导入 (EXPDP 和 IMPDP)的作用: 1 ...
- RuntimeError: one of the variables needed for gradient computation has been modified by an inplace operation
问题 在用pytorch跑生成对抗网络的时候,出现错误Runtime Error: one of the variables needed for gradient computation has b ...
- Js模块化开发的理解
Js模块化开发的理解 模块化是一个语言发展的必经之路,其能够帮助开发者拆分和组织代码,随着前端技术的发展,前端编写的代码量也越来越大,就需要对代码有很好的管理,而模块化能够帮助开发者解决命名冲突.管理 ...
- js音乐播放器【简洁】
辞职的第二天没有去找工作还,准备回家. 但到了火车站才发现沃特玛的买的票不是在这个火车站坐. 这就耽误了行程...... 说出来真舒服!!!淦 代码 这里已经上传到码云了,大家可以直接引用. 目前只有 ...
- NOI2020D1T1美食家
传送门:QAQQAQ 完了完了NOI签到题全班打不出来,真就全部成为时代的眼泪了... 首先$O(mT)$的$dp$显然,然后因为$T$很大$w$很小矩阵快速幂显然,但是有$k=200$卡不过去. 然 ...
- 手写Koa.js源码
用Node.js写一个web服务器,我前面已经写过两篇文章了: 第一篇是不使用任何框架也能搭建一个web服务器,主要是熟悉Node.js原生API的使用:使用Node.js原生API写一个web服务器 ...
- 知识补充之Django信号量
信号 抛砖引玉 1.如何对数据库的增加操作记录日志 2.信号比中间件的区别 Django中提供了"信号调度",用于在框架执行操作时解耦.通俗来讲,就是一些动作发生的时候,信号允许特 ...