Implementation of the MuSig system for n-of-n multisignatures.
Project description
MuSig
This is a simple-to-use implementation of the MuSig protocol.
Status
Implementation of the original MuSig protocol is complete. Implementation of MuSig2 is planned for a future release, as is an adaptor signature system.
Issues can be tracked here.
Installation
pip install musig
Usage
For details on the maths and safe use of the protocol, see docs/musig.md. The below examples should be sufficient to get started using this package. There is extensive type/value checking to enforce proper usage. If any bugs are encountered, please submit the bug report by opening an issue.
1-of-1 MuSig
For 1-of-1 MuSig, use the SingleSigKey class as shown in examples/1-of-1-musig.py. This is unlikely to be useful in practice, but it is hopefully helpful in making the process more understandable -- n-of-n MuSig requires a minimum of 3 communication rounds, so the 1-of-1 SingleSigKey exists to demonstrate the underlying maths in an uncomplicated way.
n-of-n MuSig
For n-of-n MuSig, where n>1, use the SigningSession class as shown in the file examples/2-of-2-musig.py. In it, we are doing 2-of-2 MuSig for simplicity, but the process works with any number of participants (though it has not been optimized for absurdly large numbers of participants).
Usage notes
The following classes have a public method that will return a copy with only
the values that are safe to share/distribute to others:
NoncePartialSignaturePublicKey
It is important to call public() on these instances before sharing because the
objects will otherwise serialize with sensitive/private information.
Another thing to note is that all objects except ProtocolError (an exception)
and ProtocolState (which is an enum) can be serialized by passing it as an
argument for str(), bytes(), or json.dumps(), and those outputs can be
deserialized by calling cls.from_str(), cls.from_bytes(), and
cls(json.loads()) respectively. ProtocolError can be serialized to and from
bytes and str, but it is not json compatible.
The final thing to note is that to make the objects serializable with the json
library, the classes inherit from the builtin dict class. To prevent misuse, the
__setitem__ method has been rewritten to allow only those keys that map to a
defined property to be set with a value. For maximum json compatibility, values
(and names of dict values) are serialized to base64 where possible/appropriate.
If you want to implement the abstract classes to build your own implementation
using the helper functions, or if you need some special functionality that the
included classes do not provide, it is notable that the __init__ method must
be overwritten for json deserialization to function properly.
Testing
Open a terminal in the root directory and run the following:
python -m unittest discover -s tests
There are currently 128 unit tests and 3 e2e tests. The 1-of-1 and 2-of-2 examples were adapted as e2e tests, and a third e2e test demonstrates n-of-n MuSig very concisely (omitting the message-passing simulation for brevity).
More Resources
Documentation generated by autodox can be found here, and old, manually created documentation can be found here.
Check out the Pycelium discord server. If you experience a problem, please discuss it on the Discord server. All suggestions for improvement are also welcome, and the best place for that is also Discord. If you experience a bug and do not use Discord, open an issue on Github.
(NB: I wrote this library in 2021 and only just now got around to cleaning it up enough to publish as a package. The code style and API are fairly idiosyncratic and will likely be updated substantially in the future after getting feedback and experience using it in practice.)
ISC License
Copyleft (c) 2021-2025, Jonathan Voss (k98kurz)
Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyleft notice and this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file musig-0.0.1.tar.gz.
File metadata
- Download URL: musig-0.0.1.tar.gz
- Upload date:
- Size: 44.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.10.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
46448a7c9353afdd870198845125f6d43ad08dd0a526a4194c30864b21db808d
|
|
| MD5 |
0ba882a7eaec2aa76e1d852d01f6606d
|
|
| BLAKE2b-256 |
a90265c2cb61d911ec0cb6e611be83702e2ea198ab683219a604be5f4a698cfa
|
File details
Details for the file musig-0.0.1-py3-none-any.whl.
File metadata
- Download URL: musig-0.0.1-py3-none-any.whl
- Upload date:
- Size: 26.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.10.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6050d1ab177c098cbe29c8d887a59f0c6cba9bc14b2842673a24ba45f5fa3cd8
|
|
| MD5 |
c0dedf0672be96af00709cf7a63526d4
|
|
| BLAKE2b-256 |
58b440f184436ecb4ad47dfc602f26879eaa7552bbe6b675c17d69179b15f719
|