curve.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. package curve
  2. import (
  3. "errors"
  4. "fmt"
  5. "log"
  6. "math/big"
  7. )
  8. var (
  9. MinP = big.NewInt(11)
  10. )
  11. // 4a³ + 27b² = 0 - проверка кривой на сингулярность
  12. func New(a, b *big.Int) (*Curve, error) {
  13. c := &Curve{
  14. a: new(big.Int).Set(a),
  15. b: new(big.Int).Set(b),
  16. }
  17. return c, c.IsSingular()
  18. }
  19. type Curve struct {
  20. a *big.Int // константа a
  21. b *big.Int // константа b
  22. p *big.Int // размер конечного поля
  23. g *Point // базовая точка подгруппы
  24. n *big.Int // порядок подгруппы
  25. h *big.Int // кофактор подгруппы
  26. }
  27. func (s *Curve) A() (res *big.Int) {
  28. return intCopy(s.a)
  29. }
  30. func (s *Curve) B() (res *big.Int) {
  31. return intCopy(s.b)
  32. }
  33. // Проверка, установлен ли размер конечного поля
  34. func (s *Curve) IsValidP() (err error) {
  35. if err = s.IsValidConstants(); err == nil {
  36. if s.p == nil {
  37. err = errors.New("размер конечного поля не установлен")
  38. }
  39. }
  40. return
  41. }
  42. func (s *Curve) IsValidG() (err error) {
  43. if err = s.IsValidP(); err == nil {
  44. if s.g == nil {
  45. err = errors.New("базовая точка подгруппы g не определена")
  46. }
  47. }
  48. return
  49. }
  50. func (s *Curve) IsValidN() (err error) {
  51. if err = s.IsValidG(); err == nil {
  52. if s.n == nil {
  53. err = errors.New("порядок подгруппы не определен")
  54. }
  55. }
  56. return
  57. }
  58. // Возвращает строку уравнения кривой
  59. func (s *Curve) FormulaString() string {
  60. return fmt.Sprintf(
  61. "y² = x³ + %sx + %s",
  62. mustBigInt(s.a),
  63. mustBigInt(s.b),
  64. )
  65. }
  66. // Проверка кривой на сингулярность (4a³ + 27b² = 0)
  67. func (s *Curve) IsSingular() (err error) {
  68. if err = s.IsValidConstants(); err != nil {
  69. return
  70. }
  71. l := new(big.Int).Mul(new(big.Int).Exp(s.a, big.NewInt(3), nil), big.NewInt(4))
  72. r := new(big.Int).Mul(new(big.Int).Exp(s.b, big.NewInt(2), nil), big.NewInt(27))
  73. if l.Add(l, r).Cmp(big.NewInt(0)) == 0 {
  74. err = fmt.Errorf("кривая [ %s ] сингулярна и не рекомендуется к использованию", s.FormulaString())
  75. }
  76. return
  77. }
  78. // Проверка констант уравнения на валидность
  79. func (s *Curve) IsValidConstants() error {
  80. if s.a == nil {
  81. return errors.New("константа [ a ] не определена")
  82. }
  83. if s.b == nil {
  84. return errors.New("константа [ b ] не определена")
  85. }
  86. return nil
  87. }
  88. // Установка размера конечного поля
  89. func (s *Curve) SetP(p *big.Int) (err error) {
  90. if s.p != nil {
  91. return errors.New("резмер конечного поля уже определен")
  92. }
  93. if err = s.IsValidConstants(); err != nil {
  94. return
  95. }
  96. if !IsPrime(p) {
  97. err = fmt.Errorf("число [ %s ] не является простым. Размер конечного поля должен быть простым числом", p)
  98. return
  99. }
  100. s.p = p
  101. /*if !IsPrime(Div(Sub(p, 1), 2)) {
  102. }*/
  103. /*
  104. if p.Cmp(MinP) < 0 {
  105. err = fmt.Errorf("слишком маленький размер конечного поля [ %s ]", p)
  106. return
  107. }
  108. */
  109. //s.p = new(big.Int).Set(p)
  110. // сброс подгруппы
  111. s.n = nil
  112. s.h = nil
  113. return
  114. }
  115. func (s *Curve) Point(x, y *big.Int) *Point {
  116. return NewPoint(x, y, s)
  117. }
  118. func (s *Curve) InverseMod(k, p *big.Int) (res *big.Int, err error) {
  119. if err = s.IsValidP(); err != nil {
  120. return
  121. }
  122. if k == nil {
  123. err = errors.New("[ k ] не определено")
  124. return
  125. } else if rk := k.Cmp(intZero); rk == 0 {
  126. err = errors.New("[ k ] == 0. Деление на 0 невозможно")
  127. return
  128. } else if rk < 0 {
  129. // k ** -1 = p - (-k) ** -1 (mod p)
  130. // point - self.inverseMod(-k, point)
  131. var iMod *big.Int
  132. if iMod, err = s.InverseMod(Neg(k), p); err == nil {
  133. res = Sub(p, iMod)
  134. }
  135. } else {
  136. s, olds := big.NewInt(0), big.NewInt(1)
  137. t, oldt := big.NewInt(1), big.NewInt(0)
  138. r, oldr := intCopy(p), intCopy(k)
  139. for r.Cmp(intZero) != 0 {
  140. quot := Div(oldr, r)
  141. oldr, r = r, Sub(oldr, Mul(quot, r))
  142. olds, s = s, Sub(olds, Mul(quot, s))
  143. oldt, t = t, Sub(oldt, Mul(quot, t))
  144. }
  145. // gcd, x, y = old_r, old_s, old_t
  146. res = Mod(olds, p)
  147. }
  148. return
  149. }
  150. // Определение координат пары зеркальных точек по x
  151. func (s *Curve) points(dx *big.Int) (p1, p2 *Point, err error) {
  152. dxx := intCopy(dx)
  153. // Инициализируем результирующие точки пустыми
  154. p1, p2 = s.Point(nil, nil), s.Point(nil, nil)
  155. var dy *big.Int
  156. // Операции, обратной делению по модулю, не существует, поэтому
  157. // для определения координаты по y, необходимо, путем перебора с подстановкой y,
  158. // проверить равенство левой части уравнения, разделенного по модулю на p с
  159. // правой частью, так же разделенной по модулю на p
  160. for i := big.NewInt(0); i.Cmp(s.p) < 0; i.Add(i, big.NewInt(1)) {
  161. // i ** 2 % s.p == (dx ** 3 + s.a * dx + self.b) % self.p > 0
  162. l := Mod(Exp64(i, 2), s.p)
  163. r := Mod(
  164. Add(
  165. Exp64(dx, 3),
  166. Add(
  167. Mul(s.a, dx),
  168. s.b,
  169. ),
  170. ),
  171. s.p,
  172. )
  173. if l.Cmp(r) == 0 {
  174. dy = i
  175. break
  176. }
  177. }
  178. if dy == nil {
  179. err = fmt.Errorf("не удалось найти пару зеркальных точек на кривой после x [ %s ]", dxx)
  180. } else {
  181. p1.x, p1.y = dx, dy
  182. p2.x = intCopy(p1.x)
  183. p2.y = Mod(Neg(p1.y), s.p)
  184. }
  185. return
  186. }
  187. func (s *Curve) searhClosePoints(x *big.Int) (p1, p2 *Point, err error) {
  188. cx := intCopy(x)
  189. if x.Cmp(s.p) >= 0 {
  190. x = Sub(s.p, big.NewInt(1))
  191. } else if x.Cmp(intZero) < 0 {
  192. x = big.NewInt(0)
  193. }
  194. //kx := Sub64(x, 1)
  195. kx := intCopy(x)
  196. for {
  197. //log.Println("xkx", x, kx)
  198. if x.Cmp(s.p) >= 0 && kx.Cmp(intZero) <= 0 {
  199. err = fmt.Errorf("не удалось найти точки, близкие к [ %s ]", cx)
  200. break
  201. } else {
  202. // Поиск вправо
  203. if x.Cmp(s.p) <= 0 {
  204. if p1, p2, err = s.points(x); err == nil {
  205. if p1.IsInCurve() == nil {
  206. return
  207. }
  208. }
  209. x = Add64(x, 1)
  210. }
  211. // Поиск влево
  212. if kx.Cmp(intZero) < 0 {
  213. if p1, p2, err = s.points(kx); err == nil {
  214. if p1.IsInCurve() == nil {
  215. return
  216. }
  217. }
  218. kx = Sub64(kx, 1)
  219. }
  220. }
  221. }
  222. return
  223. }
  224. func (s *Curve) SetG(g *Point) (err error) {
  225. if s.g != nil {
  226. err = errors.New("базовая точка подгруппы g уже определена")
  227. return
  228. }
  229. if err = s.IsValidP(); err != nil {
  230. return
  231. }
  232. if err = g.IsInCurve(); err == nil && g.curve == s {
  233. s.g = g
  234. }
  235. return
  236. }
  237. func (s *Curve) SetGRandom() (err error) {
  238. if s.g != nil {
  239. err = errors.New("базовая точка подгруппы g уже определена")
  240. return
  241. }
  242. if err = s.IsValidP(); err != nil {
  243. return
  244. }
  245. x := Random(
  246. Div(s.p, big.NewInt(2)),
  247. Sub(s.p, big.NewInt(10)),
  248. )
  249. //log.Println(x)
  250. p1, p2, err := s.searhClosePoints(x)
  251. if err != nil {
  252. return
  253. }
  254. if err = p1.IsInCurve(); err == nil {
  255. s.g = p1
  256. } else if err = p2.IsInCurve(); err == nil {
  257. s.g = p2
  258. } else {
  259. err = errors.New("не удалось найти точку g, пренадлежащую кривой")
  260. }
  261. return
  262. }
  263. // Поиск порядка подгруппы
  264. func (s *Curve) SetN() (err error) {
  265. if s.n != nil {
  266. return errors.New("порядок подгруппы уже определен")
  267. }
  268. if err = s.IsValidP(); err != nil {
  269. return
  270. }
  271. dx, dy := intCopy(s.g.x), Mod(Neg(s.g.y), s.p)
  272. tmpG := s.g.Copy()
  273. s.n = big.NewInt(1)
  274. for {
  275. s.n.Add(s.n, big.NewInt(1))
  276. if tmpG, err = tmpG.Add(s.g); err != nil {
  277. return
  278. }
  279. if tmpG.x.Cmp(dx) == 0 && tmpG.y.Cmp(dy) == 0 {
  280. s.n.Add(s.n, big.NewInt(1))
  281. return
  282. }
  283. }
  284. }
  285. func (s *Curve) KeyPub(priv *big.Int) (pub *Point, err error) {
  286. if err = s.IsValidN(); err != nil {
  287. return
  288. }
  289. pub, err = s.g.Mul(priv)
  290. return
  291. }
  292. func (s *Curve) RandomKeyPair() (priv *big.Int, pub *Point, err error) {
  293. if err = s.IsValidN(); err != nil {
  294. return
  295. }
  296. priv = Random(big.NewInt(1), Sub64(s.n, 1))
  297. pub, err = s.g.Mul(priv)
  298. return
  299. }
  300. // Вычисление общего секрета
  301. func (s *Curve) PointSecret(mPriv *big.Int, fPub *Point) (res *Point, err error) {
  302. if err = s.IsValidN(); err != nil {
  303. return
  304. }
  305. res, err = fPub.Mul(mPriv)
  306. return
  307. }
  308. type ELKeyPriv struct {
  309. d *big.Int
  310. }
  311. type ELKeyPub struct {
  312. e1 *Point
  313. e2 *Point
  314. }
  315. type ELKeyPair struct {
  316. curve *Curve
  317. priv *ELKeyPriv
  318. pub *ELKeyPub
  319. dTable map[byte]*Point
  320. }
  321. func (s *ELKeyPair) setupDataTable() error {
  322. if s.dTable != nil {
  323. return nil
  324. }
  325. s.dTable = make(map[byte]*Point)
  326. dx := big.NewInt(0)
  327. //var points []*Point
  328. //var p *Point
  329. for i := byte(0); i <= 254; i++ {
  330. p, _, err := s.curve.searhClosePoints(dx)
  331. if err != nil {
  332. return err
  333. }
  334. // check already exists - todo
  335. dx = Add64(p.x, 1)
  336. s.dTable[i] = p
  337. log.Println(i, p)
  338. }
  339. return nil
  340. }
  341. type ELMessage struct {
  342. c1 *Point
  343. cd []*Point
  344. }
  345. // c1 = r x e1
  346. // c2 = P + r x e2
  347. func (s *ELKeyPair) EncodeMessage(data []byte) (mes *ELMessage, err error) {
  348. r := Random(big.NewInt(1), Sub64(s.curve.p, 1))
  349. c1, err := s.pub.e1.Mul(r)
  350. if err != nil {
  351. return nil, err
  352. }
  353. mes = &ELMessage{
  354. c1: c1,
  355. cd: make([]*Point, len(data)),
  356. }
  357. for i, d := range data {
  358. p := s.dTable[d]
  359. c2, err := s.pub.e2.Mul(r)
  360. if err != nil {
  361. return nil, err
  362. }
  363. cd, err := p.Add(c2)
  364. if err != nil {
  365. return nil, err
  366. }
  367. mes.cd[i] = cd
  368. }
  369. return mes, nil
  370. }
  371. // p = c2 – (d x c1)
  372. func (s *ELKeyPair) DecodeMessage(mes *ELMessage) ([]byte, error) {
  373. res := make([]byte, len(mes.cd))
  374. part, err := mes.c1.Mul(s.priv.d)
  375. if err != nil {
  376. return nil, err
  377. }
  378. part, _ = part.Neg()
  379. for _, c2 := range mes.cd {
  380. p, err := c2.Add(part)
  381. if err != nil {
  382. return nil, err
  383. }
  384. log.Println(p)
  385. }
  386. return res, nil
  387. }
  388. // https://intuit.ru/studies/professional_retraining/940/courses/408/lecture/9373?page=6
  389. func (s *Curve) ELKeyPair() (*ELKeyPair, error) {
  390. e1, _, err := s.searhClosePoints(Random(big.NewInt(1), Sub64(s.p, 1)))
  391. if err != nil {
  392. return nil, err
  393. }
  394. d := Random(big.NewInt(1), Sub64(s.n, 1))
  395. e2, err := e1.Mul(d)
  396. if err != nil {
  397. return nil, err
  398. }
  399. res := &ELKeyPair{
  400. curve: s,
  401. priv: &ELKeyPriv{
  402. d: d,
  403. },
  404. pub: &ELKeyPub{
  405. e1: e1,
  406. e2: e2,
  407. },
  408. }
  409. return res, res.setupDataTable()
  410. }
  411. /*
  412. func (s *Curve) HEncode(b byte, pub *Point) (cm1, cm2 *Point, err error) {
  413. var p1 *Point
  414. if p1, _, err = s.points(big.NewInt(int64(b))); err != nil {
  415. return
  416. }
  417. c1 = pub.Add()
  418. }
  419. */
  420. /*
  421. // 𝐶𝑚 = (𝑘 × 𝐺, 𝑃𝑚 + 𝑘 × 𝑃𝐵)
  422. func (s *Curve) HellmanEncode(b byte, pub *Point) (cm1, cm2 *Point, err error) {
  423. k := Random(big.NewInt(1), Sub64(s.n, 1))
  424. if cm1, err = s.g.Mul(k); err != nil {
  425. return
  426. }
  427. cm2, err = pub.Mul(Add64(k, int64(b)))
  428. return
  429. }
  430. */
  431. /*
  432. func (s *Curve) HellmanDecode(cm1, cm2 *Point, priv *big.Int) {
  433. }
  434. */