Skip to content

Commit 684e8a2

Browse files
authored
Merge pull request #21 from starkbank/feature/ecdsa-methods
Add toCompressed and fromCompressed method
2 parents 9736f3d + 37af8f5 commit 684e8a2

File tree

6 files changed

+119
-0
lines changed

6 files changed

+119
-0
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,10 @@ Given a version number MAJOR.MINOR.PATCH, increment:
1313

1414

1515
## [Unreleased]
16+
### Added
17+
- toCompressed and fromCompressed to PublicKey resource
18+
- modularSquareRoot to Math resource
19+
- Y to curve method
1620

1721
## [2.0.2] - 2022-10-27
1822
### Fixed

src/curve.php

+12
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,18 @@ function oidString()
5959
return join(".", $this->oid);
6060
}
6161

62+
function y($x, $isEven)
63+
{
64+
$ySquared = gmp_mod(gmp_add(gmp_add(gmp_powm($x, 3, $this->P), gmp_mul($this->A, $x)), $this->B), $this->P);
65+
$y = Math::modularSquareRoot($ySquared, $this->P);
66+
67+
if ($isEven != (gmp_intval(gmp_mod($y, 2)) == 0))
68+
{
69+
$y = gmp_sub($this->P, $y);
70+
}
71+
return $y;
72+
}
73+
6274
public static $supportedCurves = [];
6375
public static $_curvesByOid = [];
6476

src/math.php

+5
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,11 @@
77

88
class Math
99
{
10+
public static function modularSquareRoot($value, $prime)
11+
{
12+
return gmp_powm($value, gmp_div_q(gmp_add($prime, 1), 4), $prime);
13+
}
14+
1015
/**
1116
Fast way to multiply point and scalar in elliptic curves
1217

src/publickey.php

+35
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@ class PublicKey
1414
{
1515
private static $pemTemplate = "-----BEGIN PUBLIC KEY-----\n{content}\n-----END PUBLIC KEY-----\n";
1616
private static $ecdsaPublicKeyOid = array(1, 2, 840, 10045, 2, 1);
17+
private static $evenTag = "02";
18+
private static $oddTag = "03";
1719
public $point;
1820
public $curve;
1921

@@ -35,6 +37,21 @@ function toString($encoded=false)
3537
return $string;
3638
}
3739

40+
function toCompressed()
41+
{
42+
$baseLength = gmp_intval(2 * $this->curve->length());
43+
44+
if (gmp_intval($this->point->y % 2) == 0)
45+
{
46+
$parityTag = self::$evenTag;
47+
} else {
48+
$parityTag = self::$oddTag;
49+
}
50+
51+
$xHex = str_pad(Binary::hexFromInt($this->point->x), $baseLength, "0", STR_PAD_LEFT);
52+
return $parityTag . $xHex;
53+
}
54+
3855
function toDer()
3956
{
4057
$hexadecimal = Der::encodeConstructed(
@@ -99,4 +116,22 @@ static function fromString($string, $curve=null, $validatePoint=true)
99116
throw new Exception(sprintf("Point (%d,%d) * %s.N is not at infinity", $p->x, $p->y, $curve->name));
100117
return $publicKey;
101118
}
119+
120+
static function fromCompressed($string, $curve=null)
121+
{
122+
$curve = is_null($curve) ? Curve::$supportedCurves["secp256k1"] : $curve;
123+
$parityTag = substr($string, 0, 2);
124+
$xHex = substr($string, 2, strlen($string));
125+
126+
if (!in_array($parityTag, array(self::$evenTag, self::$oddTag)))
127+
{
128+
throw new Exception(sprintf("Compressed string should start with 02 or 03"));
129+
}
130+
131+
$x = Binary::intFromHex($xHex);
132+
$isEven = $parityTag === self::$evenTag;
133+
$y = $curve->y($x, $isEven);
134+
135+
return new PublicKey($point=new Point(Binary::intFromHex($x), Binary::intFromHex($y)), $curve=$curve);
136+
}
102137
}

tests/test.php

+1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ public function run() {
2424
}
2525

2626
include_once("testPublicKey.php");
27+
include_once("testCompPubKey.php");
2728
include_once("testEcdsa.php");
2829
include_once("testPrivateKey.php");
2930
include_once("testSignature.php");

tests/testCompPubKey.php

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
<?php
2+
3+
namespace EllipticCurve\Test;
4+
5+
use EllipticCurve\PublicKey;
6+
use EllipticCurve\Test\TestCase;
7+
8+
9+
echo "\n\nRunning Compressed Public Key Test tests:";
10+
\Test\printHeader("Compressed Public Key Test");
11+
12+
13+
class TestCompPubKey extends TestCase
14+
{
15+
public function testBatch()
16+
{
17+
for ($i = 1; $i <= 1000; $i++) {
18+
$privateKey = new \EllipticCurve\PrivateKey();
19+
$publicKey = $privateKey->publicKey();
20+
$publicKeyString = $publicKey->toCompressed();
21+
$recoveredPublicKey = \EllipticCurve\PublicKey::fromCompressed($publicKeyString, $publicKey->curve);
22+
\Test\assertEqual($publicKey->point->x, $recoveredPublicKey->point->x);
23+
\Test\assertEqual($publicKey->point->y, $recoveredPublicKey->point->y);
24+
}
25+
26+
}
27+
28+
public function testFromCompressedEven()
29+
{
30+
$publicKeyCompressed = "0252972572d465d016d4c501887b8df303eee3ed602c056b1eb09260dfa0da0ab2";
31+
$publicKey1 = \EllipticCurve\PublicKey::fromCompressed($publicKeyCompressed);
32+
$publicKey2 = \EllipticCurve\PublicKey::fromPem("\n-----BEGIN PUBLIC KEY-----\nMFYwEAYHKoZIzj0CAQYFK4EEAAoDQgAEUpclctRl0BbUxQGIe43zA+7j7WAsBWse\nsJJg36DaCrKIdC9NyX2e22/ZRrq8AC/fsG8myvEXuUBe15J1dj/bHA==\n-----END PUBLIC KEY-----\n");
33+
\Test\assertEqual($publicKey1->toPem(), $publicKey2->toPem());
34+
}
35+
36+
public function testFromCompressedOdd()
37+
{
38+
$publicKeyCompressed = "0318ed2e1ec629e2d3dae7be1103d4f911c24e0c80e70038f5eb5548245c475f50";
39+
$publicKey1 = \EllipticCurve\PublicKey::fromCompressed($publicKeyCompressed);
40+
$publicKey2 = \EllipticCurve\PublicKey::fromPem("\n-----BEGIN PUBLIC KEY-----\nMFYwEAYHKoZIzj0CAQYFK4EEAAoDQgAEGO0uHsYp4tPa574RA9T5EcJODIDnADj1\n61VIJFxHX1BMIg0B4cpBnLG6SzOTthXpndIKpr8HEHj3D9lJAI50EQ==\n-----END PUBLIC KEY-----\n");
41+
\Test\assertEqual($publicKey1->toPem(), $publicKey2->toPem());
42+
}
43+
44+
public function testToCompressedEven()
45+
{
46+
$publicKey = \EllipticCurve\PublicKey::fromPem("-----BEGIN PUBLIC KEY-----\nMFYwEAYHKoZIzj0CAQYFK4EEAAoDQgAEUpclctRl0BbUxQGIe43zA+7j7WAsBWse\nsJJg36DaCrKIdC9NyX2e22/ZRrq8AC/fsG8myvEXuUBe15J1dj/bHA==\n-----END PUBLIC KEY-----");
47+
$publicKeyCompressed = $publicKey->toCompressed();
48+
\Test\assertEqual($publicKeyCompressed, "0252972572d465d016d4c501887b8df303eee3ed602c056b1eb09260dfa0da0ab2");
49+
50+
}
51+
52+
public function testToCompressedOdd()
53+
{
54+
$publicKey = \EllipticCurve\PublicKey::fromPem("-----BEGIN PUBLIC KEY-----\nMFYwEAYHKoZIzj0CAQYFK4EEAAoDQgAEGO0uHsYp4tPa574RA9T5EcJODIDnADj1\n61VIJFxHX1BMIg0B4cpBnLG6SzOTthXpndIKpr8HEHj3D9lJAI50EQ==\n-----END PUBLIC KEY-----");
55+
$publicKeyCompressed = $publicKey->toCompressed();
56+
\Test\assertEqual($publicKeyCompressed, "0318ed2e1ec629e2d3dae7be1103d4f911c24e0c80e70038f5eb5548245c475f50");
57+
}
58+
}
59+
60+
61+
$tests = new TestCompPubKey();
62+
$tests->run();

0 commit comments

Comments
 (0)