Author

Topic: secp256k1 signrec - sha256 to final signature ? (Read 162 times)

legendary
Activity: 2464
Merit: 4419
🔐BitcoinMessage.Tools🔑
January 09, 2023, 07:28:57 AM
#5
I am trying to understand what exactly is done in this step but I have no insight. I see that the called functions are derived from _libsecp256k1.
Code:
from ._libsecp256k1 import ffi, lib

I looked into the filesystem of that python module secp256k1, there is the file:
Quote
~/.local/lib/python3.10/site-packages/secp256k1/_libsecp256k1.cpython-310-x86_64-linux-gnu.so

but I have no clue what's inside. Can anyone explain to me, please?
Although I am not an expert and have little understanding of C language, but I think I can tell what is inside these odd-looking function calls: all these functions are present here https://github.com/bitcoin-core/secp256k1/blob/master/src/secp256k1.c I think every time the python code uses ffi (which stands for Foreign function interface it delegates to functions written in C, sends them some arguments and receives a result. So, to figure out what happens under the hood, you need to know how signature signing/verification process is implemented in C language.
legendary
Activity: 3472
Merit: 10611
In secp256 library the message is hashed once into the variable msg32, see code in __init__.py here:
I'm not the best at reading python but isn't "msg32 = digest(msg).digest()" inside the "_hash32" method performing double hash? The python docs seems to suggest each "digest()" method call performs a single hash and it is called twice here.

When you debug output msg32 at this step you will see that the result is
You probably broke something when debugging because the signature you posted is valid and it only verifies against the hash I posted above not the one you get from your debug.
hero member
Activity: 630
Merit: 731
Bitcoin g33k
thanks for the link provided.


That means the digest of "Hello, world!" is not 0x315f... but
Code:
02d6c0643e40b0db549cbbd7eb47dcab71a59d7017199ebde6b272f28fbbf95f

In secp256 library the message is hashed once into the variable msg32, see code in __init__.py here:
Code:
[...]
    def ecdsa_sign_recoverable(self, msg, raw=False, digest=hashlib.sha256):
        if not HAS_RECOVERABLE:
            raise Exception("secp256k1_recovery not enabled")

        msg32 = _hash32(msg, raw, digest)
[...]
When you debug output msg32 at this step you will see that the result is
Quote
315f5bdb76d078c43b8ac0064e4a0164612b1fce77c869345bfc94c75894edd3

I am just trying to follow the steps done in this library secp256k1. Honestly said, I still don't understand that. Using the given example

Quote
privkey = '0000000000000000000000000000000000000000000000000000000000000001'
address = '1BgGZ9tcN4rm9KBzDn7KprQz87SZ26SAMH'
message = 'Hello, world!'
signature (hex?) = 'b85d62928d63583f52b14995c9444a92e1b7998a3fcfd0c134f327d61b162c6e7ea40adb783bd4c 00f9cfdb829c7e7d5b8d8e25a797d8548aec6f41df461fab9'
signature (base64?) = 'IEkjQHms3Yy0+B8INBVgKozpZc1rf3OHf7MCk2CnrGorYk2TEnwnNSHnLuK8tRkBIAIR1c9i8NCO19EebEHCMak='
pubkey = '0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798'

can anyone show a step-by-step (at best with Python code) how to get to the final base64-encoded form and vice-versa ?
legendary
Activity: 3472
Merit: 10611
When signing messages in order to prevent possible exploits, some additional bytes are added to the beginning of the message before it is hashed twice using SHA256.
- The bytes are the length of the fixed message
- The fixed message ("Bitcoin Signed Message:\n")
- The length of the message
- The message

That means the digest of "Hello, world!" is not 0x315f... but
Code:
02d6c0643e40b0db549cbbd7eb47dcab71a59d7017199ebde6b272f28fbbf95f

As for signing, there is nothing special. It is using the same ECDSA algorithm as always, but also returns a single byte that helps the verifier recover only one public from the signature when they are verifying the whole thing.
When encoding that signature it uses the following structure [1-byte-recid][32-byte-r][32-byte-s]

BIP-137 has more explanation: https://github.com/bitcoin/bips/blob/master/bip-0137.mediawiki

Quote
How do you convert this base64 to get to the hex representation and vice-versa ?
Usually all programming languages have some sort of converter that does Base64 conversion, if not finding a library that does base conversion in the programming language you are working with is not hard. For example in C# it is System.Convert.FromBase64String()
Same with Base16 (or hex).
hero member
Activity: 630
Merit: 731
Bitcoin g33k
Hello all. I am referring to secp256k1 library in Python where you can signrec a message.
Following example:
Quote
privkey = '0000000000000000000000000000000000000000000000000000000000000001'
message = 'Hello, world!'

after signing it will produce the signature
Quote
b85d62928d63583f52b14995c9444a92e1b7998a3fcfd0c134f327d61b162c6e7ea40adb783bd4c 00f9cfdb829c7e7d5b8d8e25a797d8548aec6f41df461fab9

I am digging into the code and like to understand what the process looks in detail. I am stuck on the last point and hopefully someone can shed some light onto...

The signrec command line argument calls the function ecdsa_sign_recoverable
Code:
[...]
    elif args.action == 'signrec':
        priv, sig = sign('ecdsa_sign_recoverable', args)
[...]

the message is hashed one time through sha256 and stored into the variable name msg32. The sha256 from the message shown is
Quote
315f5bdb76d078c43b8ac0064e4a0164612b1fce77c869345bfc94c75894edd3

Code:
[...]
    def ecdsa_sign_recoverable(self, msg, raw=False, digest=hashlib.sha256):
        if not HAS_RECOVERABLE:
            raise Exception("secp256k1_recovery not enabled")

        msg32 = _hash32(msg, raw, digest)
[...]

So far so good and understood. But on the next step the raw_sig variable is created by calling ffi.new('secp256k1_ecdsa_recoverable_signature *')

Code:
[...]
        raw_sig = ffi.new('secp256k1_ecdsa_recoverable_signature *')

        signed = lib.secp256k1_ecdsa_sign_recoverable(
            secp256k1_ctx, raw_sig, msg32, self.private_key,
            ffi.NULL, ffi.NULL)
        assert signed == 1

        return raw_sig
[...]

I am trying to understand what exactly is done in this step but I have no insight. I see that the called functions are derived from _libsecp256k1.
Code:
from ._libsecp256k1 import ffi, lib

I looked into the filesystem of that python module secp256k1, there is the file:
Quote
~/.local/lib/python3.10/site-packages/secp256k1/_libsecp256k1.cpython-310-x86_64-linux-gnu.so

but I have no clue what's inside. Can anyone explain to me, please?
What process in detail is done to get from the sha256 hash "315f5bdb76d078c43b8ac0064e4a0164612b1fce77c869345bfc94c75894edd3"
--> to the final signature ?

Quote
b85d62928d63583f52b14995c9444a92e1b7998a3fcfd0c134f327d61b162c6e7ea40adb783bd4c 00f9cfdb829c7e7d5b8d8e25a797d8548aec6f41df461fab9

My next question ... If I'm not mistaken the shown final signature in base64 representation is:
Quote
IEkjQHms3Yy0+B8INBVgKozpZc1rf3OHf7MCk2CnrGorYk2TEnwnNSHnLuK8tRkBIAIR1c9i8NCO19EebEHCMak=

How do you convert this base64 to get to the hex representation and vice-versa ?

Thanks to all.
Jump to: