2020-06-26 13:48:50 +02:00
package fr.acinq.secp256k1
2020-07-09 20:16:39 +02:00
import kotlin.random.Random
2020-07-02 17:52:21 +02:00
import kotlin.test.*
2020-06-26 13:48:50 +02:00
class Secp256k1Test {
@Test
2021-10-26 17:16:36 +02:00
fun verifyValidPrivateKey ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
assertTrue ( Secp256k1 . secKeyVerify ( priv ) )
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun verifyInvalidPrivateKey ( ) {
val greaterThanCurveOrder = Hex . decode ( " FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF " . lowercase ( ) )
assertFalse ( Secp256k1 . secKeyVerify ( greaterThanCurveOrder ) )
val zero = Hex . decode ( " 0000000000000000000000000000000000000000000000000000000000000000 " . lowercase ( ) )
assertFalse ( Secp256k1 . secKeyVerify ( zero ) )
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun createValidPublicKey ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val pub = Secp256k1 . pubkeyCreate ( priv )
2020-06-26 13:48:50 +02:00
assertEquals (
" 04C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D2103ED494718C697AC9AEBCFD19612E224DB46661011863ED2FC54E71861E2A6 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( pub ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun createInvalidPublicKey ( ) {
val priv = Hex . decode ( " FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF " . lowercase ( ) )
2020-07-02 17:52:21 +02:00
assertFailsWith < Secp256k1Exception > {
2021-10-26 17:16:36 +02:00
Secp256k1 . pubkeyCreate ( priv )
2020-07-02 17:52:21 +02:00
}
}
@Test
2021-10-26 17:16:36 +02:00
fun compressPublicKey ( ) {
2020-07-02 17:52:21 +02:00
val pub = Hex . decode ( " 04C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D2103ED494718C697AC9AEBCFD19612E224DB46661011863ED2FC54E71861E2A6 " )
val compressed = Secp256k1 . pubKeyCompress ( pub )
2021-09-22 15:22:05 +02:00
assertEquals ( " 02C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D " , Hex . encode ( compressed ) . uppercase ( ) )
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun negatePublicKey ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val pub = Secp256k1 . pubkeyCreate ( priv )
2020-06-26 13:48:50 +02:00
assertEquals (
" 04C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D2103ED494718C697AC9AEBCFD19612E224DB46661011863ED2FC54E71861E2A6 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( pub ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
2021-10-26 17:16:36 +02:00
val npub = Secp256k1 . pubKeyNegate ( pub )
2020-06-26 13:48:50 +02:00
assertEquals (
" 04C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2DDEFC12B6B8E73968536514302E69ED1DDB24B999EFEE79C12D03AB17E79E1989 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( npub ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun parsePublicKey ( ) {
val pub = Hex . decode ( " 02C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D " . lowercase ( ) )
val parsed = Secp256k1 . pubkeyParse ( pub )
2020-06-26 13:48:50 +02:00
assertEquals (
" 04C591A8FF19AC9C4E4E5793673B83123437E975285E7B442F4EE2654DFFCA5E2D2103ED494718C697AC9AEBCFD19612E224DB46661011863ED2FC54E71861E2A6 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( parsed ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun parseInvalidPublicKey ( ) {
val pub = Hex . decode ( " 02FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF " . lowercase ( ) )
assertFailsWith < Secp256k1Exception > {
Secp256k1 . pubkeyParse ( pub )
}
}
@Test
fun combinePublicKeys ( ) {
val pub1 = Hex . decode ( " 041b84c5567b126440995d3ed5aaba0565d71e1834604819ff9c17f5e9d5dd078f70beaf8f588b541507fed6a642c5ab42dfdf8120a7f639de5122d47a69a8e8d1 " . lowercase ( ) )
val pub2 = Hex . decode ( " 044d4b6cd1361032ca9bd2aeb9d900aa4d45d9ead80ac9423374c451a7254d07662a3eada2d0fe208b6d257ceb0f064284662e857f57b66b54c198bd310ded36d0 " . lowercase ( ) )
val pub3 = Secp256k1 . pubKeyCombine ( arrayOf ( pub1 , pub2 ) )
2020-06-26 13:48:50 +02:00
assertEquals (
" 04531FE6068134503D2723133227C867AC8FA6C83C537E9A44C3C5BDBDCB1FE3379E92C265E71E481BA82A84675A47AC705A200FCD524E92D93B0E7386F26A5458 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( pub3 ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun createEcdsaSignature ( ) {
val message = Hex . decode ( " CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90 " . lowercase ( ) ) //sha256hash of "testing"
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val sig = Secp256k1 . sign ( message , priv )
2020-06-26 13:48:50 +02:00
assertEquals (
2020-07-02 17:52:21 +02:00
" 182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A21C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( sig ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
2020-06-29 17:10:58 +02:00
@Test
2021-10-26 17:16:36 +02:00
fun normalizeEcdsaSignature ( ) {
val sig = Hex . decode ( " 30440220182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A202201C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9 " . lowercase ( ) )
val ( normalized , wasNotNormalized ) = Secp256k1 . signatureNormalize ( sig )
assertFalse ( wasNotNormalized )
2020-06-29 17:10:58 +02:00
assertEquals (
" 182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A21C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( normalized ) . uppercase ( ) ,
2020-06-29 17:10:58 +02:00
)
}
2020-06-26 13:48:50 +02:00
@Test
2021-10-26 17:16:36 +02:00
fun failToCreateEcdsaSignature ( ) {
val message = Hex . decode ( " CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90 " . lowercase ( ) ) //sha256hash of "testing"
val priv = Hex . decode ( " FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF " . lowercase ( ) )
2020-07-02 17:52:21 +02:00
assertFailsWith < Secp256k1Exception > {
2021-10-26 17:16:36 +02:00
Secp256k1 . sign ( message , priv )
2020-07-02 17:52:21 +02:00
}
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun createCompactEcdsaSignature ( ) {
val message = Hex . decode ( " CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90 " . lowercase ( ) ) //sha256hash of "testing"
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val sig = Secp256k1 . sign ( message , priv )
2020-06-26 13:48:50 +02:00
assertEquals (
" 182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A21C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( sig ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun verifyValidEcdsaSignatures ( ) {
val message = Hex . decode ( " CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90 " . lowercase ( ) ) //sha256hash of "testing"
val sig = Hex . decode ( " 3044022079BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F817980220294F14E883B3F525B5367756C2A11EF6CF84B730B36C17CB0C56F0AAB2C98589 " . lowercase ( ) )
val pub = Hex . decode ( " 040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40 " . lowercase ( ) )
assertTrue ( Secp256k1 . verify ( sig , message , pub ) )
val sigCompact = Hex . decode ( " 79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798294F14E883B3F525B5367756C2A11EF6CF84B730B36C17CB0C56F0AAB2C98589 " . lowercase ( ) )
assertTrue ( Secp256k1 . verify ( sigCompact , message , pub ) )
}
@Test
fun verifyInvalidEcdsaSignatures ( ) {
val message = Hex . decode ( " CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A91 " . lowercase ( ) ) //sha256hash of "testing"
val sig = Hex . decode ( " 3044022079BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F817980220294F14E883B3F525B5367756C2A11EF6CF84B730B36C17CB0C56F0AAB2C98589 " . lowercase ( ) )
val pub = Hex . decode ( " 040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40 " . lowercase ( ) )
assertFalse ( Secp256k1 . verify ( sig , message , pub ) )
}
@Test
fun negatePrivateKey ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val npriv = Secp256k1 . privKeyNegate ( priv )
2020-06-26 13:48:50 +02:00
assertEquals (
" 981A9A7DD677A622518DA068D66D5F824E5F22F084B8A0E2F195B5662F300C11 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( npriv ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
2021-10-26 17:16:36 +02:00
val nnpriv : ByteArray = Secp256k1 . privKeyNegate ( npriv )
assertContentEquals ( priv , nnpriv )
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun addTweakToPrivateKey ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val tweak = Hex . decode ( " 3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3 " . lowercase ( ) )
val tweakedPriv = Secp256k1 . privKeyTweakAdd ( priv , tweak )
2020-06-29 17:10:58 +02:00
assertEquals (
" A168571E189E6F9A7E2D657A4B53AE99B909F7E712D1C23CED28093CD57C88F3 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( tweakedPriv ) . uppercase ( ) ,
2020-06-29 17:10:58 +02:00
)
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun multiplyPrivateKeyWithTweak ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val tweak = Hex . decode ( " 3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3 " . lowercase ( ) )
val tweakedPriv = Secp256k1 . privKeyTweakMul ( priv , tweak )
2020-06-29 17:10:58 +02:00
assertEquals (
" 97F8184235F101550F3C71C927507651BD3F1CDB4A5A33B8986ACF0DEE20FFFC " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( tweakedPriv ) . uppercase ( ) ,
2020-06-29 17:10:58 +02:00
)
2020-06-26 13:48:50 +02:00
}
@Test
2021-10-26 17:16:36 +02:00
fun addTweakToPublicKey ( ) {
val pub = Hex . decode ( " 040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40 " . lowercase ( ) )
val tweak = Hex . decode ( " 3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3 " . lowercase ( ) )
val tweakedPub = Secp256k1 . pubKeyTweakAdd ( pub , tweak )
2020-06-26 13:48:50 +02:00
assertEquals (
" 0411C6790F4B663CCE607BAAE08C43557EDC1A4D11D88DFCB3D841D0C6A941AF525A268E2A863C148555C48FB5FBA368E88718A46E205FABC3DBA2CCFFAB0796EF " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( tweakedPub ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun multiplyPublicKeyWithTweak ( ) {
val pub = Hex . decode ( " 040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40 " . lowercase ( ) )
val tweak = Hex . decode ( " 3982F19BEF1615BCCFBB05E321C10E1D4CBA3DF0E841C2E41EEB6016347653C3 " . lowercase ( ) )
val tweakedPub = Secp256k1 . pubKeyTweakMul ( pub , tweak )
2020-06-26 13:48:50 +02:00
assertEquals (
" 04E0FE6FE55EBCA626B98A807F6CAF654139E14E5E3698F01A9A658E21DC1D2791EC060D4F412A794D5370F672BC94B722640B5F76914151CFCA6E712CA48CC589 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( tweakedPub ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun createEcdhSecret ( ) {
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val pub = Hex . decode ( " 040A629506E1B65CD9D2E0BA9C75DF9C4FED0DB16DC9625ED14397F0AFC836FAE595DC53F8B0EFE61E703075BD9B143BAC75EC0E19F82A2208CAEB32BE53414C40 " . lowercase ( ) )
val secret = Secp256k1 . ecdh ( priv , pub )
2020-06-26 13:48:50 +02:00
assertEquals (
" 2A2A67007A926E6594AF3EB564FC74005B37A9C8AEF2033C4552051B5C87F043 " ,
2021-10-26 17:16:36 +02:00
Hex . encode ( secret ) . uppercase ( ) ,
2020-06-26 13:48:50 +02:00
)
}
@Test
2021-10-26 17:16:36 +02:00
fun createSymmetricEcdhSecret ( ) {
val priv1 = Hex . decode ( " 3580a881ac24eb00530a51235c42bcb65424ba121e2e7d910a70fa531a578d21 " )
val pub1 = Secp256k1 . pubkeyCreate ( priv1 )
val priv2 = Hex . decode ( " f6a353f7a5de654501c3495acde7450293f74d09086c2b7c9a4e524248d0daac " )
val pub2 = Secp256k1 . pubkeyCreate ( priv2 )
val secret1 = Secp256k1 . ecdh ( priv1 , pub2 )
val secret2 = Secp256k1 . ecdh ( priv2 , pub1 )
assertContentEquals ( secret1 , secret2 )
2020-06-26 13:48:50 +02:00
}
2020-07-02 21:39:33 +02:00
@Test
2021-10-26 17:16:36 +02:00
fun recoverPublicKeyFromEcdsaSignature ( ) {
val message = Hex . decode ( " CF80CD8AED482D5D1527D7DC72FCEFF84E6326592848447D2DC0B0E87DFC9A90 " . lowercase ( ) ) //sha256hash of "testing"
val priv = Hex . decode ( " 67E56582298859DDAE725F972992A07C6C4FB9F62A8FFF58CE3CA926A1063530 " . lowercase ( ) )
val pub = Secp256k1 . pubkeyCreate ( priv )
val sig = Secp256k1 . sign ( message , priv )
val pub0 = Secp256k1 . ecdsaRecover ( sig , message , 0 )
val pub1 = Secp256k1 . ecdsaRecover ( sig , message , 1 )
assertTrue ( pub . contentEquals ( pub0 ) || pub . contentEquals ( pub1 ) )
}
@Test
fun convertCompactEcdsaSignatureToDer ( ) {
val compact = Hex . decode ( " 182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A21C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9 " . lowercase ( ) ) //sha256hash of "testing"
val der = Secp256k1 . compact2der ( compact )
2020-07-02 21:39:33 +02:00
assertEquals (
" 30440220182A108E1448DC8F1FB467D06A0F3BB8EA0533584CB954EF8DA112F1D60E39A202201C66F36DA211C087F3AF88B50EDF4F9BDAA6CF5FD6817E74DCA34DB12390C6E9 " ,
2021-09-22 15:22:05 +02:00
Hex . encode ( der ) . uppercase ( ) ,
2020-07-02 21:39:33 +02:00
)
}
2020-07-09 20:16:39 +02:00
@Test
2021-10-26 17:16:36 +02:00
fun fuzzEcdsaSignVerify ( ) {
2020-07-09 20:16:39 +02:00
val random = Random . Default
fun randomBytes ( length : Int ) : ByteArray {
val buffer = ByteArray ( length )
random . nextBytes ( buffer )
return buffer
}
repeat ( 200 ) {
val priv = randomBytes ( 32 )
2021-10-26 17:16:36 +02:00
assertTrue ( Secp256k1 . secKeyVerify ( priv ) )
2020-07-09 20:16:39 +02:00
val pub = Secp256k1 . pubkeyCreate ( priv )
2021-10-26 17:16:36 +02:00
val message = randomBytes ( 32 )
val sig = Secp256k1 . sign ( message , priv )
assertTrue ( Secp256k1 . verify ( sig , message , pub ) )
2020-07-09 20:16:39 +02:00
val der = Secp256k1 . compact2der ( sig )
2021-10-26 17:16:36 +02:00
assertTrue ( Secp256k1 . verify ( der , message , pub ) )
2020-07-09 20:16:39 +02:00
}
}
2021-10-26 17:16:36 +02:00
2020-06-29 17:10:58 +02:00
}