I’m making a program in Python to be distributed to windows users via an installer.
The program needs to be able to download a file every day encrypted with the user’s public key and then decrypt it.
So I need to find a Python library that will let me generate public and private PGP keys, and also decrypt files encrypted with the public key.
Is this something pyCrypto will do (documentation is nebulous)? Are there other pure Python libraries? How about a standalone command line tool in any language?
All I saw so far was GNUPG but installing that on Windows does stuff to the registry and throws dll’s everywhere, and then I have to worry about whether the user already has this installed, how to backup their existing keyrings, etc. I’d rather just have a python library or command line tool and mange the keys myself.
Update: pyME might work but it doesn’t seem to be compatible with Python 2.4 which I have to use.
You don’t need
PyMe, fine though those packages may be – you will have all kinds of problems building under Windows. Instead, why not avoid the rabbit-holes and do what I did? Use
gnupg 1.4.9. You don’t need to do a full installation on end-user machines – just
iconv.dll from the distribution are sufficient, and you just need to have them somewhere in the path or accessed from your Python code using a full pathname. No changes to the registry are needed, and everything (executables and data files) can be confined to a single folder if you want.
There’s a module
GPG.py which was originally written by Andrew Kuchling, improved by Richard Jones and improved further by Steve Traugott. It’s available here, but as-is it’s not suitable for Windows because it uses
os.fork(). Although originally part of
PyCrypto, it is completely independent of the other parts of
PyCrypto and needs only gpg.exe/iconv.dll in order to work.
I have a version (
gnupg.py) derived from Traugott’s
GPG.py, which uses the
subprocess module. It works fine under Windows, at least for my purposes – I use it to do the following:
- Key management – generation, listing, export etc.
- Import keys from an external source (e.g. public keys received from a partner company)
- Encrypt and decrypt data
- Sign and verify signatures
The module I’ve got is not ideal to show right now, because it includes some other stuff which shouldn’t be there – which means I can’t release it as-is at the moment. At some point, perhaps in the next couple of weeks, I hope to be able to tidy it up, add some more unit tests (I don’t have any unit tests for sign/verify, for example) and release it (either under the original
PyCrypto licence or a similar commercial-friendly license). If you can’t wait, go with Traugott’s module and modify it yourself – it wasn’t too much work to make it work with the
This approach was a lot less painful than the others (e.g.
SWIG-based solutions, or solutions which require building with
MSYS), which I considered and experimented with. I’ve used the same (
iconv.dll) approach with systems written in other languages, e.g.
C#, with equally painless results.
P.S. It works with Python 2.4 as well as Python 2.5 and later. Not tested with other versions, though I don’t foresee any problems.
After a LOT of digging, I found a package that worked for me. Although it is said to support the generation of keys, I didn’t test it. However I did manage to decrypt a message that was encrypted using a GPG public key. The advantage of this package is that it does not require a GPG executable file on the machine, and is a Python based implementation of the OpenPGP (rather than a wrapper around the executable).
I created the private and public keys using GPG4win and kleopatra for windows
See my code below.
import pgpy emsg = pgpy.PGPMessage.from_file(<path to the file from the client that was encrypted using your public key>) key,_ = pgpy.PGPKey.from_file(<path to your private key>) with key.unlock(<your private key passpharase>): print (key.decrypt(emsg).message)
Although the question is very old. I hope this helps future users.
PyCrypto supports PGP – albeit you should test it to make sure that it works to your specifications.
Although documentation is hard to come by, if you look through Util/test.py (the module test script), you can find a rudimentary example of their PGP support:
if verbose: print ' PGP mode:', obj1=ciph.new(password, ciph.MODE_PGP, IV) obj2=ciph.new(password, ciph.MODE_PGP, IV) start=time.time() ciphertext=obj1.encrypt(str) plaintext=obj2.decrypt(ciphertext) end=time.time() if (plaintext!=str): die('Error in resulting plaintext from PGP mode') print_timing(256, end-start, verbose) del obj1, obj2
Futhermore, PublicKey/pubkey.py provides for the following relevant methods:
def encrypt(self, plaintext, K) def decrypt(self, ciphertext): def sign(self, M, K): def verify (self, M, signature): def can_sign (self): """can_sign() : bool Return a Boolean value recording whether this algorithm can generate signatures. (This does not imply that this particular key object has the private information required to to generate a signature.) """ return 1
PyMe does claim full compatibility with Python 2.4, and I quote:
The latest version of PyMe (as of this
writing) is v0.8.0. Its binary
distribution for Debian was compiled
with SWIG v1.3.33 and GCC v4.2.3 for
GPGME v1.1.6 and Python v2.3.5,
v2.4.4, and v2.5.2 (provided in
‘unstable’ distribution at the time).
Its binary distribution for Windows
was compiled with SWIG v1.3.29 and
MinGW v4.1 for GPGME v1.1.6 and Python
v2.5.2 (although the same binary get
installed and works fine in v2.4.2 as
I’m not sure why you say “it doesn’t seem to be compatible with Python 2.4 which I have to use” — specifics please?
And yes it does exist as a semi-Pythonic (SWIGd) wrapper on GPGME — that’s a popular way to develop Python extensions once you have a C library that basically does the job.
PyPgp has a much simpler approach — that’s why it’s a single, simple Python script: basically it does nothing more than “shell out” to command-line PGP commands. For example, decryption is just:
def decrypt(data): "Decrypt a string - if you have the right key." pw,pr = os.popen2('pgpv -f') pw.write(data) pw.close() ptext = pr.read() return ptext
i.e., write the encrypted cyphertext to the standard input of
pgpv -f, read pgpv’s standard output as the decrypted plaintext.
PyPgp is also a very old project, though its simplicity means that making it work with modern Python (e.g., subprocess instead of now-deprecated os.popen2) would not be hard. But you still do need PGP installed, or PyPgp won’t do anything;-).
M2Crypto has PGP module, but I have actually never tried to use it. If you try it, and it works, please let me know (I am the current M2Crypto maintainer). Some links:
Update: The PGP module does not provide ways to generate keys, but presumably these could be created with the lower level RSA, DSA etc. modules. I don’t know PGP insides, so you’d have to dig up the details. Also, if you know how to generate these using openssl command line commands, it should be reasonably easy to convert that to M2Crypto calls.
As other have noted, PyMe is the canonical solution for this, since it’s based on GpgME, which is part of the GnuPG ecosystem.
For Windows, I strongly recommend to use Gpg4win as the GnuPG distribution, for two reasons:
It’s based on GnuPG 2, which, among other things, includes
gpg2.exe, which can (finally, I might add 🙂 start
gpg-agent.exe on-demand (gpg v1.x can’t).
And secondly, it’s the only official Windows build by the GnuPG developers. E.g. it’s entirely cross-compiled from Linux to Windows, so not a iota of non-free software was used in preparing it (quite important for a security suite :).
To sign with only the exported public key file without a keyring.
With PGPy 0.5.2 (pure Python GPG RFC implementation):
key_fpath = './recipient-PUB.gpg' rsa_pub, _ = pgpy.PGPKey.from_file(rkey_fpath) rkey = rsa_pub.subkeys.values() text_message = pgpy.PGPMessage.new('my msg') encrypted_message = rkey.encrypt(text_message) print encrypted_message.__bytes__()
With gpg 1.10.0 (gpgme Python bindings – former PyME):
rkey_fpath = './recipient-PUB.gpg' cg = gpg.Context() rkey = list(cg.keylist(source = rkey_fpath)) ciphertext, result, sign_result = cg.encrypt('my msg', recipients=rkey, sign=False, always_trust=True) print ciphertext
A simple benchmark in a for loop shows me that for this simple operation on my system PGPy is ~3x time faster than gpgme Python bindings (please do not take this statement as X is faster than Y: I will invite you to test in your environment).